hb-private.hh 9.1 KB
Newer Older
B
Behdad Esfahbod 已提交
1
/*
B
Behdad Esfahbod 已提交
2 3
 * Copyright © 2007,2008,2009  Red Hat, Inc.
 * Copyright © 2011  Google, Inc.
B
Behdad Esfahbod 已提交
4
 *
B
Behdad Esfahbod 已提交
5
 *  This is part of HarfBuzz, a text shaping library.
B
Behdad Esfahbod 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *
 * Permission is hereby granted, without written agreement and without
 * license or royalty fees, to use, copy, modify, and distribute this
 * software and its documentation for any purpose, provided that the
 * above copyright notice and the following two paragraphs appear in
 * all copies of this software.
 *
 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 *
 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
 *
 * Red Hat Author(s): Behdad Esfahbod
B
Behdad Esfahbod 已提交
26
 * Google Author(s): Behdad Esfahbod
B
Behdad Esfahbod 已提交
27 28
 */

29 30
#ifndef HB_PRIVATE_HH
#define HB_PRIVATE_HH
B
Behdad Esfahbod 已提交
31

B
Behdad Esfahbod 已提交
32 33 34
#if HAVE_CONFIG_H
#include "config.h"
#endif
B
Behdad Esfahbod 已提交
35

B
Behdad Esfahbod 已提交
36
#include "hb-common.h"
37

B
Behdad Esfahbod 已提交
38
#include <stdlib.h>
B
Behdad Esfahbod 已提交
39 40
#include <string.h>
#include <assert.h>
B
Minor  
Behdad Esfahbod 已提交
41 42 43 44 45

/* We only use these two for debug output.  However, the debug code is
 * always seen by the compiler (and optimized out in non-debug builds.
 * If including these becomes a problem, we can start thinking about
 * someway around that. */
46 47
#include <stdio.h>
#include <errno.h>
B
Behdad Esfahbod 已提交
48

B
Behdad Esfahbod 已提交
49 50
HB_BEGIN_DECLS

B
Behdad Esfahbod 已提交
51 52 53 54 55 56 57 58 59 60 61 62

/* Essentials */

#ifndef NULL
# define NULL ((void *) 0)
#endif

#undef FALSE
#define FALSE 0

#undef TRUE
#define TRUE 1
B
Behdad Esfahbod 已提交
63

B
Behdad Esfahbod 已提交
64

B
Behdad Esfahbod 已提交
65 66 67 68
/* Basics */

#undef MIN
#define MIN(a,b) ((a) < (b) ? (a) : (b))
B
Minor  
Behdad Esfahbod 已提交
69

70 71 72
#undef MAX
#define MAX(a,b) ((a) > (b) ? (a) : (b))

B
Behdad Esfahbod 已提交
73 74
#undef  ARRAY_LENGTH
#define ARRAY_LENGTH(__array) ((signed int) (sizeof (__array) / sizeof (__array[0])))
75

76 77
#define HB_STMT_START do
#define HB_STMT_END   while (0)
B
Behdad Esfahbod 已提交
78

B
Behdad Esfahbod 已提交
79 80 81 82
#define _ASSERT_STATIC1(_line, _cond) typedef int _static_assert_on_line_##_line##_failed[(_cond)?1:-1]
#define _ASSERT_STATIC0(_line, _cond) _ASSERT_STATIC1 (_line, (_cond))
#define ASSERT_STATIC(_cond) _ASSERT_STATIC0 (__LINE__, (_cond))

B
Behdad Esfahbod 已提交
83 84
#define ASSERT_STATIC_EXPR(_cond) ((void) sizeof (char[(_cond) ? 1 : -1]))

B
Behdad Esfahbod 已提交
85

B
Behdad Esfahbod 已提交
86 87 88 89 90 91 92 93 94 95 96
/* Lets assert int types.  Saves trouble down the road. */

ASSERT_STATIC (sizeof (int8_t) == 1);
ASSERT_STATIC (sizeof (uint8_t) == 1);
ASSERT_STATIC (sizeof (int16_t) == 2);
ASSERT_STATIC (sizeof (uint16_t) == 2);
ASSERT_STATIC (sizeof (int32_t) == 4);
ASSERT_STATIC (sizeof (uint32_t) == 4);
ASSERT_STATIC (sizeof (int64_t) == 8);
ASSERT_STATIC (sizeof (uint64_t) == 8);

