Fix: lib: pass down API function name to some helpers
[babeltrace.git] / CONTRIBUTING.adoc
CommitLineData
cd4aac1e
PP
1// Render with Asciidoctor
2
28b765ac 3= Babeltrace{nbsp}2 contributor's guide
f9ded0e0 4Jérémie Galarneau, Philippe Proulx
fb7ff115 51 December 2020
28b765ac
PP
6:toc: left
7:toclevels: 3
fb7ff115
PP
8:icons: font
9:nofooter:
28b765ac 10:bt2: Babeltrace{nbsp}2
fb7ff115
PP
11:c-cpp: C/{cpp}
12:cpp11: {cpp}11
beb0fb75 13
f9ded0e0 14This is a partial contributor's guide for the
28b765ac 15https://babeltrace.org[{bt2}] project. If you have any
cd4aac1e
PP
16questions that are not answered by this guide, please post them on
17https://lists.lttng.org/cgi-bin/mailman/listinfo/lttng-dev[Babeltrace's
18mailing list].
19
28b765ac 20== {bt2} library
f9ded0e0 21
cd4aac1e 22=== Object reference counting and lifetime
f9ded0e0 23
28b765ac
PP
24This section covers the rationale behind the design of {bt2}'s
25object lifetime management. This applies to the {bt2} library, as
cd4aac1e
PP
26well as to the CTF writer library (although the public reference
27counting functions are not named the same way).
f9ded0e0 28
28b765ac
PP
29Starting from Babeltrace{nbsp}2.0, all publicly exposed objects inherit
30a common base: `bt_object`. This base provides a number of facilities to
f9ded0e0
PP
31all objects, chief amongst which are lifetime management functions.
32
cd4aac1e
PP
33The lifetime of some public objects is managed by reference counting. In
34this case, the API offers the `+bt_*_get_ref()+` and `+bt_*_put_ref()+`
35functions which respectively increment and decrement an object's
36reference count.
f9ded0e0 37
28b765ac 38As far as lifetime management in concerned, {bt2} makes a clear
f9ded0e0
PP
39distinction between regular objects, which have a single parent, and
40root objects, which don't.
41
cd4aac1e 42==== The problem
f9ded0e0
PP
43
44Let us consider a problematic case to illustrate the need for this
45distinction.
46
28b765ac 47A user of the {bt2} library creates a trace class, which _has_ a
cd4aac1e
PP
48stream class (the class of a stream) and that stream class, in turn,
49_has_ an event class (the class of an event).
f9ded0e0
PP
50
51Nothing prevents this user from releasing his reference on any one of
52these objects in any order. However, all objects in the
53__trace--stream class--event class__ hierarchy can be retrieved
54from any other.
55
56For instance, the user could discard his reference on both the event
cd4aac1e
PP
57class and the stream class, only keeping a reference on the trace class.
58From this trace class reference, stream classes can be enumerated,
59providing the user with a new reference to the stream class he discarded
60earlier. Event classes can also be enumerated from stream classes,
61providing the user with references to the individual event classes.
f9ded0e0
PP
62
63Conversely, the user could also hold a reference to an event class and
cd4aac1e 64retrieve its parent stream class. The trace class, in turn, can then be
f9ded0e0
PP
65retrieved from the stream class.
66
67This example illustrates what could be interpreted as a circular
68reference dependency existing between these objects. Of course, if the
69objects in such a scenario were to hold references to each other (in
70both directions), we would be in presence of a circular ownership
71resulting in a leak of both objects as their reference counts would
72never reach zero.
73
74Nonetheless, the API must offer the guarantee that holding a node to any
75node of the graph keeps all other reachable nodes alive.
76
cd4aac1e 77==== The solution
f9ded0e0 78
28b765ac 79The scheme employed in {bt2} to break this cycle consists in the
cd4aac1e 80"children" holding _reverse component references_ to their parents. That
56e18c4c
PP
81is, in the context of the trace IR, that event classes hold a reference
82to their parent stream class and stream classes hold a reference to
cd4aac1e 83their parent trace class.
f9ded0e0 84
cd4aac1e 85On the other hand, parents hold _claiming aggregation references_ to
f9ded0e0
PP
86their children. A claiming aggregation reference means that the object
87being referenced should not be deleted as long as the reference still
88exists. In this respect, it can be said that parents truly hold the
89ownership of their children, since they control their lifetime.
90Conversely, the reference counting mechanism is leveraged by children to
91notify parents that no other child indirectly exposes the parent.
92
93When a parented object's reference count reaches zero, it invokes
cd4aac1e
PP
94`+bt_*_put_ref()+` on its parent and does _not_ free itself. However,
95from that point, the object depends on its parent to signal the moment
96when it can be safely reclaimed.
97
98The invocation of `+bt_*_put_ref()+` by the last children holding a
99reference to its parent might trigger a cascade of `+bt_*_put_ref()+`
100from child to parent. Eventually, a **root** object is reached. At that
101point, if this orphaned object's reference count reaches zero, the
102object invokes the destructor method defined by everyone of its children
103as part of their base `struct bt_object`. The key point here is that the
104cascade of destructor will necessarily originate from the root and
105propagate in preorder to the children. These children will propagate the
106destruction to their own children before reclaiming their own memory.
107This ensures that a node's pointer to its parent is _always_ valid since
108the parent has the responsibility of tearing-down their children before
109cleaning themselves up.
110
111Assuming a reference to an object is _acquired_ by calling
112`+bt_*_get_ref()+` while its reference count is zero, the object
113acquires, in turn, a reference on its parent using `+bt_*_get_ref()+`.
114At that point, the child can be thought of as having converted its weak
115reference to its parent into a regular reference. That is why this
116reference is referred to as a _claiming_ aggregation reference.
117
cd4aac1e 118==== Caveats
f9ded0e0
PP
119
120This scheme imposes a number of strict rules defining the relation
121between objects:
122
123* Objects may only have one parent.
124* Objects, beside the root, are only retrievable from their direct
125 parent or children.
126
cd4aac1e 127==== Example
f9ded0e0 128
cd4aac1e
PP
129The initial situation is rather simple: **User{nbsp}A** is holding a
130reference to a trace class, **TC1**. As per the rules previously
131enounced, stream classes **SC1** and **SC2** don't hold a reference to
132**TC1** since their own reference counts are zero. The same holds true
133for **EC1**, **EC2** and **EC3** with respect to **SC1** and **SC2**.
f9ded0e0
PP
134
135image::doc/contributing-images/bt-ref01.png[]
136
cd4aac1e
PP
137In this second step, we can see that **User{nbsp}A** has acquired a
138reference on **SC2** through the trace class, **TC1**.
f9ded0e0
PP
139
140The stream class's reference count transitions from zero to one,
141triggering the acquisition of a strong reference on **TC1** from
142**SC2**.
143
cd4aac1e
PP
144Hence, at this point, the trace class's ownership is shared by
145**User{nbsp}A** and **SC2**.
f9ded0e0
PP
146
147image::doc/contributing-images/bt-ref02.png[]
148
cd4aac1e
PP
149Next, **User{nbsp}A** acquires a reference on the **EC3** event class
150through its parent stream class, **SC2**. Again, the transition of an
151object's reference count from 0 to 1 triggers the acquisition of a
152reference on its parent.
f9ded0e0 153
cd4aac1e 154Note that SC2's reference count was incremented to 2. The trace class's
f9ded0e0
PP
155reference count remains unchanged.
156
157image::doc/contributing-images/bt-ref03.png[]
158
cd4aac1e
PP
159**User{nbsp}A** decides to drop its reference on **SC2**. **SC2**'s
160reference count returns back to 1, everything else remaining unchanged.
f9ded0e0
PP
161
162image::doc/contributing-images/bt-ref04.png[]
163
cd4aac1e
PP
164**User{nbsp}A** can then decide to drop its reference on the trace
165class. This results in a reversal of the initial situation:
166**User{nbsp}A** now owns an event, **EC3**, which is keeping everything
167else alive and reachable.
f9ded0e0
PP
168
169image::doc/contributing-images/bt-ref05.png[]
170
cd4aac1e 171If another object, **User{nbsp}B**, enters the picture and acquires a
f9ded0e0
PP
172reference on the **SC1** stream class, we see that **SC1**'s reference
173count transitioned from 0 to 1, triggering the acquisition of a
174reference on **TC1**.
175
176image::doc/contributing-images/bt-ref06.png[]
177
cd4aac1e
PP
178**User{nbsp}B** hands off a reference to **EC1**, acquired through
179**SC1**, to another object, **User{nbsp}C**. The acquisition of a
180reference on **EC1**, which transitions from 0 to 1, triggers the
181acquisition of a reference on its parent, **SC1**.
f9ded0e0
PP
182
183image::doc/contributing-images/bt-ref07.png[]
184
cd4aac1e 185At some point, **User{nbsp}A** releases its reference on **EC3**. Since
f9ded0e0
PP
186**EC3**'s reference count transitions to zero, it releases its reference
187on **SC2**. **SC2**'s reference count, in turn, reaches zero and it
188releases its reference to **TC1**.
189
190**TC1**'s reference count is now 1 and no further action is taken.
191
192image::doc/contributing-images/bt-ref08.png[]
193
cd4aac1e
PP
194**User{nbsp}B** releases its reference on **SC1**. **User{nbsp}C**
195becomes the sole owner of the whole hierarchy through his ownership of
196**EC1**.
f9ded0e0
PP
197
198image::doc/contributing-images/bt-ref09.png[]
199
cd4aac1e
PP
200Finally, **User{nbsp}C** releases his ownership of **EC1**, triggering
201the release of the whole hierarchy. Let's walk through the reclamation
202of the whole graph.
f9ded0e0 203
cd4aac1e
PP
204Mirroring what happened when **User{nbsp}A** released its last reference
205on **EC3**, the release of **EC1** by **User{nbsp}C** causes its
206reference count to fall to zero.
f9ded0e0
PP
207
208This transition to zero causes **EC1** to release its reference on
209**SC1**. **SC1**'s reference count reaching zero causes it to release
210its reference on **TC1**.
211
212image::doc/contributing-images/bt-ref10.png[]
213
214Since the reference count of **TC1**, a root object, has reached zero,
215it invokes the destructor method on its children. This method is
216recursive and causes the stream classes to call the destructor method on
217their event classes.
218
219The event classes are reached and, having no children of their own, are
220reclaimed.
221
222image::doc/contributing-images/bt-ref11.png[]
223
224The stream classes having destroyed their children, are then reclaimed
cd4aac1e 225by the trace class.
f9ded0e0
PP
226
227image::doc/contributing-images/bt-ref12.png[]
228
229Finally, the stream classes having been reclaimed, **TC1** is reclaimed.
230
231image::doc/contributing-images/bt-ref13.png[]
232
233
f9ded0e0 234== Logging
beb0fb75
PP
235
236Logging is a great instrument for a developer to be able to collect
237information about a running software.
238
28b765ac 239{bt2} is a complex software with many layers. When a {bt2}
beb0fb75 240graph fails to run, what caused the failure? It could be caused by any
cd4aac1e
PP
241component, any message iterator, and any deeply nested validation of a
242CTF IR object (within the `ctf` plugin), for example. With the
243appropriate logging statements manually placed in the source code, we
244can find the cause of a bug faster.
beb0fb75 245
cd4aac1e
PP
246While <<choose-a-log-level,care must be taken>> when placing _DEBUG_ to
247_FATAL_ logging statements, you should liberally instrument your
28b765ac 248{bt2} module with _TRACE_ logging statements to help future you
cd4aac1e 249and other developers understand what's happening at run time.
beb0fb75 250
f9ded0e0 251=== Logging API
beb0fb75 252
28b765ac 253The {bt2} logging API is internal: it is not exposed to the users
cd4aac1e 254of the library; only to their developers. The only thing that a library
4a41523d 255user can control is the current log level of the library itself with
beb0fb75 256`bt_logging_set_global_level()` and the initial library's log level with
cd4aac1e 257the `LIBBABELTRACE2_INIT_LOG_LEVEL` environment variable.
beb0fb75
PP
258
259This API is based on https://github.com/wonder-mice/zf_log[zf_log], a
260lightweight, yet featureful, MIT-licensed core logging library for C and
fb7ff115 261{cpp}. The zf_log source files were modified to have the `BT_` and
cd4aac1e
PP
262`bt_` prefixes, and other small changes, like color support and using
263the project's `BT_DEBUG_MODE` definition instead of the standard
264`NDEBUG`.
beb0fb75
PP
265
266The logging functions are implemented in the logging convenience
cd4aac1e 267library (`src/logging` directory).
beb0fb75 268
cd4aac1e 269[[logging-headers]]
f9ded0e0 270==== Headers
beb0fb75
PP
271
272The logging API headers are:
273
3fadfbc0 274`<babeltrace2/logging.h>`::
cd4aac1e 275 Public header which a library user can use to set and get
a12f3d62 276 libbabeltrace2's current log level.
beb0fb75 277
cd4aac1e 278`"logging/log.h"`::
28b765ac 279 Internal, generic logging API which you can use in any {bt2}
cd4aac1e
PP
280 module. This is the translation of `zf_log.h`.
281+
282This header offers the <<gen-logging-statements,generic logging
283statement macros>>.
beb0fb75 284
cd4aac1e
PP
285`"lib/logging.h"`::
286 Specific internal header to use within the library.
287+
288This header defines `BT_LOG_OUTPUT_LEVEL` to a custom, library-wide
289hidden symbol which is the library's current log level before including
290`"logging/log.h"`.
291+
292This header offers the <<lib-logging-statements,library-specific logging
293statement macros>>.
beb0fb75 294
d9c39b0a 295`"logging/comp-logging.h"`::
cd4aac1e
PP
296 Specific internal header to use within a component class.
297+
298This header offers the <<comp-logging-statements,component-specific
299logging statement macros>>.
beb0fb75 300
cd4aac1e 301[[log-levels]]
f9ded0e0 302==== Log levels
beb0fb75 303
cd4aac1e
PP
304The internal logging API offers the following log levels, in ascending
305order of severity:
beb0fb75 306
cd4aac1e
PP
307[options="header,autowidth",cols="4"]
308|===
309|Log level name
310|Log level short name
311|Internal API enumerator
312|Public API enumerator
313
ef267d12
PP
314|_TRACE_
315|`T`
316|`BT_LOG_TRACE`
317|`BT_LOGGING_LEVEL_TRACE`
cd4aac1e
PP
318
319|_DEBUG_
320|`D`
321|`BT_LOG_DEBUG`
322|`BT_LOGGING_LEVEL_DEBUG`
323
324|_INFO_
325|`I`
326|`BT_LOG_INFO`
327|`BT_LOGGING_LEVEL_INFO`
328
770538dd 329|_WARNING_
cd4aac1e 330|`W`
770538dd
PP
331|`BT_LOG_WARNING`
332|`BT_LOGGING_LEVEL_WARNING`
cd4aac1e
PP
333
334|_ERROR_
335|`E`
336|`BT_LOG_ERROR`
337|`BT_LOGGING_LEVEL_ERROR`
338
339|_FATAL_
340|`F`
341|`BT_LOG_FATAL`
342|`BT_LOGGING_LEVEL_FATAL`
343
344|_NONE_
345|`N`
346|`BT_LOG_NONE`
347|`BT_LOGGING_LEVEL_NONE`
348|===
349
350The short name is accepted by the log level environment variables and by
351the CLI's `--log-level` options.
beb0fb75 352
cd4aac1e 353See <<choose-a-log-level,how to decide which one to use>> below.
beb0fb75 354
cd4aac1e 355There are two important log level expressions:
beb0fb75
PP
356
357[[build-time-log-level]]Build-time, minimal log level::
cd4aac1e
PP
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.).
beb0fb75
PP
362+
363All the logging statements with a level below this level are **not built
364at all**. All the logging statements with a level equal to or greater
f9ded0e0
PP
365than this level _can_ be executed, depending on the
366<<run-time-log-level,run-time log level>>.
beb0fb75
PP
367+
368You can set this level at configuration time with the
369`BABELTRACE_MINIMAL_LOG_LEVEL` environment variable, for example:
370+
371--
372----
770538dd 373$ BABELTRACE_MINIMAL_LOG_LEVEL=INFO ./configure
beb0fb75
PP
374----
375--
376+
cd4aac1e
PP
377The default build-time log level is `DEBUG`. For optimal performance,
378set it to `INFO`, which effectively disables all fast path logging in
28b765ac 379all the {bt2} modules. You can't set it to `WARNING`, `ERROR`,
dd22a91f 380`FATAL`, or `NONE` because the impact on performance is minuscule
28b765ac 381starting from the _INFO_ log level anyway and we want any {bt2}
dd22a91f 382build to always be able to print _INFO_-level logs.
beb0fb75
PP
383+
384The library's public API provides `bt_logging_get_minimal_level()` to
385get the configured minimal log level.
386
387[[run-time-log-level]]Run-time, dynamic log level::
cd4aac1e
PP
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>>.
beb0fb75 394+
f9ded0e0
PP
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
397symbol, and always make sure that `BT_LOG_OUTPUT_LEVEL` is defined to a
cd4aac1e
PP
398module-wise expression before including `"logging/log.h"`.
399+
400In the library, `"lib/logging.h"` defines its own
401`BT_LOG_OUTPUT_LEVEL` to the library's log level symbol before it
402includes `"logging/log.h"` itself.
beb0fb75 403+
a12f3d62 404In libbabeltrace2, the user can set the current run-time log level with
beb0fb75
PP
405the `bt_logging_set_global_level()` function, for example:
406+
407--
408[source,c]
409----
410bt_logging_set_global_level(BT_LOGGING_LEVEL_INFO);
411----
412--
413+
414The library's initial run-time log level is defined by the
cd4aac1e
PP
415`LIBBABELTRACE2_INIT_LOG_LEVEL` environment variable, or set to _NONE_
416if this environment variable is undefined.
417+
418Other modules have their own way of setting their run-time log level.
419+
420For example, the CLI uses the `BABELTRACE_CLI_LOG_LEVEL` environment
421variable, as well as its global `--log-level` option:
422+
423----
424$ babeltrace2 --log-level=I ...
425----
426+
427The components use their own log level (as returned by
428`bt_component_get_logging_level()`). With the CLI, you can set a
429specific component's log level with its own, position-dependent
430`--log-level` option:
431+
432----
433$ babeltrace2 /path/to/trace -c sink.ctf.fs --log-level=D
434----
435+
436Code which is common to the whole project, for example `src/common`
437and `src/compat`, use function parameters to get its run-time log
438level, for example:
439+
440[source,c]
441----
442BT_HIDDEN
443char *bt_common_get_home_plugin_path(int log_level);
444----
beb0fb75 445+
cd4aac1e
PP
446Typically, when a logging-enabled module calls such a function, it
447passes its own log level expression directly (`BT_LOG_OUTPUT_LEVEL`):
beb0fb75 448+
cd4aac1e
PP
449[source,c]
450----
451path = bt_common_get_home_plugin_path(BT_LOG_OUTPUT_LEVEL);
452----
453+
454Otherwise, just pass `BT_LOG_NONE`:
455+
456----
457path = bt_common_get_home_plugin_path(BT_LOG_NONE);
458----
459
cd4aac1e
PP
460[[gen-logging-statements]]
461==== Generic logging statement macros
462
28b765ac 463The {bt2} logging statement macros work just like `printf()`
cd4aac1e
PP
464(except the `+BT_LOG*_STR()+` ones) and contain their <<log-levels,log
465level>> (short name) in their name.
beb0fb75 466
cd4aac1e
PP
467Each of the following macros evaluate the
468<<build-time-log-level,build-time log level>> definition and
469<<run-time-log-level,run-time log level>> expression (as defined by
470`BT_LOG_OUTPUT_LEVEL`) to log conditionally.
beb0fb75 471
28b765ac
PP
472See <<logging-instrument-c-file-gen>> and
473<<logging-instrument-h-file-gen>> to learn how to be able to use the
474following macros.
beb0fb75 475
ef267d12
PP
476`+BT_LOGT("format string", ...)+`::
477 Generic trace logging statement.
beb0fb75 478
cd4aac1e
PP
479`+BT_LOGD("format string", ...)+`::
480 Generic debug logging statement.
beb0fb75 481
cd4aac1e
PP
482`+BT_LOGI("format string", ...)+`::
483 Generic info logging statement.
beb0fb75 484
cd4aac1e
PP
485`+BT_LOGW("format string", ...)+`::
486 Generic warning logging statement.
beb0fb75 487
cd4aac1e
PP
488`+BT_LOGE("format string", ...)+`::
489 Generic error logging statement.
beb0fb75 490
cd4aac1e
PP
491`+BT_LOGF("format string", ...)+`::
492 Generic fatal logging statement.
beb0fb75 493
ef267d12
PP
494`+BT_LOGT_STR("preformatted string")+`::
495 Generic preformatted string trace logging statement.
beb0fb75 496
cd4aac1e
PP
497`+BT_LOGD_STR("preformatted string")+`::
498 Generic preformatted string debug logging statement.
beb0fb75 499
cd4aac1e
PP
500`+BT_LOGI_STR("preformatted string")+`::
501 Generic preformatted string info logging statement.
beb0fb75 502
cd4aac1e
PP
503`+BT_LOGW_STR("preformatted string")+`::
504 Generic preformatted string warning logging statement.
beb0fb75 505
cd4aac1e
PP
506`+BT_LOGE_STR("preformatted string")+`::
507 Generic preformatted string error logging statement.
beb0fb75 508
cd4aac1e
PP
509`+BT_LOGF_STR("preformatted string")+`::
510 Generic preformatted string fatal logging statement.
beb0fb75 511
ef267d12
PP
512`+BT_LOGT_MEM(data_ptr, data_size, "format string", ...)+`::
513 Generic memory trace logging statement.
beb0fb75 514
cd4aac1e
PP
515`+BT_LOGD_MEM(data_ptr, data_size, "format string", ...)+`::
516 Generic memory debug logging statement.
b81626f9 517
cd4aac1e
PP
518`+BT_LOGI_MEM(data_ptr, data_size, "format string", ...)+`::
519 Generic memory info logging statement.
b81626f9 520
cd4aac1e
PP
521`+BT_LOGW_MEM(data_ptr, data_size, "format string", ...)+`::
522 Generic memory warning logging statement.
b81626f9 523
cd4aac1e
PP
524`+BT_LOGE_MEM(data_ptr, data_size, "format string", ...)+`::
525 Generic memory error logging statement.
b81626f9 526
cd4aac1e
PP
527`+BT_LOGF_MEM(data_ptr, data_size, "format string", ...)+`::
528 Generic memory fatal logging statement.
b81626f9 529
ef267d12
PP
530`+BT_LOGT_ERRNO("initial message", "format string", ...)+`::
531 Generic `errno` string trace logging statement.
b81626f9 532
cd4aac1e
PP
533`+BT_LOGD_ERRNO("initial message", "format string", ...)+`::
534 Generic `errno` string debug logging statement.
beb0fb75 535
cd4aac1e
PP
536`+BT_LOGI_ERRNO("initial message", "format string", ...)+`::
537 Generic `errno` string info logging statement.
beb0fb75 538
cd4aac1e
PP
539`+BT_LOGW_ERRNO("initial message", "format string", ...)+`::
540 Generic `errno` string warning logging statement.
beb0fb75 541
cd4aac1e
PP
542`+BT_LOGE_ERRNO("initial message", "format string", ...)+`::
543 Generic `errno` string error logging statement.
beb0fb75 544
cd4aac1e
PP
545`+BT_LOGF_ERRNO("initial message", "format string", ...)+`::
546 Generic `errno` string fatal logging statement.
beb0fb75 547
cd4aac1e
PP
548[[lib-logging-statements]]
549==== Library-specific logging statement macros
550
28b765ac
PP
551The {bt2} library contains an internal logging API based on the
552generic logging framework. You can use it to log known {bt2}
cd4aac1e
PP
553objects without having to manually log each member.
554
28b765ac
PP
555See <<logging-instrument-c-file-lib>>
556and <<logging-instrument-h-file-lib>> to
cd4aac1e
PP
557learn how to be able to use the following macros.
558
559The library logging statement macros are named `+BT_LIB_LOG*()+` instead
560of `+BT_LOG*()+`:
561
ef267d12
PP
562`+BT_LIB_LOGT("format string", ...)+`::
563 Library trace logging statement.
cd4aac1e
PP
564
565`+BT_LIB_LOGD("format string", ...)+`::
566 Library debug logging statement.
567
568`+BT_LIB_LOGI("format string", ...)+`::
569 Library info logging statement.
570
571`+BT_LIB_LOGW("format string", ...)+`::
572 Library warning logging statement.
573
574`+BT_LIB_LOGE("format string", ...)+`::
575 Library error logging statement.
576
577`+BT_LIB_LOGF("format string", ...)+`::
578 Library fatal logging statement.
579
3cd4c495
PP
580`+BT_LIB_LOGW_APPEND_CAUSE("format string", ...)+`::
581 Library warning logging statement, and unconditional error cause
582 appending.
583
584`+BT_LIB_LOGE_APPEND_CAUSE("format string", ...)+`::
585 Library error logging statement, and unconditional error cause
586 appending.
587
588`+BT_LIB_LOGF_APPEND_CAUSE("format string", ...)+`::
589 Library fatal logging statement, and unconditional error cause
590 appending.
591
cd4aac1e
PP
592The macros above accept the typical `printf()` conversion specifiers
593with the following limitations:
594
595* The `+*+` width specifier is not accepted.
596* The `+*+` precision specifier is not accepted.
597* The `j` and `t` length modifiers are not accepted.
598* The `n` format specifier is not accepted.
599* The format specifiers defined in `<inttypes.h>` are not accepted,
600 except for `PRId64`, `PRIu64`, `PRIx64`, `PRIX64`, `PRIo64`, and
601 `PRIi64`.
602
28b765ac 603The {bt2} library custom conversion specifier is accepted. Its
cd4aac1e
PP
604syntax is either `%!u` to format a UUID (`bt_uuid` type), or:
605
606. Introductory `%!` sequence.
607
608. **Optional**: `[` followed by a custom prefix for the printed fields
609 of this specifier, followed by `]`. The standard form is to end this
610 prefix with `-` so that, for example, with the prefix `tc-`, the
611 complete field name becomes `tc-addr`.
612
613. **Optional**: `pass:[+]` to print extended object members. This
614 depends on the provided format specifier.
615
616. Format specifier (see below).
617
618The available format specifiers are:
619
620[options="header,autowidth",cols="3"]
621|===
622|Specifier
623|Object
624|Expected C type
625
626|`F`
627|Trace IR field class
6769570a 628|`+const struct bt_field_class *+`
cd4aac1e
PP
629
630|`f`
631|Trace IR field
6769570a 632|`+const struct bt_field *+`
cd4aac1e
PP
633
634|`P`
635|Trace IR field path
6769570a 636|`+const struct bt_field_path *+`
cd4aac1e
PP
637
638|`E`
639|Trace IR event class
6769570a 640|`+const struct bt_event_class *+`
cd4aac1e
PP
641
642|`e`
643|Trace IR event
6769570a 644|`+const struct bt_event *+`
cd4aac1e
PP
645
646|`S`
647|Trace IR stream class.
6769570a 648|`+const struct bt_stream_class *+`
cd4aac1e
PP
649
650|`s`
651|Trace IR stream
6769570a 652|`+const struct bt_stream *+`
cd4aac1e
PP
653
654|`a`
655|Trace IR packet
6769570a 656|`+const struct bt_packet *+`
cd4aac1e
PP
657
658|`T`
659|Trace IR trace class
6769570a 660|`+const struct bt_trace_class *+`
cd4aac1e
PP
661
662|`t`
663|Trace IR trace
6769570a 664|`+const struct bt_trace *+`
cd4aac1e
PP
665
666|`K`
667|Trace IR clock class
6769570a 668|`+const struct bt_clock_class *+`
cd4aac1e
PP
669
670|`k`
671|Trace IR clock snapshot
6769570a 672|`+const struct bt_clock_snapshot *+`
cd4aac1e
PP
673
674|`v`
675|Value object
6769570a
PP
676|`+const struct bt_value *+`
677
678|`R`
679|Integer range set
680|`const struct bt_integer_range_set *`
cd4aac1e
PP
681
682|`n`
683|Message
6769570a 684|`+const struct bt_message *+`
cd4aac1e 685
a3f0c7db
SM
686|`I`
687|Message iterator class
688|`struct bt_message_iterator_class *`
689
cd4aac1e
PP
690|`i`
691|Message iterator
692|`struct bt_message_iterator *`
693
694|`C`
695|Component class
696|`struct bt_component_class *`
697
698|`c`
699|Component
6769570a 700|`+const struct bt_component *+`
cd4aac1e
PP
701
702|`p`
703|Port
6769570a 704|`+const struct bt_port *+`
cd4aac1e
PP
705
706|`x`
707|Connection
6769570a 708|`+const struct bt_connection *+`
cd4aac1e
PP
709
710|`g`
711|Graph
6769570a 712|`+const struct bt_graph *+`
cd4aac1e 713
b70d57a1
PP
714|`z`
715|Interrupter
716|`+struct bt_interrupter *+`
717
cd4aac1e
PP
718|`l`
719|Plugin
6769570a 720|`+const struct bt_plugin *+`
cd4aac1e 721
553c4bab
PP
722|`r`
723|Error cause
6769570a 724|`+const struct bt_error_cause *+`
553c4bab 725
cd4aac1e
PP
726|`o`
727|Object pool
6769570a 728|`+const struct bt_object_pool *+`
cd4aac1e
PP
729
730|`O`
731|Object
6769570a 732|`+const struct bt_object *+`
cd4aac1e
PP
733|===
734
735Conversion specifier examples:
736
737* `%!f`
738* `%![my-event-]+e`
739* `%!t`
740* `%!+F`
741
742The ``, `` string (comma and space) is printed between individual
743fields, but **not after the last one**. Therefore, you must put this
744separator in the format string between two conversion specifiers, for
745example:
746
747[source,c]
748----
749BT_LIB_LOGW("Message: count=%u, %!E, %!+K", count, event_class, clock_class);
750----
751
752Example with a custom prefix:
753
754[source,c]
755----
756BT_LIB_LOGI("Some message: %![ec-a-]e, %![ec-b-]+e", ec_a, ec_b);
757----
758
28b765ac 759It is safe to pass `NULL` as any {bt2} object parameter: the macros
cd4aac1e
PP
760only print its null address.
761
762WARNING: Build-time `printf()` format checks are disabled for the
763`+BT_LIB_LOG*()+` macros because there are custom conversion specifiers,
764so make sure to test your logging statements.
765
cd4aac1e
PP
766[[comp-logging-statements]]
767==== Component-specific logging statement macros
768
769There are available logging macros for components. They prepend a prefix
770including the component's name to the logging message.
771
28b765ac
PP
772See <<logging-instrument-c-file-compcls>> and
773<<logging-instrument-h-file-compcls>> to learn how to be able to use the
cd4aac1e
PP
774following macros.
775
776The component logging statement macros are named `+BT_COMP_LOG*()+`
777instead of `+BT_LOG*()+`:
778
ef267d12
PP
779`+BT_COMP_LOGT("format string", ...)+`::
780 Component trace logging statement.
cd4aac1e
PP
781
782`+BT_COMP_LOGD("format string", ...)+`::
783 Component debug logging statement.
784
785`+BT_COMP_LOGI("format string", ...)+`::
786 Component info logging statement.
787
788`+BT_COMP_LOGW("format string", ...)+`::
789 Component warning logging statement.
790
791`+BT_COMP_LOGE("format string", ...)+`::
792 Component error logging statement.
793
794`+BT_COMP_LOGF("format string", ...)+`::
795 Component fatal logging statement.
796
ef267d12
PP
797`+BT_COMP_LOGT_STR("preformatted string")+`::
798 Component preformatted string trace logging statement.
cd4aac1e
PP
799
800`+BT_COMP_LOGD_STR("preformatted string")+`::
801 Component preformatted string debug logging statement.
802
803`+BT_COMP_LOGI_STR("preformatted string")+`::
804 Component preformatted string info logging statement.
805
806`+BT_COMP_LOGW_STR("preformatted string")+`::
807 Component preformatted string warning logging statement.
808
809`+BT_COMP_LOGE_STR("preformatted string")+`::
810 Component preformatted string error logging statement.
811
812`+BT_COMP_LOGF_STR("preformatted string")+`::
813 Component preformatted string fatal logging statement.
814
ef267d12
PP
815`+BT_COMP_LOGT_ERRNO("initial message", "format string", ...)+`::
816 Component `errno` string trace logging statement.
cd4aac1e
PP
817
818`+BT_COMP_LOGD_ERRNO("initial message", "format string", ...)+`::
819 Component `errno` string debug logging statement.
820
821`+BT_COMP_LOGI_ERRNO("initial message", "format string", ...)+`::
822 Component `errno` string info logging statement.
823
824`+BT_COMP_LOGW_ERRNO("initial message", "format string", ...)+`::
825 Component `errno` string warning logging statement.
826
827`+BT_COMP_LOGE_ERRNO("initial message", "format string", ...)+`::
828 Component `errno` string error logging statement.
829
830`+BT_COMP_LOGF_ERRNO("initial message", "format string", ...)+`::
831 Component `errno` string fatal logging statement.
832
ef267d12
PP
833`+BT_COMP_LOGT_MEM(data_ptr, data_size, "format string", ...)+`::
834 Component memory trace logging statement.
cd4aac1e
PP
835
836`+BT_COMP_LOGD_MEM(data_ptr, data_size, "format string", ...)+`::
837 Component memory debug logging statement.
838
839`+BT_COMP_LOGI_MEM(data_ptr, data_size, "format string", ...)+`::
840 Component memory info logging statement.
841
842`+BT_COMP_LOGW_MEM(data_ptr, data_size, "format string", ...)+`::
843 Component memory warning logging statement.
844
845`+BT_COMP_LOGE_MEM(data_ptr, data_size, "format string", ...)+`::
846 Component memory error logging statement.
847
848`+BT_COMP_LOGF_MEM(data_ptr, data_size, "format string", ...)+`::
849 Component memory fatal logging statement.
beb0fb75 850
f9ded0e0 851==== Conditional logging
beb0fb75 852
cd4aac1e 853`+BT_LOG_IF(cond, statement)+`::
beb0fb75
PP
854 Execute `statement` only if `cond` is true.
855+
856Example:
857+
858--
859[source,c]
860----
861BT_LOG_IF(i < count / 2, BT_LOGD("Log this: i=%d", i));
862----
863--
864
865To check the <<build-time-log-level,build-time log level>>:
866
867[source,c]
868----
869#if BT_LOG_ENABLED_DEBUG
870...
871#endif
872----
873
cd4aac1e
PP
874This tests if the _DEBUG_ level was enabled at build time. This means
875that the current, <<run-time-log-level,run-time log level>> _could_ be
f9ded0e0
PP
876_DEBUG_, but it could also be higher. The rule of thumb is to use only
877logging statements at the same level in a `BT_LOG_ENABLED_*` conditional
878block.
beb0fb75
PP
879
880The available definitions for build-time conditions are:
881
ef267d12 882* `BT_LOG_ENABLED_TRACE`
beb0fb75
PP
883* `BT_LOG_ENABLED_DEBUG`
884* `BT_LOG_ENABLED_INFO`
770538dd 885* `BT_LOG_ENABLED_WARNING`
beb0fb75
PP
886* `BT_LOG_ENABLED_ERROR`
887* `BT_LOG_ENABLED_FATAL`
888
889To check the current, <<run-time-log-level,run-time log level>>:
890
891[source,c]
892----
893if (BT_LOG_ON_DEBUG) {
894 ...
895}
896----
897
898This tests if the _DEBUG_ log level is dynamically turned on
cd4aac1e 899(implies that it's also enabled at build time). This check could have a
beb0fb75
PP
900noticeable impact on performance.
901
902The available definitions for run-time conditions are:
903
ef267d12 904* `BT_LOG_ON_TRACE`
beb0fb75
PP
905* `BT_LOG_ON_DEBUG`
906* `BT_LOG_ON_INFO`
770538dd 907* `BT_LOG_ON_WARNING`
beb0fb75
PP
908* `BT_LOG_ON_ERROR`
909* `BT_LOG_ON_FATAL`
910
cd4aac1e
PP
911Those macros check the module-specific log level symbol (defined by
912`BT_LOG_OUTPUT_LEVEL`).
beb0fb75
PP
913
914Never, ever write code which would be executed only to compute the
915fields of a logging statement outside a conditional logging scope,
916for example:
917
918[source,c]
919----
920int number = get_number_of_event_classes_with_property_x(...);
921BT_LOGD("Bla bla: number=%d", number);
922----
923
924Do this instead:
925
926[source,c]
927----
928if (BT_LOG_ON_DEBUG) {
929 int number = get_number_of_event_classes_with_property_x(...);
930 BT_LOGD("Bla bla: number=%d", number);
931}
932----
933
934Or even this:
935
936[source,c]
937----
938BT_LOGD("Bla bla: number=%d", get_number_of_event_classes_with_property_x(...));
939----
940
cd4aac1e 941=== Guides
beb0fb75 942
cd4aac1e 943[[logging-instrument-c-file-gen]]
fb7ff115 944==== Instrument a {c-cpp} source file (generic)
beb0fb75 945
fb7ff115 946To instrument a {c-cpp} source file (`.c`/`.cpp`):
cd4aac1e
PP
947
948. At the top of the file, before the first `#include` line (if any),
949 define your file's <<choose-a-logging-tag,logging tag>> name:
4a41523d
PP
950+
951--
952[source,c]
953----
cd4aac1e 954#define BT_LOG_TAG "SUBSYS/MY-MODULE/MY-FILE"
4a41523d
PP
955----
956--
cd4aac1e
PP
957
958. Below the line above, define the source file's log level expression,
959 `BT_LOG_OUTPUT_LEVEL`. This expression is evaluated for each
960 <<gen-logging-statements,logging statement>> to know the current
961 <<run-time-log-level,run-time log level>>.
962+
963Examples:
964+
965[source,c]
966----
967/* Global log level variable */
968#define BT_LOG_OUTPUT_LEVEL module_global_log_level
969----
4a41523d 970+
cd4aac1e
PP
971[source,c]
972----
973/* Local log level variable; must exist where you use BT_LOG*() */
974#define BT_LOG_OUTPUT_LEVEL log_level
975----
976+
977[source,c]
978----
979/* Object's log level; `obj` must exist where you use BT_LOG*() */
980#define BT_LOG_OUTPUT_LEVEL (obj->log_level)
981----
4a41523d 982
cd4aac1e 983. Include `"logging/log.h"`:
4a41523d 984+
beb0fb75
PP
985[source,c]
986----
578e048b 987#include "logging/log.h"
cd4aac1e 988----
4a41523d 989
cd4aac1e
PP
990. In the file, instrument your code with the
991 <<gen-logging-statements,generic logging statement macros>>.
4a41523d 992
cd4aac1e 993[[logging-instrument-h-file-gen]]
fb7ff115 994==== Instrument a {c-cpp} header file (generic)
cd4aac1e 995
fb7ff115
PP
996To instrument a {c-cpp} header file (`.h`/`.hpp`), if you have
997`static inline` functions in it:
cd4aac1e
PP
998
999. Do not include `"logging/log.h"`!
1000
1001. Do one of:
1002
1003.. In the file, instrument your code with the
1004 <<gen-logging-statements,generic logging statement macros>>, making
1005 each of them conditional to the existence of the macro you're using:
4a41523d 1006+
cd4aac1e
PP
1007[source,c]
1008----
1009static inline
1010int some_function(int x)
1011{
1012 /* ... */
4a41523d 1013
ef267d12
PP
1014#ifdef BT_LOGT
1015 BT_LOGT(...);
cd4aac1e 1016#endif
beb0fb75 1017
cd4aac1e 1018 /* ... */
6470c171 1019
cd4aac1e
PP
1020#ifdef BT_LOGW_STR
1021 BT_LOGW_STR(...);
1022#endif
f9ded0e0 1023
cd4aac1e
PP
1024 /* ... */
1025}
1026----
1027+
fb7ff115
PP
1028The {c-cpp} source files which include this header file determine if
1029logging is enabled or not for them, and if so, what is their
cd4aac1e
PP
1030<<choose-a-logging-tag,logging tag>> and <<run-time-log-level,run-time
1031log level>> expression.
f9ded0e0 1032
cd4aac1e
PP
1033.. Require that logging be enabled:
1034+
1035[source,c]
1036----
1037/* Protection: this file uses BT_LOG*() macros directly */
1038#ifndef BT_LOG_SUPPORTED
1039# error Please include "logging/log.h" before including this file.
1040#endif
1041----
1042+
1043Then, in the file, instrument your code with the
1044<<gen-logging-statements,generic logging statement macros>>.
1045
cd4aac1e 1046[[logging-instrument-c-file-lib]]
fb7ff115 1047==== Instrument a library {c-cpp} source file
cd4aac1e 1048
fb7ff115 1049To instrument a library {c-cpp} source file (`.c`/`.cpp`):
4a41523d
PP
1050
1051. At the top of the file, before the first `#include` line (if any),
cd4aac1e
PP
1052 define your file's <<choose-a-logging-tag,logging tag>> name (this
1053 tag must start with `LIB/`):
4a41523d
PP
1054+
1055--
1056[source,c]
1057----
cd4aac1e 1058#define BT_LOG_TAG "LIB/THE-FILE"
4a41523d
PP
1059----
1060--
cd4aac1e
PP
1061
1062. Include `"lib/logging.h"`:
4a41523d 1063+
cd4aac1e
PP
1064[source,c]
1065----
1066#include "lib/logging.h"
1067----
4a41523d 1068
cd4aac1e
PP
1069. In the file, instrument your code with the
1070 <<lib-logging-statements,library logging statement macros>> or with
1071 the <<gen-logging-statements,generic logging statement macros>>.
4a41523d 1072
cd4aac1e 1073[[logging-instrument-h-file-lib]]
fb7ff115 1074==== Instrument a library {c-cpp} header file
cd4aac1e 1075
fb7ff115
PP
1076To instrument a library {c-cpp} header file (`.h`/`.hpp`), if you have
1077`static inline` functions in it:
cd4aac1e
PP
1078
1079. Do not include `"lib/logging.h"`!
1080
1081. Require that library logging be enabled:
1082+
1083[source,c]
1084----
1085/* Protection: this file uses BT_LIB_LOG*() macros directly */
1086#ifndef BT_LIB_LOG_SUPPORTED
1087# error Please include "lib/logging.h" before including this file.
1088#endif
1089----
1090
1091. In the file, instrument your code with the
1092 <<lib-logging-statements,library logging statement macros>> or with
1093 the <<gen-logging-statements,generic logging statement macros>>.
f9ded0e0 1094
cd4aac1e 1095[[logging-instrument-c-file-compcls]]
fb7ff115 1096==== Instrument a component class {c-cpp} source file
cd4aac1e 1097
fb7ff115 1098To instrument a component class {c-cpp} source file (`.c`/`.cpp`):
cd4aac1e
PP
1099
1100. At the top of the file, before the first `#include` line (if any),
1101 define your file's <<choose-a-logging-tag,logging tag>> name (this tag
1102 must start with `PLUGIN/` followed by the component class identifier):
4a41523d
PP
1103+
1104--
1105[source,c]
1106----
cd4aac1e
PP
1107#define BT_LOG_TAG "PLUGIN/SRC.MY-PLUGIN.MY-SRC"
1108----
1109--
4a41523d 1110
cd4aac1e
PP
1111. Below the line above, define the source file's log level expression,
1112 `BT_LOG_OUTPUT_LEVEL`. This expression is evaluated for each
1113 <<comp-logging-statements,logging statement>> to know the current
1114 <<run-time-log-level,run-time log level>>.
1115+
1116For a component class file, it is usually a member of a local component
1117private structure variable:
1118+
1119[source,c]
1120----
1121#define BT_LOG_OUTPUT_LEVEL (my_comp->log_level)
1122----
4a41523d 1123
cd4aac1e
PP
1124. Below the line above, define `BT_COMP_LOG_SELF_COMP` to an expression
1125 which, evaluated in the context of the
1126 <<comp-logging-statements,logging statements>>, evaluates to the self
1127 component address (`+bt_self_component *+`) of the component.
1128+
1129This is usually a member of a local component private structure
1130variable:
1131+
1132[source,c]
1133----
1134#define BT_COMP_LOG_SELF_COMP (my_comp->self_comp)
1135----
1136
d9c39b0a 1137. Include `"logging/comp-logging.h"`:
cd4aac1e
PP
1138+
1139[source,c]
1140----
d9c39b0a 1141#include "logging/comp-logging.h"
cd4aac1e
PP
1142----
1143
1144. In the component initialization method, make sure to set the
1145 component private structure's log level member to the initial
1146 component's log level:
1147+
1148[source,c]
1149----
1150struct my_comp {
1151 bt_logging_level log_level;
4a41523d 1152 /* ... */
cd4aac1e 1153};
4a41523d 1154
cd4aac1e
PP
1155BT_HIDDEN
1156bt_self_component_status my_comp_init(
1157 bt_self_component_source *self_comp_src,
1158 bt_value *params, void *init_method_data)
1159{
1160 struct my_comp *my_comp = g_new0(struct my_comp, 1);
1161 bt_self_component *self_comp =
1162 bt_self_component_source_as_self_component(self_comp_src);
1163 const bt_component *comp = bt_self_component_as_component(self_comp);
1164
1165 BT_ASSERT(my_comp);
1166 my_comp->log_level = bt_component_get_logging_level(comp);
4a41523d
PP
1167
1168 /* ... */
1169}
1170----
cd4aac1e
PP
1171
1172. In the file, instrument your code with the
1173 <<comp-logging-statements,component logging statement macros>>.
1174
cd4aac1e 1175[[logging-instrument-h-file-compcls]]
fb7ff115 1176==== Instrument a component class {c-cpp} header file
cd4aac1e 1177
fb7ff115
PP
1178To instrument a component class {c-cpp} header file (`.h`/`.hpp`), if
1179you have `static inline` functions in it:
cd4aac1e 1180
d9c39b0a 1181. Do not include `"logging/comp-logging.h"`!
cd4aac1e
PP
1182
1183. Require that component logging be enabled:
4a41523d 1184+
cd4aac1e
PP
1185[source,c]
1186----
1187/* Protection: this file uses BT_COMP_LOG*() macros directly */
1188#ifndef BT_COMP_LOG_SUPPORTED
d9c39b0a 1189# error Please include "logging/comp-logging.h" before including this file.
cd4aac1e
PP
1190#endif
1191----
1192
1193. In the file, instrument your code with the
1194 <<comp-logging-statements,component logging statement macros>>.
1195
cd4aac1e
PP
1196[[choose-a-logging-tag]]
1197==== Choose a logging tag
beb0fb75 1198
fb7ff115
PP
1199Each logging-enabled {c-cpp} source file must define `BT_LOG_TAG` to a
1200logging tag. A logging tag is a namespace to identify the logging
1201messages of this specific source file.
beb0fb75 1202
cd4aac1e
PP
1203In general, a logging tag name _must_ be only uppercase letters, digits,
1204and the `-`, `.`, and `/` characters.
beb0fb75 1205
cd4aac1e
PP
1206Use `/` to show the subsystem to source file hierarchy.
1207
28b765ac 1208For the {bt2} library, start with `LIB/`.
cd4aac1e
PP
1209
1210For the CTF writer library, start with `CTF-WRITER/`.
1211
1212For component classes, use:
beb0fb75 1213
93841983 1214[verse]
cd4aac1e
PP
1215`PLUGIN/__CCTYPE__.__PNAME__.__CCNAME__[/__FILE__]`
1216
1217With:
1218
1219`__CCTYPE__`::
1220 Component class's type (`SRC`, `FLT`, or `SINK`).
beb0fb75 1221
cd4aac1e 1222`__PNAME__`::
93841983 1223 Plugin's name.
beb0fb75 1224
cd4aac1e 1225`__CCNAME__`::
93841983 1226 Component class's name.
beb0fb75 1227
cd4aac1e 1228`__FILE__`::
93841983 1229 Additional information to specify the source file name or module.
beb0fb75 1230
cd4aac1e 1231For plugins (files common to many component classes), use:
beb0fb75 1232
cd4aac1e
PP
1233[verse]
1234`PLUGIN/__PNAME__[/__FILE__]`
beb0fb75 1235
cd4aac1e 1236With:
beb0fb75 1237
cd4aac1e
PP
1238`__PNAME__`::
1239 Plugin's name.
1240
1241`__FILE__`::
1242 Additional information to specify the source file name or module.
1243
cd4aac1e
PP
1244[[choose-a-log-level]]
1245==== Choose a log level
beb0fb75
PP
1246
1247Choosing the appropriate level for your logging statement is very
1248important.
1249
cd4aac1e 1250[options="header,autowidth",cols="1,2,3a,4"]
beb0fb75 1251|===
cd4aac1e 1252|Log level |Description |Use cases |Expected impact on performance
beb0fb75
PP
1253
1254|_FATAL_
cd4aac1e
PP
1255|
1256The program, library, or plugin cannot continue to work in this
beb0fb75
PP
1257condition: it must be terminated immediately.
1258
b4b9064d 1259A _FATAL_-level logging statement should always be followed by
4a41523d 1260`abort()`.
beb0fb75
PP
1261|
1262* Unexpected return values from system calls.
4a41523d
PP
1263* Logic error in internal code, for example an unexpected value in a
1264 `switch` statement.
cd4aac1e 1265* Failed assertion (within `BT_ASSERT()`).
bdb288b3
PP
1266* Unsatisfied library precondition (within `BT_ASSERT_PRE()` or
1267 `BT_ASSERT_PRE_DEV()`).
1268* Unsatisfied library postcondition (within `BT_ASSERT_POST()` or
1269 `BT_ASSERT_POST_DEV()`).
dd22a91f 1270|Almost none: always enabled.
beb0fb75
PP
1271
1272|_ERROR_
cd4aac1e
PP
1273|
1274An important error which is somewhat not fatal, that is, the program,
beb0fb75
PP
1275library, or plugin can continue to work after this, but you judge that
1276it should be reported to the user.
1277
1278Usually, the program cannot recover from such an error, but it can at
1279least exit cleanly.
1280|
1281* Memory allocation errors.
cd4aac1e
PP
1282* Wrong component initialization parameters.
1283* Corrupted, unrecoverable trace data.
beb0fb75
PP
1284* Failed to perform an operation which should work considering the
1285 implementation and the satisfied preconditions. For example, the
1286 failure to create an empty object (no parameters): most probably
1287 failed internally because of an allocation error.
93841983 1288* Almost any error in terminal elements: CLI and plugins.
dd22a91f 1289|Almost none: always enabled.
beb0fb75 1290
770538dd 1291|_WARNING_
cd4aac1e
PP
1292|
1293An error which still allows the execution to continue, but you judge
1294that it should be reported to the user.
6c1b33a8 1295
770538dd
PP
1296_WARNING_-level logging statements are for any error or weird action
1297that is directly or indirectly caused by the user, often through some
1298bad input data. For example, not having enough memory is considered
1299beyond the user's control, so we always log memory errors with an
1300_ERROR_ level (not _FATAL_ because we usually don't abort in this
1301condition).
beb0fb75 1302|
cd4aac1e
PP
1303* Missing data within something that is expected to have it, but there's
1304 an alternative.
1305* Invalid file, but recoverable/fixable.
dd22a91f 1306|Almost none: always enabled.
beb0fb75
PP
1307
1308|_INFO_
beb0fb75 1309|
cd4aac1e
PP
1310Any useful information which a non-developer user would possibly
1311understand.
1312
1313Anything logged with this level must _not_ happen repetitively on the
1314fast path, that is, nothing related to each message, for example. This
1315level is used for sporadic and one-shot events.
1316|
1317* CLI or component configuration report.
1318* Successful plugin, component, or message iterator initialization.
1319* In the library: anything related to plugins, graphs, component
1320 classes, components, message iterators, connections, and ports which
1321 is not on the fast path.
beb0fb75 1322* Successful connection to or disconnection from another system.
4a41523d 1323* An _optional_ subsystem cannot be loaded.
cd4aac1e
PP
1324* An _optional_ field/datum cannot be found.
1325|
dd22a91f 1326Very little: always enabled.
beb0fb75
PP
1327
1328|_DEBUG_
beb0fb75 1329|
28b765ac 1330Something that only {bt2} developers would be interested into,
cd4aac1e
PP
1331which can occur on the fast path, but not more often than once per
1332message.
1333
1334The _DEBUG_ level is the default <<build-time-log-level,build-time log
1335level>> as, since it's not _too_ verbose, the performance is similar to
1336an _INFO_ build.
1337|
1338* Object construction and destruction.
1339* Object recycling (except fields).
1340* Object copying (except fields and values).
1341* Object freezing (whatever the type, as freezing only occurs in
1342 developer mode).
9b4f9b42 1343* Object interruption.
cd4aac1e
PP
1344* Calling user methods and logging the result.
1345* Setting object properties (except fields and values).
1346|
ef267d12 1347Noticeable, but not as much as the _TRACE_ level: could be executed
cd4aac1e
PP
1348in production if you're going to need a thorough log for support
1349tickets without having to rebuild the project.
beb0fb75 1350
ef267d12 1351|_TRACE_
cd4aac1e
PP
1352|
1353Low-level debugging context information (anything that does not fit the
1354other log levels). More appropriate for tracing in general.
beb0fb75
PP
1355|
1356* Reference count change.
cd4aac1e 1357* Fast path, low level state machine's state change.
beb0fb75
PP
1358* Get or set an object's property.
1359* Object comparison's intermediate results.
1360|Huge: not executed in production.
1361|===
1362
cd4aac1e
PP
1363[IMPORTANT]
1364--
770538dd 1365Make sure not to use a _WARNING_ (or higher) log level when the
cd4aac1e
PP
1366condition leading to the logging statement can occur under normal
1367circumstances.
1368
1369For example, a public function to get some object or
1370property from an object by name or key that fails to find the value is
1371not a warning scenario: the user could legitimately use this function to
1372check if the name/key exists in the object. In this case, use the
ef267d12 1373_TRACE_ level (or do not log at all).
cd4aac1e 1374--
b4b9064d 1375
b4b9064d 1376[[message]]
cd4aac1e 1377==== Write an appropriate message
beb0fb75
PP
1378
1379Follow those rules when you write a logging statement's message:
1380
cd4aac1e
PP
1381* Use an English sentence which starts with a capital letter.
1382
1383* Start the sentence with the appropriate verb tense depending on the
1384 context. For example:
beb0fb75
PP
1385+
1386--
b4b9064d
PP
1387** Beginning of operation (present continuous): _Creating ..._,
1388 _Copying ..._, _Serializing ..._, _Freezing ..._, _Destroying ..._
1389** End of operation (simple past): _Created ..._, _Successfully created ..._,
1390 _Failed to create ..._, _Set ..._ (simple past of _to set_ which is
1391 also _set_)
beb0fb75
PP
1392--
1393+
1394For warning and error messages, you can start the message with _Cannot_
b4b9064d 1395or _Failed to_ followed by a verb if it's appropriate.
beb0fb75
PP
1396
1397* Do not include the log level in the message itself. For example,
1398 do not start the message with _Error while_ or _Warning:_.
1399
cd4aac1e
PP
1400* Do not put newlines, tabs, or other special characters in the message,
1401 unless you want to log a string with such characters. Note that
1402 multiline logging messages can be hard to parse, analyze, and filter,
1403 however, so prefer multiple logging statements over a single statement
1404 with newlines.
beb0fb75
PP
1405
1406* **If there are fields that your logging statement must record**,
1407 follow the message with `:` followed by a space, then with the list of
1408 fields (more about this below). If there are no fields, end the
1409 sentence with a period.
1410
1411The statement's fields _must_ be a comma-separated list of
cd4aac1e
PP
1412`__name__=__value__` tokens. Keep `__name__` as simple as possible; use
1413kebab case if possible. If `__value__` is a non-alphanumeric string, put
1414it between double quotes (`"%s"` specifier). Always use the `PRId64` and
1415`PRIu64` specifiers to log an `int64_t` or an `uint64_t` value. Use `%d`
1416to log a boolean value.
beb0fb75
PP
1417
1418Example:
1419
cd4aac1e
PP
1420 "Cannot read stream data for indexing: path=\"%s\", name=\"%s\", "
1421 "stream-id=%" PRIu64 ", stream-fd=%d, "
1422 "index=%" PRIu64 ", status=%s, is-mapped=%d"
beb0fb75 1423
cd4aac1e
PP
1424By following a standard format for the statement fields, it is easier to
1425use tools like https://www.elastic.co/products/logstash[Logstash] or
1426even https://www.splunk.com/[Splunk] to split fields and analyze logs.
beb0fb75
PP
1427
1428Prefer the following suffixes in field names:
1429
1430[options="header,autowidth"]
1431|===
1432|Field name suffix |Description |Format specifier
1433
1434|`-addr` |Memory address |`%p`
1435|`-fd` |File descriptor |`%d`
cd4aac1e
PP
1436|`-fp` |File stream (`+FILE *+`) |`%p`
1437|`-id` |Object's ID |`%" PRIu64 "`
1438|`-index` |Index |`%" PRIu64 "`
beb0fb75 1439|`-name` |Object's name |`\"%s\"`
beb0fb75
PP
1440|===
1441
f9ded0e0 1442=== Output
beb0fb75
PP
1443
1444The log is printed to the standard error stream. A log line contains the
cd4aac1e
PP
1445time, the process and thread IDs, the <<log-levels,log level>>, the
1446<<choose-a-logging-tag,logging tag>>, the source's function name, file
1447name and line number, and the <<message,message>>.
beb0fb75 1448
28b765ac 1449When {bt2} supports terminal color codes (depends on the
f9ded0e0
PP
1450`BABELTRACE_TERM_COLOR` environment variable's value and what the
1451standard output and error streams are plugged into), _INFO_-level lines
770538dd 1452are blue, _WARNING_-level lines are yellow, and _ERROR_-level and
f9ded0e0
PP
1453_FATAL_-level lines are red.
1454
1455Log line example:
beb0fb75 1456
cd4aac1e
PP
1457----
145805-11 00:58:03.691 23402 23402 D VALUES bt_value_destroy@values.c:498 Destroying value: addr=0xb9c3eb0
1459----
beb0fb75 1460
cd4aac1e
PP
1461You can easily filter the log with `grep` or `ag`. For example, to keep
1462only the _DEBUG_-level logging messages that the `FIELD-CLASS` module
beb0fb75
PP
1463generates:
1464
cd4aac1e
PP
1465----
1466$ babeltrace2 --log-level=D /path/to/trace |& ag 'D FIELD-CLASS'
1467----
1468
cd4aac1e
PP
1469== Valgrind
1470
1471To use Valgrind on an application (for example, the CLI or a test) which
1472loads libbabeltrace2, use:
1473
1474----
1475$ G_SLICE=always-malloc G_DEBUG=gc-friendly PYTHONMALLOC=malloc \
916d0d91 1476 LIBBABELTRACE2_NO_DLCLOSE=1 valgrind --leak-check=full app
cd4aac1e
PP
1477----
1478
1479`G_SLICE=always-malloc` and `G_DEBUG=gc-friendly` is for GLib and
1480`PYTHONMALLOC=malloc` is for the Python interpreter, if it is used by
1481the Python plugin provider (Valgrind will probably show a lot of errors
1482which originate from the Python interpreter anyway).
1483
16b7b5e7 1484`LIBBABELTRACE2_NO_DLCLOSE=1` makes libbabeltrace2 not close the shared
cd4aac1e
PP
1485libraries (plugins) which it loads. You need this to see the appropriate
1486backtrace when Valgrind shows errors.
c637d729
FD
1487
1488== Testing
1489
800400d8
PP
1490[[test-env]]
1491=== Environment
c637d729 1492
8aed890c
SM
1493Running `make check` in the build directory (regardless of whether the build is
1494in-tree or out-of-tree) automatically sets up the appropriate environment for
1495tests to run in, so nothing more is needed.
1496
1497If building in-tree, you can run single tests from the tree directly:
1498
1499----
1500$ ./tests/plugins/sink.text.pretty/test_enum
1501----
1502
1503If building out-of-tree, you can get the appropriate environment by sourcing
1504the `tests/utils/env.sh` file residing in the build directory against which you
1505want to run tests.
1506
1507----
1508$ source /path/to/my/build/tests/utils/env.sh
1509$ ./tests/plugins/sink.text.pretty/test_enum
800400d8
PP
1510----
1511
1512==== Python
1513
1514You can use the `tests/utils/run_python_bt2` script to run any command
1515within an environment making the build's `bt2` Python package available.
1516
1517`run_python_bt2` uses <<test-env,`utils.sh`>> which needs to know the
1518build directory, so make sure you set the `BT_TESTS_BUILDDIR`
1519environment variable correctly _if you build out of tree_, for example:
1520
1521----
1522$ export BT_TESTS_BUILDDIR=/path/to/build/babeltrace/tests
1523----
1524
1525You can run any command which needs the `bt2` Python package through
1526`run_python_bt2`, for example:
c637d729
FD
1527
1528----
800400d8 1529$ ./tests/utils/run_python_bt2 ipython3
c637d729
FD
1530----
1531
800400d8
PP
1532=== Report format
1533
1534All test scripts output the test results following the
1535https://testanything.org/[Test Anything Protocol] (TAP) format.
1536
1537The TAP format has two mechanisms to print additional information about
1538a test:
1539
1540* Print a line starting with `#` to the standard output.
1541+
1542This is usually done with the `diag()` C function or the `diag` shell
1543function.
1544
1545* Print to the standard error.
1546
800400d8
PP
1547=== Python bindings
1548
1549The `bt2` Python package tests are located in
1550`tests/bindings/python/bt2`.
1551
800400d8
PP
1552==== Python test runner
1553
28b765ac 1554`tests/utils/python/testrunner.py` is {bt2}'s Python test runner
800400d8
PP
1555which loads Python files containing unit tests, finds all the test
1556cases, and runs the tests, producing a TAP report.
1557
1558You can see the test runner command's help with:
c637d729
FD
1559
1560----
800400d8 1561$ python3 ./tests/utils/python/testrunner.py --help
c637d729
FD
1562----
1563
800400d8
PP
1564By default, the test runner reports failing tests (TAP's `not{nbsp}ok`
1565line), but continues to run other tests. You can use the `--failfast`
1566option to make the test runner fail as soon as a test fails.
c637d729 1567
800400d8
PP
1568==== Guides
1569
1570To run all the `bt2` Python package tests:
1571
1572* Run:
1573+
1574----
1575$ ./tests/utils/run_python_bt2 ./tests/bindings/python/bt2/test_python_bt2
1576----
1577+
1578or:
1579+
1580----
1581$ ./tests/utils/run_python_bt2 python3 ./tests/utils/python/testrunner.py \
1582 ./tests/bindings/python/bt2/ -p '*.py'
1583----
1584
1585To run **all the tests** in a test module (for example,
1586`test_value.py`):
1587
1588* Run:
1589+
1590----
1591$ ./tests/utils/run_python_bt2 python3 ./tests/utils/python/testrunner.py \
1592 ./tests/bindings/python/bt2 -t test_value
1593----
1594
1595To run a **specific test case** (for example, `RealValueTestCase` within
1596`test_value.py`):
1597
1598* Run:
1599+
1600----
1601$ ./tests/utils/run_python_bt2 python3 ./tests/utils/python/testrunner.py \
1602 ./tests/bindings/python/bt2/ -t test_value.RealValueTestCase
1603----
1604
1605To run a **specific test** (for example,
1606`RealValueTestCase.test_assign_pos_int` within `test_value.py`):
1607
1608* Run:
1609+
c637d729 1610----
800400d8
PP
1611$ ./tests/utils/run_python_bt2 python3 ./tests/utils/python/testrunner.py \
1612 ./tests/bindings/python/bt2/ -t test_value.RealValueTestCase.test_assign_pos_int
c637d729 1613----
fb7ff115
PP
1614
1615== {cpp} usage
1616
1617Some parts of {bt2} are written in {cpp}.
1618
1619This section shows what's important to know about {cpp} to contribute
1620to {bt2}.
1621
1622[IMPORTANT]
1623====
1624{bt2} only has {cpp} sources for _internal_ code.
1625
1626In other words, libbabeltrace2 _must_ expose a pure C99 API to preserve
1627ABI compatibility over time.
1628====
1629
1630=== Standard and dependencies
1631
1632The {bt2} project is configured to use the {cpp11} standard.
1633
1634{cpp11} makes it possible to build {bt2} with a broad range of
1635compilers, from GCC{nbsp}4.8 and Clang{nbsp}3.3.
1636
1637=== Automake/Libtool requirements
1638
1639To add a {cpp} source file to a part of the project, use the `.cpp`
1640extension and add it to the list of source files in `Makefile.am` as
1641usual.
1642
1643If a program or a shared library has a direct {cpp} source file, then
1644Libtool uses the {cpp} linker to create the result, dynamically
1645linking important runtime libraries such as libstdc++ and libgcc_s.
1646
1647Because a Libtool _convenience library_ is just an archive (`.a`), it's
1648_not_ dynamically linked to runtime libraries, even if one of its direct
1649sources is a {cpp} file. This means that for each program or shared
1650library named `my_target` in `Makefile.am` which is linked to a
1651convenience library having {cpp} sources (recursively), you _must_ do
1652one of:
1653
1654* Have at least one direct {cpp} source file in the
1655 `+*_my_target_SOURCES+` list.
1656
1657* Add:
1658+
1659----
1660nodist_EXTRA_my_target_SOURCES = dummy.cpp
1661----
1662+
1663See
1664https://www.gnu.org/software/automake/manual/automake.html#Libtool-Convenience-Libraries[Libtool
1665Convenience Libraries] to learn more.
1666
1667For a given program or library, you _cannot_ have a C{nbsp}file and a
1668{cpp}{nbsp}file having the same name, for example `list.c` and
1669`list.cpp`.
1670
1671=== Coding style
1672
1673==== Whitespaces, indentation, and line breaks
1674
1675All the project's {cpp} files follow the
1676https://clang.llvm.org/docs/ClangFormat.html[clang-format]
1677https://clang.llvm.org/docs/ClangFormatStyleOptions.html[style] of the
1678`.clang-format` file for whitespaces, indentation, and line breaks.
1679
1680You _must_ format modified and new {cpp} files with clang-format before
1681you create a contribution patch.
1682
ff7ea753 1683You need clang-format{nbsp}13 to use the project's `.clang-format` file.
fb7ff115
PP
1684
1685To automatically format all the project's {cpp} files, run:
1686
1687----
b61d5465
PP
1688$ ./tools/format-cpp
1689----
1690
1691Use the `FORMATTER` environment variable to override the default
1692formatter (`clang-format{nbsp}-i`):
1693
1694----
1695$ FORMATTER='clang-format-10 -i' ./tools/format-cpp
fb7ff115
PP
1696----
1697
1698==== Naming
1699
1700* Use camel case with a lowercase first letter for:
1701** Variable names: `size`, `objSize`.
1702** Function/method names: `size()`, `formatAndPrint()`.
1703
1704* Use camel case with an uppercase first letter for:
1705** Types: `Pistachio`, `NutManager`.
1706** Template parameters: `PlanetT`, `TotalSize`.
1707
1708* Use snake case with uppercase letters for:
1709** Definition/macro names: `MARK_AS_UNUSED()`, `SOME_FEATURE_EXISTS`.
1710** Enumerators: `Type::SIGNED_INT`, `Scope::FUNCTION`.
1711
1712* Use only lowercase letters and digits for namespaces: `mylib`, `bt2`.
1713
1714* Use the suffix `T` for type template parameters:
1715+
1716[source,cpp]
1717----
1718template <typename NameT, typename ItemT>
1719----
1720
1721* Name a template parameter pack `Args`.
1722+
1723[source,cpp]
1724----
1725template <typename NameT, typename... Args>
1726----
1727
1728* Use an underscore prefix for private and protected methods and member
1729 type names: `_tryConnect()`, `_NodeType`.
1730
1731* Use the prefix `_m` for private and protected member variable names:
1732 `_mLogger`, `_mSize`, `_mFieldClass`.
1733
1734* Name setters and getters like the property name, without `set` and
1735 `get` prefixes.
1736
1737* Use the `is` or `has` prefix, if possible, to name the functions which
1738 return `bool`.
1739
1740=== Coding convention
1741
1742In general, the project's contributors make an effort to follow,
1743for {cpp11} code:
1744
1745* The
1746 https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md[{cpp} Core Guidelines].
1747
1748* Scott Meyers's
1749 "`https://www.oreilly.com/library/view/effective-modern-c/9781491908419/[Effective Modern {cpp}]`".
1750
1751Here are a few important reminders:
1752
1753* Namespace your code.
1754
1755* Create one header/source file pair per class when possible.
1756+
1757For a class named `MyClass`, name the corresponding files `my-class.hpp`
1758and `my-class.cpp`.
1759
1760* When defining a class, put constructors as the first methods, whatever
1761 their access (public/protected/private), then the destructor, and then
1762 the rest.
1763
1764* Declare variables as close to where they are used as possible.
1765
1766* Use `auto` when possible.
1767
1768* Use `const` as much as possible, even for pointer
1769 (`+const char* const+`) and numeric values (`const unsigned int`)
1770 which never need to change.
1771
1772* Implement simple setters, getters, and one-liners in header files and
1773 everything else that's not a template in source files.
1774
1775* Make methods `const noexcept` or `const` as much as possible.
1776
1777* Make constructors `explicit` unless you really need an implicit
1778 constructor (which is rare).
1779
1780* Use `std::unique_ptr` to manage memory when possible.
1781+
1782However, use references (`+*my_unique_ptr+`) and raw pointers
1783(`+my_unique_ptr.get()+`) when not transferring ownership.
1784
1785* Use `nullptr`, not `NULL` nor 0.
1786
1787* Return by value (rvalue) instead of by output parameter (non-const
1788 lvalue reference), even complex objects, unless you can prove that the
1789 performance is improved when returning by parameter.
1790
1791* For a function parameter or a return value of which the type needs to
1792 be a reference or pointer, use:
1793+
1794If the value is mandatory:::
1795 A reference.
1796If the value is optional:::
1797 A raw pointer.
1798
1799* Don't use `+std::move()+` when you already have an rvalue, which
1800 means:
1801** Don't write `+return std::move(...);+` as this can interfere with
1802 RVO.
1803** Don't use `+std::move()+` with a function call
1804 (`+std::move(func())+`).
1805
1806* For each possible move/copy constructor or assignment operator, do one
1807 of:
1808** Write a custom one.
1809** Mark it as defaulted (`default`)
1810** Mark it as deleted (`delete`).
1811
1812* Use scoped enumerations (`+enum class+`).
1813
1814* Mark classes known to be final with the `final` keyword.
1815
1816* Use type aliases (`using`), not type definitions (`typedef`).
1817
1818* Use anonymous namespaces for local functions instead of `static`.
1819
1820* Don't pollute the global namespace:
1821** Don't use `using namespace xyz` anywhere.
1822** Use only namespace aliases in source files (`.cpp`), trying to
1823 use them in the smallest possible scope (function, or even smaller).
1824
1825* Return a structure with named members instead of a generic container
1826 such as `std::pair` or `std::tuple`.
1827
1828* When a class inherits a base class with virtual methods, use the
1829 `override` keyword to mark overridden virtual methods, and do not use
1830 the `virtual` keyword again.
1831
1832* Define overloaded operators only if their meaning is obvious,
1833 unsurprising, and consistent with the corresponding built-in
1834 operators.
1835+
1836For example, use `+|+` as a bitwise- or logical-or, not as a shell-style
1837pipe.
1838
1839* Use RAII wrappers when managing system resources or interacting with
1840 C{nbsp}libraries.
1841+
1842In other words, don't rely on ``goto``s and error labels to clean up as
1843you would do in{nbsp}C.
1844+
1845Use the RAII, Luke.
1846
1847* Throw an exception when there's an unexpected, exceptional condition,
1848 https://isocpp.org/wiki/faq/exceptions#ctors-can-throw[including from
1849 a constructor], instead of returning a status code.
1850
1851* Accept a by-value parameter and move it (when it's moveable) when you
1852 intend to copy it anyway.
1853+
1854You can do this with most STL containers.
1855+
1856Example:
1857+
1858[source,cpp]
1859----
1860void Obj::doSomething(std::string str)
1861{
1862 _mName = std::move(str);
1863 // ...
1864}
1865----
1866
1867.`baby.hpp`
1868====
1869This example shows a {cpp} header which follows the {bt2} {cpp} coding
1870convention.
1871
1872[source,cpp]
1873----
1874/*
1875 * SPDX-License-Identifier: MIT
1876 *
1877 * Copyright 2020 Harry Burnett <hburnett@reese.choco>
1878 */
1879
1880#ifndef BABELTRACE_BABY_HPP
1881#define BABELTRACE_BABY_HPP
1882
1883#include <string>
1884#include <unordered_set>
1885#include <utility>
1886
1887namespace life {
1888
1889class Toy;
1890
1891/*
1892 * A baby is a little human.
1893 */
1894class Baby : public Human
1895{
1896public:
1897 using Toys = std::unordered_set<Toy>;
1898
1899 enum class Gender
1900 {
1901 MALE,
1902 FEMALE,
1903 UNKNOWN,
1904 };
1905
1906 Baby() = default;
1907 explicit Baby(const Toys& toys);
1908 Baby(const Baby&) = delete;
1909 Baby(Baby&&) = delete;
1910 Baby& operator=(const Baby&) = delete;
1911 Baby& operator=(Baby&&) = delete;
1912
1913protected:
1914 explicit Baby(Gender initialGender = Gender::UNKNOWN);
1915
1916public:
1917 /*
1918 * Eats `weight` grams of food.
1919 */
1920 void eat(unsigned long weight);
1921
1922 /*
1923 * Sleeps for `duration` seconds.
1924 */
1925 void sleep(double duration);
1926
1927 /*
1928 * Sets this baby's name to `name`.
1929 */
1930 void name(std::string name)
1931 {
1932 _mName = std::move(name);
1933 }
1934
1935 /*
1936 * This baby's name.
1937 */
1938 const std::string& name() const noexcept
1939 {
1940 return _mName;
1941 }
1942
1943protected:
1944 void _addTeeth(unsigned long index);
1945 void _grow(double size) override;
1946
1947private:
1948 std::string _mName {"Paul"};
1949 Toys _mToys;
1950};
1951
1952} // namespace life
1953
1954#endif // BABELTRACE_BABY_HPP
1955----
1956====
This page took 0.14682 seconds and 4 git commands to generate.