1 // Render with Asciidoctor
3 = Babeltrace{nbsp}2 contributor's guide
4 Jérémie Galarneau, Philippe Proulx
10 :bt2: Babeltrace{nbsp}2
14 This is a partial contributor's guide for the
15 https://babeltrace.org[{bt2}] project. If you have any
16 questions that are not answered by this guide, please post them on
17 https://lists.lttng.org/cgi-bin/mailman/listinfo/lttng-dev[Babeltrace's
22 === Object reference counting and lifetime
24 This section covers the rationale behind the design of {bt2}'s
25 object lifetime management. This applies to the {bt2} library, as
26 well as to the CTF writer library (although the public reference
27 counting functions are not named the same way).
29 Starting from Babeltrace{nbsp}2.0, all publicly exposed objects inherit
30 a common base: `bt_object`. This base provides a number of facilities to
31 all objects, chief amongst which are lifetime management functions.
33 The lifetime of some public objects is managed by reference counting. In
34 this case, the API offers the `+bt_*_get_ref()+` and `+bt_*_put_ref()+`
35 functions which respectively increment and decrement an object's
38 As far as lifetime management in concerned, {bt2} makes a clear
39 distinction between regular objects, which have a single parent, and
40 root objects, which don't.
44 Let us consider a problematic case to illustrate the need for this
47 A user of the {bt2} library creates a trace class, which _has_ a
48 stream class (the class of a stream) and that stream class, in turn,
49 _has_ an event class (the class of an event).
51 Nothing prevents this user from releasing his reference on any one of
52 these objects in any order. However, all objects in the
53 __trace--stream class--event class__ hierarchy can be retrieved
56 For instance, the user could discard his reference on both the event
57 class and the stream class, only keeping a reference on the trace class.
58 From this trace class reference, stream classes can be enumerated,
59 providing the user with a new reference to the stream class he discarded
60 earlier. Event classes can also be enumerated from stream classes,
61 providing the user with references to the individual event classes.
63 Conversely, the user could also hold a reference to an event class and
64 retrieve its parent stream class. The trace class, in turn, can then be
65 retrieved from the stream class.
67 This example illustrates what could be interpreted as a circular
68 reference dependency existing between these objects. Of course, if the
69 objects in such a scenario were to hold references to each other (in
70 both directions), we would be in presence of a circular ownership
71 resulting in a leak of both objects as their reference counts would
74 Nonetheless, the API must offer the guarantee that holding a node to any
75 node of the graph keeps all other reachable nodes alive.
79 The scheme employed in {bt2} to break this cycle consists in the
80 "children" holding _reverse component references_ to their parents. That
81 is, in the context of the trace IR, that event classes hold a reference
82 to their parent stream class and stream classes hold a reference to
83 their parent trace class.
85 On the other hand, parents hold _claiming aggregation references_ to
86 their children. A claiming aggregation reference means that the object
87 being referenced should not be deleted as long as the reference still
88 exists. In this respect, it can be said that parents truly hold the
89 ownership of their children, since they control their lifetime.
90 Conversely, the reference counting mechanism is leveraged by children to
91 notify parents that no other child indirectly exposes the parent.
93 When a parented object's reference count reaches zero, it invokes
94 `+bt_*_put_ref()+` on its parent and does _not_ free itself. However,
95 from that point, the object depends on its parent to signal the moment
96 when it can be safely reclaimed.
98 The invocation of `+bt_*_put_ref()+` by the last children holding a
99 reference to its parent might trigger a cascade of `+bt_*_put_ref()+`
100 from child to parent. Eventually, a **root** object is reached. At that
101 point, if this orphaned object's reference count reaches zero, the
102 object invokes the destructor method defined by everyone of its children
103 as part of their base `struct bt_object`. The key point here is that the
104 cascade of destructor will necessarily originate from the root and
105 propagate in preorder to the children. These children will propagate the
106 destruction to their own children before reclaiming their own memory.
107 This ensures that a node's pointer to its parent is _always_ valid since
108 the parent has the responsibility of tearing-down their children before
109 cleaning themselves up.
111 Assuming a reference to an object is _acquired_ by calling
112 `+bt_*_get_ref()+` while its reference count is zero, the object
113 acquires, in turn, a reference on its parent using `+bt_*_get_ref()+`.
114 At that point, the child can be thought of as having converted its weak
115 reference to its parent into a regular reference. That is why this
116 reference is referred to as a _claiming_ aggregation reference.
120 This scheme imposes a number of strict rules defining the relation
123 * Objects may only have one parent.
124 * Objects, beside the root, are only retrievable from their direct
129 The initial situation is rather simple: **User{nbsp}A** is holding a
130 reference to a trace class, **TC1**. As per the rules previously
131 enounced, stream classes **SC1** and **SC2** don't hold a reference to
132 **TC1** since their own reference counts are zero. The same holds true
133 for **EC1**, **EC2** and **EC3** with respect to **SC1** and **SC2**.
135 image::doc/contributing-images/bt-ref01.png[]
137 In this second step, we can see that **User{nbsp}A** has acquired a
138 reference on **SC2** through the trace class, **TC1**.
140 The stream class's reference count transitions from zero to one,
141 triggering the acquisition of a strong reference on **TC1** from
144 Hence, at this point, the trace class's ownership is shared by
145 **User{nbsp}A** and **SC2**.
147 image::doc/contributing-images/bt-ref02.png[]
149 Next, **User{nbsp}A** acquires a reference on the **EC3** event class
150 through its parent stream class, **SC2**. Again, the transition of an
151 object's reference count from 0 to 1 triggers the acquisition of a
152 reference on its parent.
154 Note that SC2's reference count was incremented to 2. The trace class's
155 reference count remains unchanged.
157 image::doc/contributing-images/bt-ref03.png[]
159 **User{nbsp}A** decides to drop its reference on **SC2**. **SC2**'s
160 reference count returns back to 1, everything else remaining unchanged.
162 image::doc/contributing-images/bt-ref04.png[]
164 **User{nbsp}A** can then decide to drop its reference on the trace
165 class. This results in a reversal of the initial situation:
166 **User{nbsp}A** now owns an event, **EC3**, which is keeping everything
167 else alive and reachable.
169 image::doc/contributing-images/bt-ref05.png[]
171 If another object, **User{nbsp}B**, enters the picture and acquires a
172 reference on the **SC1** stream class, we see that **SC1**'s reference
173 count transitioned from 0 to 1, triggering the acquisition of a
174 reference on **TC1**.
176 image::doc/contributing-images/bt-ref06.png[]
178 **User{nbsp}B** hands off a reference to **EC1**, acquired through
179 **SC1**, to another object, **User{nbsp}C**. The acquisition of a
180 reference on **EC1**, which transitions from 0 to 1, triggers the
181 acquisition of a reference on its parent, **SC1**.
183 image::doc/contributing-images/bt-ref07.png[]
185 At some point, **User{nbsp}A** releases its reference on **EC3**. Since
186 **EC3**'s reference count transitions to zero, it releases its reference
187 on **SC2**. **SC2**'s reference count, in turn, reaches zero and it
188 releases its reference to **TC1**.
190 **TC1**'s reference count is now 1 and no further action is taken.
192 image::doc/contributing-images/bt-ref08.png[]
194 **User{nbsp}B** releases its reference on **SC1**. **User{nbsp}C**
195 becomes the sole owner of the whole hierarchy through his ownership of
198 image::doc/contributing-images/bt-ref09.png[]
200 Finally, **User{nbsp}C** releases his ownership of **EC1**, triggering
201 the release of the whole hierarchy. Let's walk through the reclamation
204 Mirroring what happened when **User{nbsp}A** released its last reference
205 on **EC3**, the release of **EC1** by **User{nbsp}C** causes its
206 reference count to fall to zero.
208 This transition to zero causes **EC1** to release its reference on
209 **SC1**. **SC1**'s reference count reaching zero causes it to release
210 its reference on **TC1**.
212 image::doc/contributing-images/bt-ref10.png[]
214 Since the reference count of **TC1**, a root object, has reached zero,
215 it invokes the destructor method on its children. This method is
216 recursive and causes the stream classes to call the destructor method on
219 The event classes are reached and, having no children of their own, are
222 image::doc/contributing-images/bt-ref11.png[]
224 The stream classes having destroyed their children, are then reclaimed
227 image::doc/contributing-images/bt-ref12.png[]
229 Finally, the stream classes having been reclaimed, **TC1** is reclaimed.
231 image::doc/contributing-images/bt-ref13.png[]
236 Logging is a great instrument for a developer to be able to collect
237 information about a running software.
239 {bt2} is a complex software with many layers. When a {bt2}
240 graph fails to run, what caused the failure? It could be caused by any
241 component, any message iterator, and any deeply nested validation of a
242 CTF IR object (within the `ctf` plugin), for example. With the
243 appropriate logging statements manually placed in the source code, we
244 can find the cause of a bug faster.
246 While <<choose-a-log-level,care must be taken>> when placing _DEBUG_ to
247 _FATAL_ logging statements, you should liberally instrument your
248 {bt2} module with _TRACE_ logging statements to help future you
249 and other developers understand what's happening at run time.
253 The {bt2} logging API is internal: it is not exposed to the users
254 of the library; only to their developers. The only thing that a library
255 user can control is the current log level of the library itself with
256 `bt_logging_set_global_level()` and the initial library's log level with
257 the `LIBBABELTRACE2_INIT_LOG_LEVEL` environment variable.
259 This API is based on https://github.com/wonder-mice/zf_log[zf_log], a
260 lightweight, yet featureful, MIT-licensed core logging library for C and
261 {cpp}. The zf_log source files were modified to have the `BT_` and
262 `bt_` prefixes, and other small changes, like color support and using
263 the project's `BT_DEBUG_MODE` definition instead of the standard
266 The logging functions are implemented in the logging convenience
267 library (`src/logging` directory).
272 The logging API headers are:
274 `<babeltrace2/logging.h>`::
275 Public header which a library user can use to set and get
276 libbabeltrace2's current log level.
279 Internal, generic logging API which you can use in any {bt2}
280 module. This is the translation of `zf_log.h`.
282 This header offers the <<gen-logging-statements,generic logging
286 Specific internal header to use within the library.
288 This header defines `BT_LOG_OUTPUT_LEVEL` to a custom, library-wide
289 hidden symbol which is the library's current log level before including
292 This header offers the <<lib-logging-statements,library-specific logging
295 `"logging/comp-logging.h"`::
296 Specific internal header to use within a component class.
298 This header offers the <<comp-logging-statements,component-specific
299 logging statement macros>>.
304 The internal logging API offers the following log levels, in ascending
307 [options="header,autowidth",cols="4"]
310 |Log level short name
311 |Internal API enumerator
312 |Public API enumerator
317 |`BT_LOGGING_LEVEL_TRACE`
322 |`BT_LOGGING_LEVEL_DEBUG`
327 |`BT_LOGGING_LEVEL_INFO`
332 |`BT_LOGGING_LEVEL_WARNING`
337 |`BT_LOGGING_LEVEL_ERROR`
342 |`BT_LOGGING_LEVEL_FATAL`
347 |`BT_LOGGING_LEVEL_NONE`
350 The short name is accepted by the log level environment variables and by
351 the CLI's `--log-level` options.
353 See <<choose-a-log-level,how to decide which one to use>> below.
355 There are two important log level expressions:
357 [[build-time-log-level]]Build-time, minimal log level::
358 The minimal log level, or build-time log level, is set at build time
359 and determines the minimal log level of the logging statements which
360 can be executed. This applies to all the modules (CLI, library,
361 plugins, bindings, etc.).
363 All the logging statements with a level below this level are **not built
364 at all**. All the logging statements with a level equal to or greater
365 than this level _can_ be executed, depending on the
366 <<run-time-log-level,run-time log level>>.
368 You can set this level at configuration time with the
369 `BABELTRACE_MINIMAL_LOG_LEVEL` environment variable, for example:
373 $ BABELTRACE_MINIMAL_LOG_LEVEL=INFO ./configure
377 The default build-time log level is `DEBUG`. For optimal performance,
378 set it to `INFO`, which effectively disables all fast path logging in
379 all the {bt2} modules. You can't set it to `WARNING`, `ERROR`,
380 `FATAL`, or `NONE` because the impact on performance is minuscule
381 starting from the _INFO_ log level anyway and we want any {bt2}
382 build to always be able to print _INFO_-level logs.
384 The library's public API provides `bt_logging_get_minimal_level()` to
385 get the configured minimal log level.
387 [[run-time-log-level]]Run-time, dynamic log level::
388 The dynamic log level is set at run time and determines the current,
389 _active_ log level. All the logging statements with a level below
390 this level are not executed, **but they still evaluate the
391 condition**. All the logging statements with a level equal to or
392 greater than this level are executed, provided that their level is
393 also <<build-time-log-level,enabled at build time>>.
395 `zf_log` has a concept of a global run-time log level which uses the
396 `_bt_log_global_output_lvl` symbol. In practice, we never use this
397 symbol, and always make sure that `BT_LOG_OUTPUT_LEVEL` is defined to a
398 module-wise expression before including `"logging/log.h"`.
400 In the library, `"lib/logging.h"` defines its own
401 `BT_LOG_OUTPUT_LEVEL` to the library's log level symbol before it
402 includes `"logging/log.h"` itself.
404 In libbabeltrace2, the user can set the current run-time log level with
405 the `bt_logging_set_global_level()` function, for example:
410 bt_logging_set_global_level(BT_LOGGING_LEVEL_INFO);
414 The library's initial run-time log level is defined by the
415 `LIBBABELTRACE2_INIT_LOG_LEVEL` environment variable, or set to _NONE_
416 if this environment variable is undefined.
418 Other modules have their own way of setting their run-time log level.
420 For example, the CLI uses the `BABELTRACE_CLI_LOG_LEVEL` environment
421 variable, as well as its global `--log-level` option:
424 $ babeltrace2 --log-level=I ...
427 The components use their own log level (as returned by
428 `bt_component_get_logging_level()`). With the CLI, you can set a
429 specific component's log level with its own, position-dependent
430 `--log-level` option:
433 $ babeltrace2 /path/to/trace -c sink.ctf.fs --log-level=D
436 Code which is common to the whole project, for example `src/common`
437 and `src/compat`, use function parameters to get its run-time log
442 char *bt_common_get_home_plugin_path(int log_level);
445 Typically, when a logging-enabled module calls such a function, it
446 passes its own log level expression directly (`BT_LOG_OUTPUT_LEVEL`):
450 path = bt_common_get_home_plugin_path(BT_LOG_OUTPUT_LEVEL);
453 Otherwise, just pass `BT_LOG_NONE`:
456 path = bt_common_get_home_plugin_path(BT_LOG_NONE);
459 [[gen-logging-statements]]
460 ==== Generic logging statement macros
462 The {bt2} logging statement macros work just like `printf()`
463 (except the `+BT_LOG*_STR()+` ones) and contain their <<log-levels,log
464 level>> (short name) in their name.
466 Each of the following macros evaluate the
467 <<build-time-log-level,build-time log level>> definition and
468 <<run-time-log-level,run-time log level>> expression (as defined by
469 `BT_LOG_OUTPUT_LEVEL`) to log conditionally.
471 See <<logging-instrument-c-file-gen>> and
472 <<logging-instrument-h-file-gen>> to learn how to be able to use the
475 `+BT_LOGT("format string", ...)+`::
476 Generic trace logging statement.
478 `+BT_LOGD("format string", ...)+`::
479 Generic debug logging statement.
481 `+BT_LOGI("format string", ...)+`::
482 Generic info logging statement.
484 `+BT_LOGW("format string", ...)+`::
485 Generic warning logging statement.
487 `+BT_LOGE("format string", ...)+`::
488 Generic error logging statement.
490 `+BT_LOGF("format string", ...)+`::
491 Generic fatal logging statement.
493 `+BT_LOGT_STR("preformatted string")+`::
494 Generic preformatted string trace logging statement.
496 `+BT_LOGD_STR("preformatted string")+`::
497 Generic preformatted string debug logging statement.
499 `+BT_LOGI_STR("preformatted string")+`::
500 Generic preformatted string info logging statement.
502 `+BT_LOGW_STR("preformatted string")+`::
503 Generic preformatted string warning logging statement.
505 `+BT_LOGE_STR("preformatted string")+`::
506 Generic preformatted string error logging statement.
508 `+BT_LOGF_STR("preformatted string")+`::
509 Generic preformatted string fatal logging statement.
511 `+BT_LOGT_MEM(data_ptr, data_size, "format string", ...)+`::
512 Generic memory trace logging statement.
514 `+BT_LOGD_MEM(data_ptr, data_size, "format string", ...)+`::
515 Generic memory debug logging statement.
517 `+BT_LOGI_MEM(data_ptr, data_size, "format string", ...)+`::
518 Generic memory info logging statement.
520 `+BT_LOGW_MEM(data_ptr, data_size, "format string", ...)+`::
521 Generic memory warning logging statement.
523 `+BT_LOGE_MEM(data_ptr, data_size, "format string", ...)+`::
524 Generic memory error logging statement.
526 `+BT_LOGF_MEM(data_ptr, data_size, "format string", ...)+`::
527 Generic memory fatal logging statement.
529 `+BT_LOGT_ERRNO("initial message", "format string", ...)+`::
530 Generic `errno` string trace logging statement.
532 `+BT_LOGD_ERRNO("initial message", "format string", ...)+`::
533 Generic `errno` string debug logging statement.
535 `+BT_LOGI_ERRNO("initial message", "format string", ...)+`::
536 Generic `errno` string info logging statement.
538 `+BT_LOGW_ERRNO("initial message", "format string", ...)+`::
539 Generic `errno` string warning logging statement.
541 `+BT_LOGE_ERRNO("initial message", "format string", ...)+`::
542 Generic `errno` string error logging statement.
544 `+BT_LOGF_ERRNO("initial message", "format string", ...)+`::
545 Generic `errno` string fatal logging statement.
547 [[lib-logging-statements]]
548 ==== Library-specific logging statement macros
550 The {bt2} library contains an internal logging API based on the
551 generic logging framework. You can use it to log known {bt2}
552 objects without having to manually log each member.
554 See <<logging-instrument-c-file-lib>>
555 and <<logging-instrument-h-file-lib>> to
556 learn how to be able to use the following macros.
558 The library logging statement macros are named `+BT_LIB_LOG*()+` instead
561 `+BT_LIB_LOGT("format string", ...)+`::
562 Library trace logging statement.
564 `+BT_LIB_LOGD("format string", ...)+`::
565 Library debug logging statement.
567 `+BT_LIB_LOGI("format string", ...)+`::
568 Library info logging statement.
570 `+BT_LIB_LOGW("format string", ...)+`::
571 Library warning logging statement.
573 `+BT_LIB_LOGE("format string", ...)+`::
574 Library error logging statement.
576 `+BT_LIB_LOGF("format string", ...)+`::
577 Library fatal logging statement.
579 `+BT_LIB_LOGW_APPEND_CAUSE("format string", ...)+`::
580 Library warning logging statement, and unconditional error cause
583 `+BT_LIB_LOGE_APPEND_CAUSE("format string", ...)+`::
584 Library error logging statement, and unconditional error cause
587 The macros above accept the typical `printf()` conversion specifiers
588 with the following limitations:
590 * The `+*+` width specifier is not accepted.
591 * The `+*+` precision specifier is not accepted.
592 * The `j` and `t` length modifiers are not accepted.
593 * The `n` format specifier is not accepted.
594 * The format specifiers defined in `<inttypes.h>` are not accepted,
595 except for `PRId64`, `PRIu64`, `PRIx64`, `PRIX64`, `PRIo64`, and
598 The {bt2} library custom conversion specifier is accepted. Its
599 syntax is either `%!u` to format a UUID (`bt_uuid` type), or:
601 . Introductory `%!` sequence.
603 . **Optional**: `[` followed by a custom prefix for the printed fields
604 of this specifier, followed by `]`. The standard form is to end this
605 prefix with `-` so that, for example, with the prefix `tc-`, the
606 complete field name becomes `tc-addr`.
608 . **Optional**: `pass:[+]` to print extended object members. This
609 depends on the provided format specifier.
611 . Format specifier (see below).
613 The available format specifiers are:
615 [options="header,autowidth",cols="3"]
622 |Trace IR field class
623 |`+const struct bt_field_class *+`
627 |`+const struct bt_field *+`
631 |`+const struct bt_field_path *+`
634 |Trace IR event class
635 |`+const struct bt_event_class *+`
639 |`+const struct bt_event *+`
642 |Trace IR stream class.
643 |`+const struct bt_stream_class *+`
647 |`+const struct bt_stream *+`
651 |`+const struct bt_packet *+`
654 |Trace IR trace class
655 |`+const struct bt_trace_class *+`
659 |`+const struct bt_trace *+`
662 |Trace IR clock class
663 |`+const struct bt_clock_class *+`
666 |Trace IR clock snapshot
667 |`+const struct bt_clock_snapshot *+`
671 |`+const struct bt_value *+`
675 |`const struct bt_integer_range_set *`
679 |`+const struct bt_message *+`
682 |Message iterator class
683 |`struct bt_message_iterator_class *`
687 |`struct bt_message_iterator *`
691 |`struct bt_component_class *`
695 |`+const struct bt_component *+`
699 |`+const struct bt_port *+`
703 |`+const struct bt_connection *+`
707 |`+const struct bt_graph *+`
711 |`+struct bt_interrupter *+`
715 |`+const struct bt_plugin *+`
719 |`+const struct bt_error_cause *+`
723 |`+const struct bt_object_pool *+`
727 |`+const struct bt_object *+`
730 Conversion specifier examples:
737 The ``, `` string (comma and space) is printed between individual
738 fields, but **not after the last one**. Therefore, you must put this
739 separator in the format string between two conversion specifiers, for
744 BT_LIB_LOGW("Message: count=%u, %!E, %!+K", count, event_class, clock_class);
747 Example with a custom prefix:
751 BT_LIB_LOGI("Some message: %![ec-a-]e, %![ec-b-]+e", ec_a, ec_b);
754 It is safe to pass `NULL` as any {bt2} object parameter: the macros
755 only print its null address.
757 WARNING: Build-time `printf()` format checks are disabled for the
758 `+BT_LIB_LOG*()+` macros because there are custom conversion specifiers,
759 so make sure to test your logging statements.
761 [[comp-logging-statements]]
762 ==== Component-specific logging statement macros
764 There are available logging macros for components. They prepend a prefix
765 including the component's name to the logging message.
767 See <<logging-instrument-c-file-compcls>> and
768 <<logging-instrument-h-file-compcls>> to learn how to be able to use the
771 The component logging statement macros are named `+BT_COMP_LOG*()+`
772 instead of `+BT_LOG*()+`:
774 `+BT_COMP_LOGT("format string", ...)+`::
775 Component trace logging statement.
777 `+BT_COMP_LOGD("format string", ...)+`::
778 Component debug logging statement.
780 `+BT_COMP_LOGI("format string", ...)+`::
781 Component info logging statement.
783 `+BT_COMP_LOGW("format string", ...)+`::
784 Component warning logging statement.
786 `+BT_COMP_LOGE("format string", ...)+`::
787 Component error logging statement.
789 `+BT_COMP_LOGF("format string", ...)+`::
790 Component fatal logging statement.
792 `+BT_COMP_LOGT_STR("preformatted string")+`::
793 Component preformatted string trace logging statement.
795 `+BT_COMP_LOGD_STR("preformatted string")+`::
796 Component preformatted string debug logging statement.
798 `+BT_COMP_LOGI_STR("preformatted string")+`::
799 Component preformatted string info logging statement.
801 `+BT_COMP_LOGW_STR("preformatted string")+`::
802 Component preformatted string warning logging statement.
804 `+BT_COMP_LOGE_STR("preformatted string")+`::
805 Component preformatted string error logging statement.
807 `+BT_COMP_LOGF_STR("preformatted string")+`::
808 Component preformatted string fatal logging statement.
810 `+BT_COMP_LOGT_ERRNO("initial message", "format string", ...)+`::
811 Component `errno` string trace logging statement.
813 `+BT_COMP_LOGD_ERRNO("initial message", "format string", ...)+`::
814 Component `errno` string debug logging statement.
816 `+BT_COMP_LOGI_ERRNO("initial message", "format string", ...)+`::
817 Component `errno` string info logging statement.
819 `+BT_COMP_LOGW_ERRNO("initial message", "format string", ...)+`::
820 Component `errno` string warning logging statement.
822 `+BT_COMP_LOGE_ERRNO("initial message", "format string", ...)+`::
823 Component `errno` string error logging statement.
825 `+BT_COMP_LOGF_ERRNO("initial message", "format string", ...)+`::
826 Component `errno` string fatal logging statement.
828 `+BT_COMP_LOGT_MEM(data_ptr, data_size, "format string", ...)+`::
829 Component memory trace logging statement.
831 `+BT_COMP_LOGD_MEM(data_ptr, data_size, "format string", ...)+`::
832 Component memory debug logging statement.
834 `+BT_COMP_LOGI_MEM(data_ptr, data_size, "format string", ...)+`::
835 Component memory info logging statement.
837 `+BT_COMP_LOGW_MEM(data_ptr, data_size, "format string", ...)+`::
838 Component memory warning logging statement.
840 `+BT_COMP_LOGE_MEM(data_ptr, data_size, "format string", ...)+`::
841 Component memory error logging statement.
843 `+BT_COMP_LOGF_MEM(data_ptr, data_size, "format string", ...)+`::
844 Component memory fatal logging statement.
846 ==== Conditional logging
848 `+BT_LOG_IF(cond, statement)+`::
849 Execute `statement` only if `cond` is true.
856 BT_LOG_IF(i < count / 2, BT_LOGD("Log this: i=%d", i));
860 To check the <<build-time-log-level,build-time log level>>:
864 #if BT_LOG_ENABLED_DEBUG
869 This tests if the _DEBUG_ level was enabled at build time. This means
870 that the current, <<run-time-log-level,run-time log level>> _could_ be
871 _DEBUG_, but it could also be higher. The rule of thumb is to use only
872 logging statements at the same level in a `BT_LOG_ENABLED_*` conditional
875 The available definitions for build-time conditions are:
877 * `BT_LOG_ENABLED_TRACE`
878 * `BT_LOG_ENABLED_DEBUG`
879 * `BT_LOG_ENABLED_INFO`
880 * `BT_LOG_ENABLED_WARNING`
881 * `BT_LOG_ENABLED_ERROR`
882 * `BT_LOG_ENABLED_FATAL`
884 To check the current, <<run-time-log-level,run-time log level>>:
888 if (BT_LOG_ON_DEBUG) {
893 This tests if the _DEBUG_ log level is dynamically turned on
894 (implies that it's also enabled at build time). This check could have a
895 noticeable impact on performance.
897 The available definitions for run-time conditions are:
902 * `BT_LOG_ON_WARNING`
906 Those macros check the module-specific log level symbol (defined by
907 `BT_LOG_OUTPUT_LEVEL`).
909 Never, ever write code which would be executed only to compute the
910 fields of a logging statement outside a conditional logging scope,
915 int number = get_number_of_event_classes_with_property_x(...);
916 BT_LOGD("Bla bla: number=%d", number);
923 if (BT_LOG_ON_DEBUG) {
924 int number = get_number_of_event_classes_with_property_x(...);
925 BT_LOGD("Bla bla: number=%d", number);
933 BT_LOGD("Bla bla: number=%d", get_number_of_event_classes_with_property_x(...));
938 [[logging-instrument-c-file-gen]]
939 ==== Instrument a {c-cpp} source file (generic)
941 To instrument a {c-cpp} source file (`.c`/`.cpp`):
943 . At the top of the file, before the first `#include` line (if any),
944 define your file's <<choose-a-logging-tag,logging tag>> name:
949 #define BT_LOG_TAG "SUBSYS/MY-MODULE/MY-FILE"
953 . Below the line above, define the source file's log level expression,
954 `BT_LOG_OUTPUT_LEVEL`. This expression is evaluated for each
955 <<gen-logging-statements,logging statement>> to know the current
956 <<run-time-log-level,run-time log level>>.
962 /* Global log level variable */
963 #define BT_LOG_OUTPUT_LEVEL module_global_log_level
968 /* Local log level variable; must exist where you use BT_LOG*() */
969 #define BT_LOG_OUTPUT_LEVEL log_level
974 /* Object's log level; `obj` must exist where you use BT_LOG*() */
975 #define BT_LOG_OUTPUT_LEVEL (obj->log_level)
978 . Include `"logging/log.h"`:
982 #include "logging/log.h"
985 . In the file, instrument your code with the
986 <<gen-logging-statements,generic logging statement macros>>.
988 [[logging-instrument-h-file-gen]]
989 ==== Instrument a {c-cpp} header file (generic)
991 To instrument a {c-cpp} header file (`.h`/`.hpp`), if you have
992 `static inline` functions in it:
994 . Do not include `"logging/log.h"`!
998 .. In the file, instrument your code with the
999 <<gen-logging-statements,generic logging statement macros>>, making
1000 each of them conditional to the existence of the macro you're using:
1005 int some_function(int x)
1023 The {c-cpp} source files which include this header file determine if
1024 logging is enabled or not for them, and if so, what is their
1025 <<choose-a-logging-tag,logging tag>> and <<run-time-log-level,run-time
1026 log level>> expression.
1028 .. Require that logging be enabled:
1032 /* Protection: this file uses BT_LOG*() macros directly */
1033 #ifndef BT_LOG_SUPPORTED
1034 # error Please include "logging/log.h" before including this file.
1038 Then, in the file, instrument your code with the
1039 <<gen-logging-statements,generic logging statement macros>>.
1041 [[logging-instrument-c-file-lib]]
1042 ==== Instrument a library {c-cpp} source file
1044 To instrument a library {c-cpp} source file (`.c`/`.cpp`):
1046 . At the top of the file, before the first `#include` line (if any),
1047 define your file's <<choose-a-logging-tag,logging tag>> name (this
1048 tag must start with `LIB/`):
1053 #define BT_LOG_TAG "LIB/THE-FILE"
1057 . Include `"lib/logging.h"`:
1061 #include "lib/logging.h"
1064 . In the file, instrument your code with the
1065 <<lib-logging-statements,library logging statement macros>> or with
1066 the <<gen-logging-statements,generic logging statement macros>>.
1068 [[logging-instrument-h-file-lib]]
1069 ==== Instrument a library {c-cpp} header file
1071 To instrument a library {c-cpp} header file (`.h`/`.hpp`), if you have
1072 `static inline` functions in it:
1074 . Do not include `"lib/logging.h"`!
1076 . Require that library logging be enabled:
1080 /* Protection: this file uses BT_LIB_LOG*() macros directly */
1081 #ifndef BT_LIB_LOG_SUPPORTED
1082 # error Please include "lib/logging.h" before including this file.
1086 . In the file, instrument your code with the
1087 <<lib-logging-statements,library logging statement macros>> or with
1088 the <<gen-logging-statements,generic logging statement macros>>.
1090 [[logging-instrument-c-file-compcls]]
1091 ==== Instrument a component class {c-cpp} source file
1093 To instrument a component class {c-cpp} source file (`.c`/`.cpp`):
1095 . At the top of the file, before the first `#include` line (if any),
1096 define your file's <<choose-a-logging-tag,logging tag>> name (this tag
1097 must start with `PLUGIN/` followed by the component class identifier):
1102 #define BT_LOG_TAG "PLUGIN/SRC.MY-PLUGIN.MY-SRC"
1106 . Below the line above, define the source file's log level expression,
1107 `BT_LOG_OUTPUT_LEVEL`. This expression is evaluated for each
1108 <<comp-logging-statements,logging statement>> to know the current
1109 <<run-time-log-level,run-time log level>>.
1111 For a component class file, it is usually a member of a local component
1112 private structure variable:
1116 #define BT_LOG_OUTPUT_LEVEL (my_comp->log_level)
1119 . Below the line above, define `BT_COMP_LOG_SELF_COMP` to an expression
1120 which, evaluated in the context of the
1121 <<comp-logging-statements,logging statements>>, evaluates to the self
1122 component address (`+bt_self_component *+`) of the component.
1124 This is usually a member of a local component private structure
1129 #define BT_COMP_LOG_SELF_COMP (my_comp->self_comp)
1132 . Include `"logging/comp-logging.h"`:
1136 #include "logging/comp-logging.h"
1139 . In the component initialization method, make sure to set the
1140 component private structure's log level member to the initial
1141 component's log level:
1146 bt_logging_level log_level;
1150 bt_self_component_status my_comp_init(
1151 bt_self_component_source *self_comp_src,
1152 bt_value *params, void *init_method_data)
1154 struct my_comp *my_comp = g_new0(struct my_comp, 1);
1155 bt_self_component *self_comp =
1156 bt_self_component_source_as_self_component(self_comp_src);
1157 const bt_component *comp = bt_self_component_as_component(self_comp);
1160 my_comp->log_level = bt_component_get_logging_level(comp);
1166 . In the file, instrument your code with the
1167 <<comp-logging-statements,component logging statement macros>>.
1169 [[logging-instrument-h-file-compcls]]
1170 ==== Instrument a component class {c-cpp} header file
1172 To instrument a component class {c-cpp} header file (`.h`/`.hpp`), if
1173 you have `static inline` functions in it:
1175 . Do not include `"logging/comp-logging.h"`!
1177 . Require that component logging be enabled:
1181 /* Protection: this file uses BT_COMP_LOG*() macros directly */
1182 #ifndef BT_COMP_LOG_SUPPORTED
1183 # error Please include "logging/comp-logging.h" before including this file.
1187 . In the file, instrument your code with the
1188 <<comp-logging-statements,component logging statement macros>>.
1190 [[choose-a-logging-tag]]
1191 ==== Choose a logging tag
1193 Each logging-enabled {c-cpp} source file must define `BT_LOG_TAG` to a
1194 logging tag. A logging tag is a namespace to identify the logging
1195 messages of this specific source file.
1197 In general, a logging tag name _must_ be only uppercase letters, digits,
1198 and the `-`, `.`, and `/` characters.
1200 Use `/` to show the subsystem to source file hierarchy.
1202 For the {bt2} library, start with `LIB/`.
1204 For the CTF writer library, start with `CTF-WRITER/`.
1206 For component classes, use:
1209 `PLUGIN/__CCTYPE__.__PNAME__.__CCNAME__[/__FILE__]`
1214 Component class's type (`SRC`, `FLT`, or `SINK`).
1220 Component class's name.
1223 Additional information to specify the source file name or module.
1225 For plugins (files common to many component classes), use:
1228 `PLUGIN/__PNAME__[/__FILE__]`
1236 Additional information to specify the source file name or module.
1238 [[choose-a-log-level]]
1239 ==== Choose a log level
1241 Choosing the appropriate level for your logging statement is very
1244 [options="header,autowidth",cols="1,2,3a,4"]
1246 |Log level |Description |Use cases |Expected impact on performance
1250 The program, library, or plugin cannot continue to work in this
1251 condition: it must be terminated immediately.
1253 A _FATAL_-level logging statement should always be followed by
1256 * Unexpected return values from system calls.
1257 * Logic error in internal code, for example an unexpected value in a
1259 * Failed assertion (within `BT_ASSERT()`).
1260 * Unsatisfied library precondition (within `BT_ASSERT_PRE()` or
1261 `BT_ASSERT_PRE_DEV()`).
1262 * Unsatisfied library postcondition (within `BT_ASSERT_POST()` or
1263 `BT_ASSERT_POST_DEV()`).
1264 |Almost none: always enabled.
1268 An important error which is somewhat not fatal, that is, the program,
1269 library, or plugin can continue to work after this, but you judge that
1270 it should be reported to the user.
1272 Usually, the program cannot recover from such an error, but it can at
1275 * Memory allocation errors.
1276 * Wrong component initialization parameters.
1277 * Corrupted, unrecoverable trace data.
1278 * Failed to perform an operation which should work considering the
1279 implementation and the satisfied preconditions. For example, the
1280 failure to create an empty object (no parameters): most probably
1281 failed internally because of an allocation error.
1282 * Almost any error in terminal elements: CLI and plugins.
1283 |Almost none: always enabled.
1287 An error which still allows the execution to continue, but you judge
1288 that it should be reported to the user.
1290 _WARNING_-level logging statements are for any error or weird action
1291 that is directly or indirectly caused by the user, often through some
1292 bad input data. For example, not having enough memory is considered
1293 beyond the user's control, so we always log memory errors with an
1294 _ERROR_ level (not _FATAL_ because we usually don't abort in this
1297 * Missing data within something that is expected to have it, but there's
1299 * Invalid file, but recoverable/fixable.
1300 |Almost none: always enabled.
1304 Any useful information which a non-developer user would possibly
1307 Anything logged with this level must _not_ happen repetitively on the
1308 fast path, that is, nothing related to each message, for example. This
1309 level is used for sporadic and one-shot events.
1311 * CLI or component configuration report.
1312 * Successful plugin, component, or message iterator initialization.
1313 * In the library: anything related to plugins, graphs, component
1314 classes, components, message iterators, connections, and ports which
1315 is not on the fast path.
1316 * Successful connection to or disconnection from another system.
1317 * An _optional_ subsystem cannot be loaded.
1318 * An _optional_ field/datum cannot be found.
1320 Very little: always enabled.
1324 Something that only {bt2} developers would be interested into,
1325 which can occur on the fast path, but not more often than once per
1328 The _DEBUG_ level is the default <<build-time-log-level,build-time log
1329 level>> as, since it's not _too_ verbose, the performance is similar to
1332 * Object construction and destruction.
1333 * Object recycling (except fields).
1334 * Object copying (except fields and values).
1335 * Object freezing (whatever the type, as freezing only occurs in
1337 * Object interruption.
1338 * Calling user methods and logging the result.
1339 * Setting object properties (except fields and values).
1341 Noticeable, but not as much as the _TRACE_ level: could be executed
1342 in production if you're going to need a thorough log for support
1343 tickets without having to rebuild the project.
1347 Low-level debugging context information (anything that does not fit the
1348 other log levels). More appropriate for tracing in general.
1350 * Reference count change.
1351 * Fast path, low level state machine's state change.
1352 * Get or set an object's property.
1353 * Object comparison's intermediate results.
1354 |Huge: not executed in production.
1359 Make sure not to use a _WARNING_ (or higher) log level when the
1360 condition leading to the logging statement can occur under normal
1363 For example, a public function to get some object or
1364 property from an object by name or key that fails to find the value is
1365 not a warning scenario: the user could legitimately use this function to
1366 check if the name/key exists in the object. In this case, use the
1367 _TRACE_ level (or do not log at all).
1371 ==== Write an appropriate message
1373 Follow those rules when you write a logging statement's message:
1375 * Use an English sentence which starts with a capital letter.
1377 * Start the sentence with the appropriate verb tense depending on the
1378 context. For example:
1381 ** Beginning of operation (present continuous): _Creating ..._,
1382 _Copying ..._, _Serializing ..._, _Freezing ..._, _Destroying ..._
1383 ** End of operation (simple past): _Created ..._, _Successfully created ..._,
1384 _Failed to create ..._, _Set ..._ (simple past of _to set_ which is
1388 For warning and error messages, you can start the message with _Cannot_
1389 or _Failed to_ followed by a verb if it's appropriate.
1391 * Do not include the log level in the message itself. For example,
1392 do not start the message with _Error while_ or _Warning:_.
1394 * Do not put newlines, tabs, or other special characters in the message,
1395 unless you want to log a string with such characters. Note that
1396 multiline logging messages can be hard to parse, analyze, and filter,
1397 however, so prefer multiple logging statements over a single statement
1400 * **If there are fields that your logging statement must record**,
1401 follow the message with `:` followed by a space, then with the list of
1402 fields (more about this below). If there are no fields, end the
1403 sentence with a period.
1405 The statement's fields _must_ be a comma-separated list of
1406 `__name__=__value__` tokens. Keep `__name__` as simple as possible; use
1407 kebab case if possible. If `__value__` is a non-alphanumeric string, put
1408 it between double quotes (`"%s"` specifier). Always use the `PRId64` and
1409 `PRIu64` specifiers to log an `int64_t` or an `uint64_t` value. Use `%d`
1410 to log a boolean value.
1414 "Cannot read stream data for indexing: path=\"%s\", name=\"%s\", "
1415 "stream-id=%" PRIu64 ", stream-fd=%d, "
1416 "index=%" PRIu64 ", status=%s, is-mapped=%d"
1418 By following a standard format for the statement fields, it is easier to
1419 use tools like https://www.elastic.co/products/logstash[Logstash] or
1420 even https://www.splunk.com/[Splunk] to split fields and analyze logs.
1422 Prefer the following suffixes in field names:
1424 [options="header,autowidth"]
1426 |Field name suffix |Description |Format specifier
1428 |`-addr` |Memory address |`%p`
1429 |`-fd` |File descriptor |`%d`
1430 |`-fp` |File stream (`+FILE *+`) |`%p`
1431 |`-id` |Object's ID |`%" PRIu64 "`
1432 |`-index` |Index |`%" PRIu64 "`
1433 |`-name` |Object's name |`\"%s\"`
1438 The log is printed to the standard error stream. A log line contains the
1439 time, the process and thread IDs, the <<log-levels,log level>>, the
1440 <<choose-a-logging-tag,logging tag>>, the source's function name, file
1441 name and line number, and the <<message,message>>.
1443 When {bt2} supports terminal color codes (depends on the
1444 `BABELTRACE_TERM_COLOR` environment variable's value and what the
1445 standard output and error streams are plugged into), _INFO_-level lines
1446 are blue, _WARNING_-level lines are yellow, and _ERROR_-level and
1447 _FATAL_-level lines are red.
1452 05-11 00:58:03.691 23402 23402 D VALUES bt_value_destroy@values.c:498 Destroying value: addr=0xb9c3eb0
1455 You can easily filter the log with `grep` or `ag`. For example, to keep
1456 only the _DEBUG_-level logging messages that the `FIELD-CLASS` module
1460 $ babeltrace2 --log-level=D /path/to/trace |& ag 'D FIELD-CLASS'
1465 To use Valgrind on an application (for example, the CLI or a test) which
1466 loads libbabeltrace2, use:
1469 $ G_SLICE=always-malloc G_DEBUG=gc-friendly PYTHONMALLOC=malloc \
1470 LIBBABELTRACE2_NO_DLCLOSE=1 valgrind --leak-check=full app
1473 `G_SLICE=always-malloc` and `G_DEBUG=gc-friendly` is for GLib and
1474 `PYTHONMALLOC=malloc` is for the Python interpreter, if it is used by
1475 the Python plugin provider (Valgrind will probably show a lot of errors
1476 which originate from the Python interpreter anyway).
1478 `LIBBABELTRACE2_NO_DLCLOSE=1` makes libbabeltrace2 not close the shared
1479 libraries (plugins) which it loads. You need this to see the appropriate
1480 backtrace when Valgrind shows errors.
1487 Running `make check` in the build directory (regardless of whether the build is
1488 in-tree or out-of-tree) automatically sets up the appropriate environment for
1489 tests to run in, so nothing more is needed.
1491 If building in-tree, you can run single tests from the tree directly:
1494 $ ./tests/plugins/sink.text.pretty/test-enum.sh
1497 If building out-of-tree, you can get the appropriate environment by sourcing
1498 the `tests/utils/env.sh` file residing in the build directory against which you
1502 $ source /path/to/my/build/tests/utils/env.sh
1503 $ ./tests/plugins/sink.text.pretty/test-enum.sh
1508 You can use the `tests/utils/run-python-bt2.sh` script to run any
1509 command within an environment making the build's `bt2` Python package
1512 `run-python-bt2.sh` uses <<test-env,`utils.sh`>> which needs to know the
1513 build directory, so make sure you set the `BT_TESTS_BUILDDIR`
1514 environment variable correctly _if you build out of tree_, for example:
1517 $ export BT_TESTS_BUILDDIR=/path/to/build/babeltrace/tests
1520 You can run any command which needs the `bt2` Python package through
1521 `run-python-bt2.sh`, for example:
1524 $ ./tests/utils/run-python-bt2.sh ipython3
1529 All test scripts output the test results following the
1530 https://testanything.org/[Test Anything Protocol] (TAP) format.
1532 The TAP format has two mechanisms to print additional information about
1535 * Print a line starting with `#` to the standard output.
1537 This is usually done with the `diag()` C function or the `diag` shell
1540 * Print to the standard error.
1544 The `bt2` Python package tests are located in
1545 `tests/bindings/python/bt2`.
1547 ==== Python test runner
1549 `tests/utils/python/testrunner.py` is {bt2}'s Python test runner
1550 which loads Python files containing unit tests, finds all the test
1551 cases, and runs the tests, producing a TAP report.
1553 You can see the test runner command's help with:
1556 $ python3 ./tests/utils/python/testrunner.py --help
1559 By default, the test runner reports failing tests (TAP's `not{nbsp}ok`
1560 line), but continues to run other tests. You can use the `--failfast`
1561 option to make the test runner fail as soon as a test fails.
1565 To run all the `bt2` Python package tests:
1570 $ ./tests/utils/run-python-bt2.sh ./tests/bindings/python/bt2/test-python-bt2.sh
1576 $ ./tests/utils/run-python-bt2.sh python3 ./tests/utils/python/testrunner.py \
1577 ./tests/bindings/python/bt2/ -p '*.py'
1580 To run **all the tests** in a test module (for example,
1586 $ ./tests/utils/run-python-bt2.sh python3 ./tests/utils/python/testrunner.py \
1587 ./tests/bindings/python/bt2 -t test_value
1590 To run a **specific test case** (for example, `RealValueTestCase` within
1596 $ ./tests/utils/run-python-bt2.sh python3 ./tests/utils/python/testrunner.py \
1597 ./tests/bindings/python/bt2/ -t test_value.RealValueTestCase
1600 To run a **specific test** (for example,
1601 `RealValueTestCase.test_assign_pos_int` within `test_value.py`):
1606 $ ./tests/utils/run-python-bt2.sh python3 ./tests/utils/python/testrunner.py \
1607 ./tests/bindings/python/bt2/ -t test_value.RealValueTestCase.test_assign_pos_int
1612 Some parts of {bt2} are written in {cpp}.
1614 This section shows what's important to know about {cpp} to contribute
1619 {bt2} only has {cpp} sources for _internal_ code.
1621 In other words, libbabeltrace2 _must_ expose a pure C99 API to preserve
1622 ABI compatibility over time.
1625 === Standard and dependencies
1627 The {bt2} project is configured to use the {cpp11} standard.
1629 {cpp11} makes it possible to build {bt2} with a broad range of
1630 compilers, from GCC{nbsp}4.8 and Clang{nbsp}3.3.
1632 === Automake/Libtool requirements
1634 To add a {cpp} source file to a part of the project, use the `.cpp`
1635 extension and add it to the list of source files in `Makefile.am` as
1638 If a program or a shared library has a direct {cpp} source file, then
1639 Libtool uses the {cpp} linker to create the result, dynamically
1640 linking important runtime libraries such as libstdc++ and libgcc_s.
1642 Because a Libtool _convenience library_ is just an archive (`.a`), it's
1643 _not_ dynamically linked to runtime libraries, even if one of its direct
1644 sources is a {cpp} file. This means that for each program or shared
1645 library named `my_target` in `Makefile.am` which is linked to a
1646 convenience library having {cpp} sources (recursively), you _must_ do
1649 * Have at least one direct {cpp} source file in the
1650 `+*_my_target_SOURCES+` list.
1655 nodist_EXTRA_my_target_SOURCES = dummy.cpp
1659 https://www.gnu.org/software/automake/manual/automake.html#Libtool-Convenience-Libraries[Libtool
1660 Convenience Libraries] to learn more.
1662 For a given program or library, you _cannot_ have a C{nbsp}file and a
1663 {cpp}{nbsp}file having the same name, for example `list.c` and
1668 ==== Whitespaces, indentation, and line breaks
1670 All the project's {cpp} files follow the
1671 https://clang.llvm.org/docs/ClangFormat.html[clang-format]
1672 https://clang.llvm.org/docs/ClangFormatStyleOptions.html[style] of the
1673 `.clang-format` file for whitespaces, indentation, and line breaks.
1675 You _must_ format modified and new {cpp} files with clang-format before
1676 you create a contribution patch.
1678 You need clang-format{nbsp}15 to use the project's `.clang-format` file.
1680 To automatically format all the project's {cpp} files, run:
1683 $ ./tools/format-cpp.sh
1686 Pass a directory path to only format the {cpp} files it contains:
1689 $ ./tools/format-cpp.sh ./src/cli
1692 Use the `FORMATTER` environment variable to override the default
1693 formatter (`clang-format{nbsp}-i`):
1696 $ FORMATTER='my-clang-format-15 -i' ./tools/format-cpp.sh
1701 * Use camel case with a lowercase first letter for:
1702 ** Variable names: `size`, `objSize`.
1703 ** Function/method names: `size()`, `formatAndPrint()`.
1705 * Use camel case with an uppercase first letter for:
1706 ** Types: `Pistachio`, `NutManager`.
1707 ** Template parameters: `PlanetT`, `TotalSize`.
1709 * Use snake case with uppercase letters for:
1710 ** Definition/macro names: `MARK_AS_UNUSED()`, `SOME_FEATURE_EXISTS`.
1711 ** Enumerators: `Type::SIGNED_INT`, `Scope::FUNCTION`.
1713 * Use only lowercase letters and digits for namespaces: `mylib`, `bt2`.
1715 * Use the suffix `T` for type template parameters:
1719 template <typename NameT, typename ItemT>
1722 * Name a template parameter pack `Args`.
1726 template <typename NameT, typename... Args>
1729 * Use an underscore prefix for private and protected methods and member
1730 type names: `_tryConnect()`, `_NodeType`.
1732 * Use the prefix `_m` for private and protected member variable names:
1733 `_mLogger`, `_mSize`, `_mFieldClass`.
1735 * Name setters and getters like the property name, without `set` and
1738 * Use the `is` or `has` prefix, if possible, to name the functions which
1741 === Coding convention
1743 In general, the project's contributors make an effort to follow,
1747 https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md[{cpp} Core Guidelines].
1750 "`https://www.oreilly.com/library/view/effective-modern-c/9781491908419/[Effective Modern {cpp}]`".
1752 Here are a few important reminders:
1754 * Namespace your code.
1756 * Create one header/source file pair per class when possible.
1758 For a class named `MyClass`, name the corresponding files `my-class.hpp`
1761 * When defining a class, put constructors as the first methods, whatever
1762 their access (public/protected/private), then the destructor, and then
1765 * Declare variables as close to where they are used as possible.
1767 * Use `auto` when possible.
1769 * Use `const` as much as possible, even for pointer
1770 (`+const char* const+`) and numeric values (`const unsigned int`)
1771 which never need to change.
1773 * Implement simple setters, getters, and one-liners in header files and
1774 everything else that's not a template in source files.
1776 * Make methods `const noexcept` or `const` as much as possible.
1778 * Make constructors `explicit` unless you really need an implicit
1779 constructor (which is rare).
1781 * Use `std::unique_ptr` to manage memory when possible.
1783 However, use references (`+*my_unique_ptr+`) and raw pointers
1784 (`+my_unique_ptr.get()+`) when not transferring ownership.
1786 * Use `nullptr`, not `NULL` nor 0.
1788 * Return by value (rvalue) instead of by output parameter (non-const
1789 lvalue reference), even complex objects, unless you can prove that the
1790 performance is improved when returning by parameter.
1792 * For a function parameter or a return value of which the type needs to
1793 be a reference or pointer, use:
1795 If the value is mandatory:::
1797 If the value is optional:::
1800 * Don't use `+std::move()+` when you already have an rvalue, which
1802 ** Don't write `+return std::move(...);+` as this can interfere with
1804 ** Don't use `+std::move()+` with a function call
1805 (`+std::move(func())+`).
1807 * For each possible move/copy constructor or assignment operator, do one
1809 ** Write a custom one.
1810 ** Mark it as defaulted (`default`)
1811 ** Mark it as deleted (`delete`).
1813 * Use scoped enumerations (`+enum class+`).
1815 * Mark classes known to be final with the `final` keyword.
1817 * Use type aliases (`using`), not type definitions (`typedef`).
1819 * Use anonymous namespaces for local functions instead of `static`.
1821 * Don't pollute the global namespace:
1822 ** Don't use `using namespace xyz` anywhere.
1823 ** Use only namespace aliases in source files (`.cpp`), trying to
1824 use them in the smallest possible scope (function, or even smaller).
1826 * Return a structure with named members instead of a generic container
1827 such as `std::pair` or `std::tuple`.
1829 * When a class inherits a base class with virtual methods, use the
1830 `override` keyword to mark overridden virtual methods, and do not use
1831 the `virtual` keyword again.
1833 * Define overloaded operators only if their meaning is obvious,
1834 unsurprising, and consistent with the corresponding built-in
1837 For example, use `+|+` as a bitwise- or logical-or, not as a shell-style
1840 * Use RAII wrappers when managing system resources or interacting with
1843 In other words, don't rely on ``goto``s and error labels to clean up as
1844 you would do in{nbsp}C.
1848 * Throw an exception when there's an unexpected, exceptional condition,
1849 https://isocpp.org/wiki/faq/exceptions#ctors-can-throw[including from
1850 a constructor], instead of returning a status code.
1852 * Accept a by-value parameter and move it (when it's moveable) when you
1853 intend to copy it anyway.
1855 You can do this with most STL containers.
1861 void Obj::doSomething(std::string str)
1863 _mName = std::move(str);
1870 This example shows a {cpp} header which follows the {bt2} {cpp} coding
1876 * SPDX-License-Identifier: MIT
1878 * Copyright 2020 Harry Burnett <hburnett@reese.choco>
1881 #ifndef BABELTRACE_BABY_HPP
1882 #define BABELTRACE_BABY_HPP
1885 #include <unordered_set>
1893 * A baby is a little human.
1895 class Baby : public Human
1898 using Toys = std::unordered_set<Toy>;
1908 explicit Baby(const Toys& toys);
1909 Baby(const Baby&) = delete;
1910 Baby(Baby&&) = delete;
1911 Baby& operator=(const Baby&) = delete;
1912 Baby& operator=(Baby&&) = delete;
1915 explicit Baby(Gender initialGender = Gender::UNKNOWN);
1919 * Eats `weight` grams of food.
1921 void eat(unsigned long weight);
1924 * Sleeps for `duration` seconds.
1926 void sleep(double duration);
1929 * Sets this baby's name to `name`.
1931 void name(std::string name)
1933 _mName = std::move(name);
1939 const std::string& name() const noexcept
1945 void _addTeeth(unsigned long index);
1946 void _grow(double size) override;
1949 std::string _mName {"Paul"};
1955 #endif // BABELTRACE_BABY_HPP
1963 All Python code must be formatted using the version of
1964 https://github.com/psf/black[Black] specified in `dev-requirements.txt`.
1966 All Python imports must be sorted using the version of
1967 https://pycqa.github.io/isort/[isort] indicated in `dev-requirements.txt`.