B
Behdad Esfahbod 已提交
97 98 99
ASSERT_STATIC (sizeof (hb_codepoint_t) == 4);
ASSERT_STATIC (sizeof (hb_position_t) == 4);
ASSERT_STATIC (sizeof (hb_mask_t) == 4);
100
ASSERT_STATIC (sizeof (hb_var_int_t) == 4);
B
Behdad Esfahbod 已提交
101

B
Behdad Esfahbod 已提交
102 103
/* Misc */

B
Behdad Esfahbod 已提交
104

B
Behdad Esfahbod 已提交
105
#if defined(__GNUC__) && (__GNUC__ > 2) && defined(__OPTIMIZE__)
106
#define _HB_BOOLEAN_EXPR(expr) ((expr) ? 1 : 0)
107 108
#define likely(expr) (__builtin_expect (_HB_BOOLEAN_EXPR(expr), 1))
#define unlikely(expr) (__builtin_expect (_HB_BOOLEAN_EXPR(expr), 0))
B
Behdad Esfahbod 已提交
109
#else
110 111
#define likely(expr) (expr)
#define unlikely(expr) (expr)
B
Behdad Esfahbod 已提交
112 113 114 115 116 117 118 119
#endif

#ifndef __GNUC__
#undef __attribute__
#define __attribute__(x)
#endif

#if __GNUC__ >= 3
120 121
#define HB_PURE_FUNC	__attribute__((pure))
#define HB_CONST_FUNC	__attribute__((const))
B
Behdad Esfahbod 已提交
122
#else
123 124
#define HB_PURE_FUNC
#define HB_CONST_FUNC
B
Behdad Esfahbod 已提交
125
#endif
126
#if __GNUC__ >= 4
127
#define HB_UNUSED	__attribute__((unused))
128
#else
129
#define HB_UNUSED
130
#endif
B
Behdad Esfahbod 已提交
131

B
Behdad Esfahbod 已提交
132
#ifndef HB_INTERNAL
B
Behdad Esfahbod 已提交
133
# define HB_INTERNAL __attribute__((__visibility__("hidden")))
B
Behdad Esfahbod 已提交
134 135
#endif

B
Behdad Esfahbod 已提交
136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151

#if (defined(__WIN32__) && !defined(__WINE__)) || defined(_MSC_VER)
#define snprintf _snprintf
#endif

#ifdef _MSC_VER
#undef inline
#define inline __inline
#endif

#ifdef __STRICT_ANSI__
#undef inline
#define inline __inline__
#endif


152 153 154 155 156 157 158 159 160
#if __GNUC__ >= 3
#define HB_FUNC __PRETTY_FUNCTION__
#elif defined(_MSC_VER)
#define HB_FUNC __FUNCSIG__
#else
#define HB_FUNC __func__
#endif


B
Behdad Esfahbod 已提交
161
/* Return the number of 1 bits in mask. */
B
Behdad Esfahbod 已提交
162
static inline HB_CONST_FUNC unsigned int
B
Behdad Esfahbod 已提交
163 164 165
_hb_popcount32 (uint32_t mask)
{
#if __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 4)
B
Behdad Esfahbod 已提交
166
  return __builtin_popcount (mask);
B
Behdad Esfahbod 已提交
167
#else
B
Behdad Esfahbod 已提交
168 169 170 171 172
  /* "HACKMEM 169" */
  register uint32_t y;
  y = (mask >> 1) &033333333333;
  y = mask - y - ((y >>1) & 033333333333);
  return (((y + (y >> 3)) & 030707070707) % 077);
B
Behdad Esfahbod 已提交
173 174 175
#endif
}

