Allocate user-data-array on the heap
This saves each object allocation 72 bytes. Now object overhead is just 16 bytes (on x86) instead of 88 bytes. Neat.
This commit is contained in:
parent
61920b21ca
commit
16e4ccf7b4
|
@ -62,7 +62,6 @@ struct hb_reference_count_t
|
||||||
|
|
||||||
/* user_data */
|
/* user_data */
|
||||||
|
|
||||||
#define HB_USER_DATA_ARRAY_INIT {HB_MUTEX_INIT, HB_LOCKABLE_SET_INIT}
|
|
||||||
struct hb_user_data_array_t
|
struct hb_user_data_array_t
|
||||||
{
|
{
|
||||||
struct hb_user_data_item_t {
|
struct hb_user_data_item_t {
|
||||||
|
@ -97,9 +96,9 @@ struct hb_user_data_array_t
|
||||||
struct hb_object_header_t
|
struct hb_object_header_t
|
||||||
{
|
{
|
||||||
hb_reference_count_t ref_count;
|
hb_reference_count_t ref_count;
|
||||||
hb_user_data_array_t user_data;
|
hb_user_data_array_t *user_data;
|
||||||
|
|
||||||
#define HB_OBJECT_HEADER_STATIC {HB_REFERENCE_COUNT_INIT, HB_USER_DATA_ARRAY_INIT}
|
#define HB_OBJECT_HEADER_STATIC {HB_REFERENCE_COUNT_INIT, nullptr}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ASSERT_POD ();
|
ASSERT_POD ();
|
||||||
|
@ -133,7 +132,7 @@ template <typename Type>
|
||||||
static inline void hb_object_init (Type *obj)
|
static inline void hb_object_init (Type *obj)
|
||||||
{
|
{
|
||||||
obj->header.ref_count.init (1);
|
obj->header.ref_count.init (1);
|
||||||
obj->header.user_data.init ();
|
obj->header.user_data = nullptr;
|
||||||
}
|
}
|
||||||
template <typename Type>
|
template <typename Type>
|
||||||
static inline bool hb_object_is_inert (const Type *obj)
|
static inline bool hb_object_is_inert (const Type *obj)
|
||||||
|
@ -172,7 +171,11 @@ template <typename Type>
|
||||||
static inline void hb_object_fini (Type *obj)
|
static inline void hb_object_fini (Type *obj)
|
||||||
{
|
{
|
||||||
obj->header.ref_count.fini (); /* Do this before user_data */
|
obj->header.ref_count.fini (); /* Do this before user_data */
|
||||||
obj->header.user_data.fini ();
|
if (obj->header.user_data)
|
||||||
|
{
|
||||||
|
obj->header.user_data->fini ();
|
||||||
|
free (obj->header.user_data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
template <typename Type>
|
template <typename Type>
|
||||||
static inline bool hb_object_set_user_data (Type *obj,
|
static inline bool hb_object_set_user_data (Type *obj,
|
||||||
|
@ -184,17 +187,34 @@ static inline bool hb_object_set_user_data (Type *obj,
|
||||||
if (unlikely (!obj || hb_object_is_inert (obj)))
|
if (unlikely (!obj || hb_object_is_inert (obj)))
|
||||||
return false;
|
return false;
|
||||||
assert (hb_object_is_valid (obj));
|
assert (hb_object_is_valid (obj));
|
||||||
return obj->header.user_data.set (key, data, destroy, replace);
|
|
||||||
|
retry:
|
||||||
|
hb_user_data_array_t *user_data = (hb_user_data_array_t *) hb_atomic_ptr_get (&obj->header.user_data);
|
||||||
|
if (unlikely (!user_data))
|
||||||
|
{
|
||||||
|
user_data = (hb_user_data_array_t *) calloc (sizeof (hb_user_data_array_t), 1);
|
||||||
|
if (unlikely (!user_data))
|
||||||
|
return false;
|
||||||
|
user_data->init ();
|
||||||
|
if (unlikely (!hb_atomic_ptr_cmpexch (&obj->header.user_data, nullptr, user_data)))
|
||||||
|
{
|
||||||
|
user_data->fini ();
|
||||||
|
free (user_data);
|
||||||
|
goto retry;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return user_data->set (key, data, destroy, replace);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Type>
|
template <typename Type>
|
||||||
static inline void *hb_object_get_user_data (Type *obj,
|
static inline void *hb_object_get_user_data (Type *obj,
|
||||||
hb_user_data_key_t *key)
|
hb_user_data_key_t *key)
|
||||||
{
|
{
|
||||||
if (unlikely (!obj || hb_object_is_inert (obj)))
|
if (unlikely (!obj || hb_object_is_inert (obj) || !obj->header.user_data))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
assert (hb_object_is_valid (obj));
|
assert (hb_object_is_valid (obj));
|
||||||
return obj->header.user_data.get (key);
|
return obj->header.user_data->get (key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue