Bug 1862332 [wpt PR 42877] - WebKit export of https://bugs.webkit.org/show_bug.cgi...
[gecko.git] / gfx / harfbuzz / src / hb-debug.hh
blob559db4067e240b4cfb9959d68e29d351994e886c
1 /*
2 * Copyright © 2017 Google, Inc.
4 * This is part of HarfBuzz, a text shaping library.
6 * Permission is hereby granted, without written agreement and without
7 * license or royalty fees, to use, copy, modify, and distribute this
8 * software and its documentation for any purpose, provided that the
9 * above copyright notice and the following two paragraphs appear in
10 * all copies of this software.
12 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16 * DAMAGE.
18 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
21 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
24 * Google Author(s): Behdad Esfahbod
27 #ifndef HB_DEBUG_HH
28 #define HB_DEBUG_HH
30 #include "hb.hh"
31 #include "hb-atomic.hh"
32 #include "hb-algs.hh"
35 #ifndef HB_DEBUG
36 #define HB_DEBUG 0
37 #endif
41 * Global runtime options.
44 struct hb_options_t
46 bool unused : 1; /* In-case sign bit is here. */
47 bool initialized : 1;
48 bool uniscribe_bug_compatible : 1;
51 union hb_options_union_t {
52 int i;
53 hb_options_t opts;
55 static_assert ((sizeof (hb_atomic_int_t) >= sizeof (hb_options_union_t)), "");
57 HB_INTERNAL void
58 _hb_options_init ();
60 extern HB_INTERNAL hb_atomic_int_t _hb_options;
62 static inline hb_options_t
63 hb_options ()
65 #ifdef HB_NO_GETENV
66 return hb_options_t ();
67 #endif
68 /* Make a local copy, so we can access bitfield threadsafely. */
69 hb_options_union_t u;
70 u.i = _hb_options;
72 if (unlikely (!u.i))
74 _hb_options_init ();
75 u.i = _hb_options;
78 return u.opts;
83 * Debug output (needs enabling at compile time.)
86 static inline bool
87 _hb_debug (unsigned int level,
88 unsigned int max_level)
90 return level < max_level;
93 #define DEBUG_LEVEL_ENABLED(WHAT, LEVEL) (_hb_debug ((LEVEL), HB_DEBUG_##WHAT))
94 #define DEBUG_ENABLED(WHAT) (DEBUG_LEVEL_ENABLED (WHAT, 0))
96 static inline void
97 _hb_print_func (const char *func)
99 if (func)
101 unsigned int func_len = strlen (func);
102 /* Skip "static" */
103 if (0 == strncmp (func, "static ", 7))
104 func += 7;
105 /* Skip "typename" */
106 if (0 == strncmp (func, "typename ", 9))
107 func += 9;
108 /* Skip return type */
109 const char *space = strchr (func, ' ');
110 if (space)
111 func = space + 1;
112 /* Skip parameter list */
113 const char *paren = strchr (func, '(');
114 if (paren)
115 func_len = paren - func;
116 fprintf (stderr, "%.*s", (int) func_len, func);
120 template <int max_level> static inline void
121 _hb_debug_msg_va (const char *what,
122 const void *obj,
123 const char *func,
124 bool indented,
125 unsigned int level,
126 int level_dir,
127 const char *message,
128 va_list ap) HB_PRINTF_FUNC(7, 0);
129 template <int max_level> static inline void
130 _hb_debug_msg_va (const char *what,
131 const void *obj,
132 const char *func,
133 bool indented,
134 unsigned int level,
135 int level_dir,
136 const char *message,
137 va_list ap)
139 if (!_hb_debug (level, max_level))
140 return;
142 fprintf (stderr, "%-10s", what ? what : "");
144 if (obj)
145 fprintf (stderr, "(%*p) ", (int) (2 * sizeof (void *)), obj);
146 else
147 fprintf (stderr, " %*s ", (int) (2 * sizeof (void *)), "");
149 if (indented) {
150 #define VBAR "\342\224\202" /* U+2502 BOX DRAWINGS LIGHT VERTICAL */
151 #define VRBAR "\342\224\234" /* U+251C BOX DRAWINGS LIGHT VERTICAL AND RIGHT */
152 #define DLBAR "\342\225\256" /* U+256E BOX DRAWINGS LIGHT ARC DOWN AND LEFT */
153 #define ULBAR "\342\225\257" /* U+256F BOX DRAWINGS LIGHT ARC UP AND LEFT */
154 #define LBAR "\342\225\264" /* U+2574 BOX DRAWINGS LIGHT LEFT */
155 static const char bars[] =
156 VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR
157 VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR
158 VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR
159 VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR
160 VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR VBAR;
161 fprintf (stderr, "%2u %s" VRBAR "%s",
162 level,
163 bars + sizeof (bars) - 1 - hb_min ((unsigned int) sizeof (bars) - 1, (unsigned int) (sizeof (VBAR) - 1) * level),
164 level_dir ? (level_dir > 0 ? DLBAR : ULBAR) : LBAR);
165 } else
166 fprintf (stderr, " " VRBAR LBAR);
168 _hb_print_func (func);
170 if (message)
172 fprintf (stderr, ": ");
173 vfprintf (stderr, message, ap);
176 fprintf (stderr, "\n");
178 template <> inline void HB_PRINTF_FUNC(7, 0)
179 _hb_debug_msg_va<0> (const char *what HB_UNUSED,
180 const void *obj HB_UNUSED,
181 const char *func HB_UNUSED,
182 bool indented HB_UNUSED,
183 unsigned int level HB_UNUSED,
184 int level_dir HB_UNUSED,
185 const char *message HB_UNUSED,
186 va_list ap HB_UNUSED) {}
188 template <int max_level> static inline void
189 _hb_debug_msg (const char *what,
190 const void *obj,
191 const char *func,
192 bool indented,
193 unsigned int level,
194 int level_dir,
195 const char *message,
196 ...) HB_PRINTF_FUNC(7, 8);
197 template <int max_level> static inline void HB_PRINTF_FUNC(7, 8)
198 _hb_debug_msg (const char *what,
199 const void *obj,
200 const char *func,
201 bool indented,
202 unsigned int level,
203 int level_dir,
204 const char *message,
205 ...)
207 va_list ap;
208 va_start (ap, message);
209 _hb_debug_msg_va<max_level> (what, obj, func, indented, level, level_dir, message, ap);
210 va_end (ap);
212 template <> inline void
213 _hb_debug_msg<0> (const char *what HB_UNUSED,
214 const void *obj HB_UNUSED,
215 const char *func HB_UNUSED,
216 bool indented HB_UNUSED,
217 unsigned int level HB_UNUSED,
218 int level_dir HB_UNUSED,
219 const char *message HB_UNUSED,
220 ...) HB_PRINTF_FUNC(7, 8);
221 template <> inline void HB_PRINTF_FUNC(7, 8)
222 _hb_debug_msg<0> (const char *what HB_UNUSED,
223 const void *obj HB_UNUSED,
224 const char *func HB_UNUSED,
225 bool indented HB_UNUSED,
226 unsigned int level HB_UNUSED,
227 int level_dir HB_UNUSED,
228 const char *message HB_UNUSED,
229 ...) {}
231 #define DEBUG_MSG_LEVEL(WHAT, OBJ, LEVEL, LEVEL_DIR, ...) _hb_debug_msg<HB_DEBUG_##WHAT> (#WHAT, (OBJ), nullptr, true, (LEVEL), (LEVEL_DIR), __VA_ARGS__)
232 #define DEBUG_MSG(WHAT, OBJ, ...) _hb_debug_msg<HB_DEBUG_##WHAT> (#WHAT, (OBJ), nullptr, false, 0, 0, __VA_ARGS__)
233 #define DEBUG_MSG_FUNC(WHAT, OBJ, ...) _hb_debug_msg<HB_DEBUG_##WHAT> (#WHAT, (OBJ), HB_FUNC, false, 0, 0, __VA_ARGS__)
237 * Printer
240 template <typename T>
241 struct hb_printer_t {
242 const char *print (const T&) { return "something"; }
245 template <>
246 struct hb_printer_t<bool> {
247 const char *print (bool v) { return v ? "true" : "false"; }
250 template <>
251 struct hb_printer_t<hb_empty_t> {
252 const char *print (hb_empty_t) { return ""; }
257 * Trace
260 template <typename T>
261 static inline void _hb_warn_no_return (bool returned)
263 if (unlikely (!returned)) {
264 fprintf (stderr, "OUCH, returned with no call to return_trace(). This is a bug, please report.\n");
267 template <>
268 /*static*/ inline void _hb_warn_no_return<hb_empty_t> (bool returned HB_UNUSED) {}
269 template <>
270 /*static*/ inline void _hb_warn_no_return<void> (bool returned HB_UNUSED) {}
272 template <int max_level, typename ret_t>
273 struct hb_auto_trace_t
275 explicit inline hb_auto_trace_t (unsigned int *plevel_,
276 const char *what_,
277 const void *obj_,
278 const char *func,
279 const char *message,
280 ...) HB_PRINTF_FUNC(6, 7)
281 : plevel (plevel_), what (what_), obj (obj_), returned (false)
283 if (plevel) ++*plevel;
285 va_list ap;
286 va_start (ap, message);
287 _hb_debug_msg_va<max_level> (what, obj, func, true, plevel ? *plevel : 0, +1, message, ap);
288 va_end (ap);
290 ~hb_auto_trace_t ()
292 _hb_warn_no_return<ret_t> (returned);
293 if (!returned) {
294 _hb_debug_msg<max_level> (what, obj, nullptr, true, plevel ? *plevel : 1, -1, " ");
296 if (plevel) --*plevel;
299 template <typename T>
300 T ret (T&& v,
301 const char *func = "",
302 unsigned int line = 0)
304 if (unlikely (returned)) {
305 fprintf (stderr, "OUCH, double calls to return_trace(). This is a bug, please report.\n");
306 return std::forward<T> (v);
309 _hb_debug_msg<max_level> (what, obj, func, true, plevel ? *plevel : 1, -1,
310 "return %s (line %u)",
311 hb_printer_t<hb_decay<decltype (v)>>().print (v), line);
312 if (plevel) --*plevel;
313 plevel = nullptr;
314 returned = true;
315 return std::forward<T> (v);
318 private:
319 unsigned int *plevel;
320 const char *what;
321 const void *obj;
322 bool returned;
324 template <typename ret_t> /* Make sure we don't use hb_auto_trace_t when not tracing. */
325 struct hb_auto_trace_t<0, ret_t>
327 explicit inline hb_auto_trace_t (unsigned int *plevel_,
328 const char *what_,
329 const void *obj_,
330 const char *func,
331 const char *message,
332 ...) HB_PRINTF_FUNC(6, 7) {}
334 template <typename T>
335 T ret (T&& v,
336 const char *func HB_UNUSED = nullptr,
337 unsigned int line HB_UNUSED = 0) { return std::forward<T> (v); }
340 /* For disabled tracing; optimize out everything.
341 * https://github.com/harfbuzz/harfbuzz/pull/605 */
342 template <typename ret_t>
343 struct hb_no_trace_t {
344 template <typename T>
345 T ret (T&& v,
346 const char *func HB_UNUSED = nullptr,
347 unsigned int line HB_UNUSED = 0) { return std::forward<T> (v); }
350 #define return_trace(RET) return trace.ret (RET, HB_FUNC, __LINE__)
354 * Instances.
357 #ifndef HB_DEBUG_ARABIC
358 #define HB_DEBUG_ARABIC (HB_DEBUG+0)
359 #endif
361 #ifndef HB_DEBUG_BLOB
362 #define HB_DEBUG_BLOB (HB_DEBUG+0)
363 #endif
365 #ifndef HB_DEBUG_CORETEXT
366 #define HB_DEBUG_CORETEXT (HB_DEBUG+0)
367 #endif
369 #ifndef HB_DEBUG_DIRECTWRITE
370 #define HB_DEBUG_DIRECTWRITE (HB_DEBUG+0)
371 #endif
373 #ifndef HB_DEBUG_FT
374 #define HB_DEBUG_FT (HB_DEBUG+0)
375 #endif
377 #ifndef HB_DEBUG_JUSTIFY
378 #define HB_DEBUG_JUSTIFY (HB_DEBUG+0)
379 #endif
381 #ifndef HB_DEBUG_OBJECT
382 #define HB_DEBUG_OBJECT (HB_DEBUG+0)
383 #endif
385 #ifndef HB_DEBUG_SHAPE_PLAN
386 #define HB_DEBUG_SHAPE_PLAN (HB_DEBUG+0)
387 #endif
389 #ifndef HB_DEBUG_UNISCRIBE
390 #define HB_DEBUG_UNISCRIBE (HB_DEBUG+0)
391 #endif
393 #ifndef HB_DEBUG_WASM
394 #define HB_DEBUG_WASM (HB_DEBUG+0)
395 #endif
398 * With tracing.
401 #ifndef HB_DEBUG_APPLY
402 #define HB_DEBUG_APPLY (HB_DEBUG+0)
403 #endif
404 #if HB_DEBUG_APPLY
405 #define TRACE_APPLY(this) \
406 hb_auto_trace_t<HB_DEBUG_APPLY, bool> trace \
407 (&c->debug_depth, c->get_name (), this, HB_FUNC, \
408 "idx %u gid %u lookup %d", \
409 c->buffer->idx, c->buffer->cur().codepoint, (int) c->lookup_index)
410 #else
411 #define TRACE_APPLY(this) hb_no_trace_t<bool> trace
412 #endif
414 #ifndef HB_DEBUG_SANITIZE
415 #define HB_DEBUG_SANITIZE (HB_DEBUG+0)
416 #endif
417 #if HB_DEBUG_SANITIZE
418 #define TRACE_SANITIZE(this) \
419 hb_auto_trace_t<HB_DEBUG_SANITIZE, bool> trace \
420 (&c->debug_depth, c->get_name (), this, HB_FUNC, \
421 " ")
422 #else
423 #define TRACE_SANITIZE(this) hb_no_trace_t<bool> trace
424 #endif
426 #ifndef HB_DEBUG_SERIALIZE
427 #define HB_DEBUG_SERIALIZE (HB_DEBUG+0)
428 #endif
429 #if HB_DEBUG_SERIALIZE
430 #define TRACE_SERIALIZE(this) \
431 hb_auto_trace_t<HB_DEBUG_SERIALIZE, bool> trace \
432 (&c->debug_depth, "SERIALIZE", c, HB_FUNC, \
433 " ")
434 #else
435 #define TRACE_SERIALIZE(this) hb_no_trace_t<bool> trace
436 #endif
438 #ifndef HB_DEBUG_SUBSET
439 #define HB_DEBUG_SUBSET (HB_DEBUG+0)
440 #endif
441 #if HB_DEBUG_SUBSET
442 #define TRACE_SUBSET(this) \
443 hb_auto_trace_t<HB_DEBUG_SUBSET, bool> trace \
444 (&c->debug_depth, c->get_name (), this, HB_FUNC, \
445 " ")
446 #else
447 #define TRACE_SUBSET(this) hb_no_trace_t<bool> trace
448 #endif
450 #ifndef HB_DEBUG_SUBSET_REPACK
451 #define HB_DEBUG_SUBSET_REPACK (HB_DEBUG+0)
452 #endif
454 #ifndef HB_DEBUG_PAINT
455 #define HB_DEBUG_PAINT (HB_DEBUG+0)
456 #endif
457 #if HB_DEBUG_PAINT
458 #define TRACE_PAINT(this) \
459 HB_UNUSED hb_auto_trace_t<HB_DEBUG_PAINT, void> trace \
460 (&c->debug_depth, c->get_name (), this, HB_FUNC, \
461 " ")
462 #else
463 #define TRACE_PAINT(this) HB_UNUSED hb_no_trace_t<void> trace
464 #endif
467 #ifndef HB_DEBUG_DISPATCH
468 #define HB_DEBUG_DISPATCH ( \
469 HB_DEBUG_APPLY + \
470 HB_DEBUG_SANITIZE + \
471 HB_DEBUG_SERIALIZE + \
472 HB_DEBUG_SUBSET + \
473 HB_DEBUG_PAINT + \
475 #endif
476 #if HB_DEBUG_DISPATCH
477 #define TRACE_DISPATCH(this, format) \
478 hb_auto_trace_t<context_t::max_debug_depth, typename context_t::return_t> trace \
479 (&c->debug_depth, c->get_name (), this, HB_FUNC, \
480 "format %u", (unsigned) format)
481 #else
482 #define TRACE_DISPATCH(this, format) hb_no_trace_t<typename context_t::return_t> trace
483 #endif
486 #ifndef HB_BUFFER_MESSAGE_MORE
487 #define HB_BUFFER_MESSAGE_MORE (HB_DEBUG+1)
488 #endif
491 #endif /* HB_DEBUG_HH */