B
Behdad Esfahbod 已提交
176 177 178 179 180
/* Returns the number of bits needed to store number */
static inline HB_CONST_FUNC unsigned int
_hb_bit_storage (unsigned int number)
{
#if defined(__GNUC__) && (__GNUC__ >= 4) && defined(__OPTIMIZE__)
B
Behdad Esfahbod 已提交
181
  return likely (number) ? (sizeof (unsigned int) * 8 - __builtin_clz (number)) : 0;
B
Behdad Esfahbod 已提交
182 183 184 185 186 187 188 189 190
#else
  register unsigned int n_bits = 0;
  while (number) {
    n_bits++;
    number >>= 1;
  }
  return n_bits;
#endif
}
B
Behdad Esfahbod 已提交
191

B
Behdad Esfahbod 已提交
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
/* Returns the number of zero bits in the least significant side of number */
static inline HB_CONST_FUNC unsigned int
_hb_ctz (unsigned int number)
{
#if defined(__GNUC__) && (__GNUC__ >= 4) && defined(__OPTIMIZE__)
  return likely (number) ? __builtin_ctz (number) : 0;
#else
  register unsigned int n_bits = 0;
  if (unlikely (!number)) return 0;
  while (!(number & 1)) {
    n_bits++;
    number >>= 1;
  }
  return n_bits;
#endif
}

B
Behdad Esfahbod 已提交
209 210 211 212
/* Type of bsearch() / qsort() compare function */
typedef int (*hb_compare_func_t) (const void *, const void *);


B
Behdad Esfahbod 已提交
213 214 215 216 217 218
/* We need external help for these */

#ifdef HAVE_GLIB

#include <glib.h>

B
Behdad Esfahbod 已提交
219
typedef volatile int hb_atomic_int_t;
B
Behdad Esfahbod 已提交
220 221 222 223 224 225 226 227 228 229 230 231 232 233
#define hb_atomic_int_fetch_and_add(AI, V)	g_atomic_int_exchange_and_add (&(AI), V)
#define hb_atomic_int_get(AI)			g_atomic_int_get (&(AI))
#define hb_atomic_int_set(AI, V)		g_atomic_int_set (&(AI), V)

typedef GStaticMutex hb_mutex_t;
#define HB_MUTEX_INIT			G_STATIC_MUTEX_INIT
#define hb_mutex_init(M)		g_static_mutex_init (&M)
#define hb_mutex_lock(M)		g_static_mutex_lock (&M)
#define hb_mutex_trylock(M)		g_static_mutex_trylock (&M)
#define hb_mutex_unlock(M)		g_static_mutex_unlock (&M)

#else

#ifdef _MSC_VER
234 235 236
#define _HB__STR2__(x) #x
#define _HB__STR1__(x) _HB__STR2__(x)
#define _HB__LOC__ __FILE__ "("_HB__STR1__(__LINE__)") : Warning Msg: "
B
Bradley Grainger 已提交
237
#pragma message(_HB__LOC__"Could not find any system to define platform macros, library will NOT be thread-safe")
B
Behdad Esfahbod 已提交
238 239 240 241
#else
#warning "Could not find any system to define platform macros, library will NOT be thread-safe"
#endif

B
Behdad Esfahbod 已提交
242
typedef volatile int hb_atomic_int_t;
B
Behdad Esfahbod 已提交
243 244
#define hb_atomic_int_fetch_and_add(AI, V)	((AI) += (V), (AI) - (V))
#define hb_atomic_int_get(AI)			(AI)
B
Minor  
Behdad Esfahbod 已提交
245
#define hb_atomic_int_set(AI, V)		HB_STMT_START { (AI) = (V); } HB_STMT_END
B
Behdad Esfahbod 已提交
246

B
Behdad Esfahbod 已提交
247
typedef volatile int hb_mutex_t;
B
Minor  
Behdad Esfahbod 已提交
248 249 250 251 252
#define HB_MUTEX_INIT				0
#define hb_mutex_init(M)			HB_STMT_START { (M) = 0; } HB_STMT_END
#define hb_mutex_lock(M)			HB_STMT_START { (M) = 1; } HB_STMT_END
#define hb_mutex_trylock(M)			((M) = 1, 1)
#define hb_mutex_unlock(M)			HB_STMT_START { (M) = 0; } HB_STMT_END
B
Behdad Esfahbod 已提交
253 254 255 256

#endif


257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
/* A reference count */

