Behdad Esfahbod | f0954d1 | 2009-07-30 15:33:57 -0400 | [diff] [blame] | 1 | /* |
Behdad Esfahbod | 2409d5f | 2011-04-21 17:14:28 -0400 | [diff] [blame] | 2 | * Copyright © 2007 Chris Wilson |
| 3 | * Copyright © 2009,2010 Red Hat, Inc. |
| 4 | * Copyright © 2011 Google, Inc. |
Behdad Esfahbod | f0954d1 | 2009-07-30 15:33:57 -0400 | [diff] [blame] | 5 | * |
Behdad Esfahbod | c755cb3 | 2010-04-22 00:11:43 -0400 | [diff] [blame] | 6 | * This is part of HarfBuzz, a text shaping library. |
Behdad Esfahbod | f0954d1 | 2009-07-30 15:33:57 -0400 | [diff] [blame] | 7 | * |
| 8 | * Permission is hereby granted, without written agreement and without |
| 9 | * license or royalty fees, to use, copy, modify, and distribute this |
| 10 | * software and its documentation for any purpose, provided that the |
| 11 | * above copyright notice and the following two paragraphs appear in |
| 12 | * all copies of this software. |
| 13 | * |
| 14 | * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR |
| 15 | * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES |
| 16 | * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN |
| 17 | * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH |
| 18 | * DAMAGE. |
| 19 | * |
| 20 | * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, |
| 21 | * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND |
| 22 | * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS |
| 23 | * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO |
| 24 | * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. |
| 25 | * |
| 26 | * Contributor(s): |
| 27 | * Chris Wilson <chris@chris-wilson.co.uk> |
Behdad Esfahbod | 35a7383 | 2009-08-01 19:30:31 -0400 | [diff] [blame] | 28 | * Red Hat Author(s): Behdad Esfahbod |
Behdad Esfahbod | 2409d5f | 2011-04-21 17:14:28 -0400 | [diff] [blame] | 29 | * Google Author(s): Behdad Esfahbod |
Behdad Esfahbod | f0954d1 | 2009-07-30 15:33:57 -0400 | [diff] [blame] | 30 | */ |
| 31 | |
Behdad Esfahbod | c57d454 | 2011-04-20 18:50:27 -0400 | [diff] [blame] | 32 | #ifndef HB_OBJECT_PRIVATE_HH |
| 33 | #define HB_OBJECT_PRIVATE_HH |
Behdad Esfahbod | f0954d1 | 2009-07-30 15:33:57 -0400 | [diff] [blame] | 34 | |
Behdad Esfahbod | c57d454 | 2011-04-20 18:50:27 -0400 | [diff] [blame] | 35 | #include "hb-private.hh" |
Behdad Esfahbod | 23c86aa | 2009-08-03 21:40:20 -0400 | [diff] [blame] | 36 | |
Behdad Esfahbod | 45bfa99 | 2011-05-10 19:12:49 -0400 | [diff] [blame] | 37 | #include "hb-mutex-private.hh" |
| 38 | |
Behdad Esfahbod | 799b3c3 | 2009-11-05 19:37:58 -0500 | [diff] [blame] | 39 | |
| 40 | |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 41 | /* Debug */ |
| 42 | |
| 43 | #ifndef HB_DEBUG_OBJECT |
Behdad Esfahbod | 11e3ec4 | 2010-11-03 15:11:04 -0400 | [diff] [blame] | 44 | #define HB_DEBUG_OBJECT (HB_DEBUG+0) |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 45 | #endif |
| 46 | |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 47 | |
Behdad Esfahbod | 56eb5ad | 2011-05-04 19:27:37 -0400 | [diff] [blame] | 48 | /* atomic_int */ |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 49 | |
| 50 | /* We need external help for these */ |
| 51 | |
| 52 | #ifdef HAVE_GLIB |
| 53 | |
| 54 | #include <glib.h> |
| 55 | |
| 56 | typedef volatile int hb_atomic_int_t; |
| 57 | #define hb_atomic_int_fetch_and_add(AI, V) g_atomic_int_exchange_and_add (&(AI), V) |
| 58 | #define hb_atomic_int_get(AI) g_atomic_int_get (&(AI)) |
| 59 | #define hb_atomic_int_set(AI, V) g_atomic_int_set (&(AI), V) |
| 60 | |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 61 | |
Behdad Esfahbod | 56eb5ad | 2011-05-04 19:27:37 -0400 | [diff] [blame] | 62 | #elif defined(_MSC_VER) |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 63 | |
| 64 | #include <intrin.h> |
| 65 | |
| 66 | typedef long hb_atomic_int_t; |
| 67 | #define hb_atomic_int_fetch_and_add(AI, V) _InterlockedExchangeAdd (&(AI), V) |
| 68 | #define hb_atomic_int_get(AI) (_ReadBarrier (), (AI)) |
| 69 | #define hb_atomic_int_set(AI, V) ((void) _InterlockedExchange (&(AI), (V))) |
| 70 | |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 71 | |
| 72 | #else |
| 73 | |
Behdad Esfahbod | 56eb5ad | 2011-05-04 19:27:37 -0400 | [diff] [blame] | 74 | #warning "Could not find any system to define atomic_int macros, library will NOT be thread-safe" |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 75 | |
| 76 | typedef volatile int hb_atomic_int_t; |
| 77 | #define hb_atomic_int_fetch_and_add(AI, V) ((AI) += (V), (AI) - (V)) |
| 78 | #define hb_atomic_int_get(AI) (AI) |
| 79 | #define hb_atomic_int_set(AI, V) ((void) ((AI) = (V))) |
| 80 | |
Behdad Esfahbod | 2000179 | 2011-05-03 00:49:06 -0400 | [diff] [blame] | 81 | |
| 82 | #endif |
| 83 | |
| 84 | |
| 85 | |
| 86 | |
Behdad Esfahbod | 852e08e | 2011-04-27 21:45:51 -0400 | [diff] [blame] | 87 | /* reference_count */ |
| 88 | |
| 89 | typedef struct { |
| 90 | hb_atomic_int_t ref_count; |
| 91 | |
| 92 | #define HB_REFERENCE_COUNT_INVALID_VALUE ((hb_atomic_int_t) -1) |
| 93 | #define HB_REFERENCE_COUNT_INVALID {HB_REFERENCE_COUNT_INVALID_VALUE} |
| 94 | |
| 95 | inline void init (int v) { ref_count = v; /* non-atomic is fine */ } |
| 96 | inline int inc (void) { return hb_atomic_int_fetch_and_add (ref_count, 1); } |
| 97 | inline int dec (void) { return hb_atomic_int_fetch_and_add (ref_count, -1); } |
| 98 | inline void set (int v) { hb_atomic_int_set (ref_count, v); } |
| 99 | |
| 100 | inline int get (void) const { return hb_atomic_int_get (ref_count); } |
| 101 | inline bool is_invalid (void) const { return get () == HB_REFERENCE_COUNT_INVALID_VALUE; } |
| 102 | |
| 103 | } hb_reference_count_t; |
| 104 | |
| 105 | |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 106 | /* user_data */ |
| 107 | |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 108 | struct hb_user_data_array_t { |
| 109 | |
Behdad Esfahbod | 21d2c92 | 2011-05-05 14:47:53 -0400 | [diff] [blame] | 110 | struct hb_user_data_item_t { |
| 111 | hb_user_data_key_t *key; |
| 112 | void *data; |
| 113 | hb_destroy_func_t destroy; |
| 114 | |
| 115 | inline bool operator == (hb_user_data_key_t *other_key) const { return key == other_key; } |
| 116 | inline bool operator == (hb_user_data_item_t &other) const { return key == other.key; } |
| 117 | |
| 118 | void finish (void) { if (destroy) destroy (data); } |
| 119 | }; |
| 120 | |
Behdad Esfahbod | 45bfa99 | 2011-05-10 19:12:49 -0400 | [diff] [blame] | 121 | hb_lockable_set_t<hb_user_data_item_t, hb_static_mutex_t> items; |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 122 | |
Behdad Esfahbod | 218e67b | 2011-05-05 15:28:37 -0400 | [diff] [blame] | 123 | HB_INTERNAL bool set (hb_user_data_key_t *key, |
| 124 | void * data, |
| 125 | hb_destroy_func_t destroy); |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 126 | |
Behdad Esfahbod | 218e67b | 2011-05-05 15:28:37 -0400 | [diff] [blame] | 127 | HB_INTERNAL void *get (hb_user_data_key_t *key); |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 128 | |
Behdad Esfahbod | 45bfa99 | 2011-05-10 19:12:49 -0400 | [diff] [blame] | 129 | HB_INTERNAL void finish (void); |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 130 | }; |
| 131 | |
| 132 | |
Behdad Esfahbod | 852e08e | 2011-04-27 21:45:51 -0400 | [diff] [blame] | 133 | /* object_header */ |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 134 | |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 135 | typedef struct _hb_object_header_t hb_object_header_t; |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 136 | |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 137 | struct _hb_object_header_t { |
| 138 | hb_reference_count_t ref_count; |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 139 | hb_user_data_array_t user_data; |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 140 | |
| 141 | #define HB_OBJECT_HEADER_STATIC {HB_REFERENCE_COUNT_INVALID} |
| 142 | |
| 143 | static inline void *create (unsigned int size) { |
| 144 | hb_object_header_t *obj = (hb_object_header_t *) calloc (1, size); |
| 145 | |
| 146 | if (likely (obj)) |
| 147 | obj->init (); |
| 148 | |
| 149 | return obj; |
| 150 | } |
| 151 | |
| 152 | inline void init (void) { |
| 153 | ref_count.init (1); |
| 154 | } |
| 155 | |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 156 | inline bool is_inert (void) const { |
| 157 | return unlikely (ref_count.is_invalid ()); |
| 158 | } |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 159 | |
| 160 | inline void reference (void) { |
| 161 | if (unlikely (!this || this->is_inert ())) |
| 162 | return; |
| 163 | ref_count.inc (); |
| 164 | } |
| 165 | |
| 166 | inline bool destroy (void) { |
| 167 | if (unlikely (!this || this->is_inert ())) |
| 168 | return false; |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 169 | if (ref_count.dec () != 1) |
| 170 | return false; |
| 171 | |
Behdad Esfahbod | df077fa | 2011-05-12 01:19:39 -0400 | [diff] [blame] | 172 | ref_count.init (HB_REFERENCE_COUNT_INVALID_VALUE); |
| 173 | |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 174 | user_data.finish (); |
| 175 | |
| 176 | return true; |
| 177 | } |
| 178 | |
| 179 | inline bool set_user_data (hb_user_data_key_t *key, |
| 180 | void * data, |
Behdad Esfahbod | 31f18ab | 2011-06-15 09:49:58 -0400 | [diff] [blame] | 181 | hb_destroy_func_t destroy_func) { |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 182 | if (unlikely (!this || this->is_inert ())) |
| 183 | return false; |
| 184 | |
Behdad Esfahbod | 31f18ab | 2011-06-15 09:49:58 -0400 | [diff] [blame] | 185 | return user_data.set (key, data, destroy_func); |
Behdad Esfahbod | 29c67d3 | 2011-04-27 21:22:32 -0400 | [diff] [blame] | 186 | } |
| 187 | |
| 188 | inline void *get_user_data (hb_user_data_key_t *key) { |
| 189 | return user_data.get (key); |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 190 | } |
| 191 | |
| 192 | inline void trace (const char *function) const { |
Behdad Esfahbod | cc06c24 | 2011-07-25 20:25:44 -0400 | [diff] [blame] | 193 | DEBUG_MSG (OBJECT, (void *) this, |
| 194 | "refcount=%d %s", |
| 195 | this ? ref_count.get () : 0, |
| 196 | function); |
Behdad Esfahbod | fca368c | 2011-04-21 18:24:02 -0400 | [diff] [blame] | 197 | } |
| 198 | |
| 199 | }; |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 200 | |
| 201 | |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 202 | |
Behdad Esfahbod | 852e08e | 2011-04-27 21:45:51 -0400 | [diff] [blame] | 203 | |
| 204 | /* object */ |
| 205 | |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 206 | template <typename Type> |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 207 | static inline void hb_object_trace (const Type *obj, const char *function) |
| 208 | { |
| 209 | obj->header.trace (function); |
| 210 | } |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 211 | template <typename Type> |
Behdad Esfahbod | 1cd5969 | 2011-05-02 19:53:39 -0400 | [diff] [blame] | 212 | static inline Type *hb_object_create (void) |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 213 | { |
| 214 | Type *obj = (Type *) hb_object_header_t::create (sizeof (Type)); |
| 215 | hb_object_trace (obj, HB_FUNC); |
| 216 | return obj; |
| 217 | } |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 218 | template <typename Type> |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 219 | static inline bool hb_object_is_inert (const Type *obj) |
| 220 | { |
Behdad Esfahbod | 1cd5969 | 2011-05-02 19:53:39 -0400 | [diff] [blame] | 221 | return unlikely (obj->header.is_inert ()); |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 222 | } |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 223 | template <typename Type> |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 224 | static inline Type *hb_object_reference (Type *obj) |
| 225 | { |
| 226 | hb_object_trace (obj, HB_FUNC); |
| 227 | obj->header.reference (); |
| 228 | return obj; |
| 229 | } |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 230 | template <typename Type> |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 231 | static inline bool hb_object_destroy (Type *obj) |
| 232 | { |
| 233 | hb_object_trace (obj, HB_FUNC); |
| 234 | return obj->header.destroy (); |
| 235 | } |
Behdad Esfahbod | 852e08e | 2011-04-27 21:45:51 -0400 | [diff] [blame] | 236 | template <typename Type> |
| 237 | static inline bool hb_object_set_user_data (Type *obj, |
| 238 | hb_user_data_key_t *key, |
| 239 | void * data, |
| 240 | hb_destroy_func_t destroy) |
| 241 | { |
| 242 | return obj->header.set_user_data (key, data, destroy); |
| 243 | } |
| 244 | |
| 245 | template <typename Type> |
| 246 | static inline void *hb_object_get_user_data (Type *obj, |
| 247 | hb_user_data_key_t *key) |
| 248 | { |
| 249 | return obj->header.get_user_data (key); |
| 250 | } |
Behdad Esfahbod | ae008b9 | 2011-04-27 16:12:12 -0400 | [diff] [blame] | 251 | |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 252 | |
Behdad Esfahbod | ec6f9c2 | 2011-04-21 18:35:58 -0400 | [diff] [blame] | 253 | |
| 254 | |
Behdad Esfahbod | 863df68 | 2010-04-28 13:29:55 -0400 | [diff] [blame] | 255 | |
Behdad Esfahbod | c57d454 | 2011-04-20 18:50:27 -0400 | [diff] [blame] | 256 | #endif /* HB_OBJECT_PRIVATE_HH */ |