typedef struct {
  hb_atomic_int_t ref_count;

#define HB_REFERENCE_COUNT_INVALID_VALUE ((hb_atomic_int_t) -1)
#define HB_REFERENCE_COUNT_INVALID {HB_REFERENCE_COUNT_INVALID_VALUE}

  inline void init (int v) { ref_count = v; /* non-atomic is fine */ }
  inline int inc (void) { return hb_atomic_int_fetch_and_add (ref_count,  1); }
  inline int dec (void) { return hb_atomic_int_fetch_and_add (ref_count, -1); }
  inline void set (int v) { return hb_atomic_int_set (ref_count, v); }

  inline int get (void) const { return hb_atomic_int_get (ref_count); }
  inline bool is_invalid (void) const { return get () == HB_REFERENCE_COUNT_INVALID_VALUE; }

} hb_reference_count_t;


B
Behdad Esfahbod 已提交
276 277
/* Big-endian handling */

B
Behdad Esfahbod 已提交
278 279
#define hb_be_uint16(v)		((uint16_t) ((((const uint8_t *)&(v))[0] << 8) + (((const uint8_t *)&(v))[1])))

280
#define hb_be_uint16_put(v,V)	HB_STMT_START { v[0] = (V>>8); v[1] = (V); } HB_STMT_END
B
Behdad Esfahbod 已提交
281 282 283
#define hb_be_uint16_get(v)	(uint16_t) ((v[0] << 8) + v[1])
#define hb_be_uint16_cmp(a,b)	(a[0] == b[0] && a[1] == b[1])

284
#define hb_be_uint32_put(v,V)	HB_STMT_START { v[0] = (V>>24); v[1] = (V>>16); v[2] = (V>>8); v[3] = (V); } HB_STMT_END
B
Behdad Esfahbod 已提交
285 286
#define hb_be_uint32_get(v)	(uint32_t) ((v[0] << 24) + (v[1] << 16) + (v[2] << 8) + v[3])
#define hb_be_uint32_cmp(a,b)	(a[0] == b[0] && a[1] == b[1] && a[2] == b[2] && a[3] == b[3])
B
Behdad Esfahbod 已提交
287 288


B
Behdad Esfahbod 已提交
289
/* ASCII tag/character handling */
B
Behdad Esfahbod 已提交
290 291

#define ISALPHA(c) (((c) >= 'a' && (c) <= 'z') || ((c) >= 'A' && (c) <= 'Z'))
292
#define ISALNUM(c) (((c) >= 'a' && (c) <= 'z') || ((c) >= 'A' && (c) <= 'Z') || ((c) >= '0' && (c) <= '9'))
B
Behdad Esfahbod 已提交
293 294 295
#define TOUPPER(c) (((c) >= 'a' && (c) <= 'z') ? (c) - 'a' + 'A' : (c))
#define TOLOWER(c) (((c) >= 'A' && (c) <= 'Z') ? (c) - 'A' + 'a' : (c))

B
Behdad Esfahbod 已提交
296 297 298 299 300
#define HB_TAG_CHAR4(s)   (HB_TAG(((const char *) s)[0], \
				  ((const char *) s)[1], \
				  ((const char *) s)[2], \
				  ((const char *) s)[3]))

B
Behdad Esfahbod 已提交
301

B
Behdad Esfahbod 已提交
302 303 304 305 306 307
/* Debug */

#ifndef HB_DEBUG
#define HB_DEBUG 0
#endif

308
static inline hb_bool_t /* always returns TRUE */
B
Behdad Esfahbod 已提交
309 310 311 312 313 314
_hb_trace (const char *what,
	   const char *function,
	   const void *obj,
	   unsigned int depth,
	   unsigned int max_depth)
{
315
  (void) ((depth < max_depth) && fprintf (stderr, "%s(%p) %-*d-> %s\n", what, obj, depth, depth, function));
B
Behdad Esfahbod 已提交
316 317 318
  return TRUE;
}

B
Behdad Esfahbod 已提交
319

320
#include "hb-object-private.hh"
B
Behdad Esfahbod 已提交
321

B
Behdad Esfahbod 已提交
322 323 324

HB_END_DECLS

325
#endif /* HB_PRIVATE_HH */