···11+= Trace2 API
22+33+The Trace2 API can be used to print debug, performance, and telemetry
44+information to stderr or a file. The Trace2 feature is inactive unless
55+explicitly enabled by enabling one or more Trace2 Targets.
66+77+The Trace2 API is intended to replace the existing (Trace1)
88+printf-style tracing provided by the existing `GIT_TRACE` and
99+`GIT_TRACE_PERFORMANCE` facilities. During initial implementation,
1010+Trace2 and Trace1 may operate in parallel.
1111+1212+The Trace2 API defines a set of high-level messages with known fields,
1313+such as (`start`: `argv`) and (`exit`: {`exit-code`, `elapsed-time`}).
1414+1515+Trace2 instrumentation throughout the Git code base sends Trace2
1616+messages to the enabled Trace2 Targets. Targets transform these
1717+messages content into purpose-specific formats and write events to
1818+their data streams. In this manner, the Trace2 API can drive
1919+many different types of analysis.
2020+2121+Targets are defined using a VTable allowing easy extension to other
2222+formats in the future. This might be used to define a binary format,
2323+for example.
2424+2525+== Trace2 Targets
2626+2727+Trace2 defines the following set of Trace2 Targets.
2828+Format details are given in a later section.
2929+3030+`GIT_TR2` (NORMAL)::
3131+3232+ a simple printf format like GIT_TRACE.
3333++
3434+------------
3535+$ export GIT_TR2=~/log.normal
3636+$ git version
3737+git version 2.20.1.155.g426c96fcdb
3838+------------
3939++
4040+------------
4141+$ cat ~/log.normal
4242+12:28:42.620009 common-main.c:38 version 2.20.1.155.g426c96fcdb
4343+12:28:42.620989 common-main.c:39 start git version
4444+12:28:42.621101 git.c:432 cmd_name version (version)
4545+12:28:42.621215 git.c:662 exit elapsed:0.001227 code:0
4646+12:28:42.621250 trace2/tr2_tgt_normal.c:124 atexit elapsed:0.001265 code:0
4747+------------
4848+4949+`GIT_TR2_PERF` (PERF)::
5050+5151+ a column-based format to replace GIT_TRACE_PERFORMANCE suitable for
5252+ development and testing, possibly to complement tools like gprof.
5353++
5454+------------
5555+$ export GIT_TR2_PERF=~/log.perf
5656+$ git version
5757+git version 2.20.1.155.g426c96fcdb
5858+------------
5959++
6060+------------
6161+$ cat ~/log.perf
6262+12:28:42.620675 common-main.c:38 | d0 | main | version | | | | | 2.20.1.155.g426c96fcdb
6363+12:28:42.621001 common-main.c:39 | d0 | main | start | | | | | git version
6464+12:28:42.621111 git.c:432 | d0 | main | cmd_name | | | | | version (version)
6565+12:28:42.621225 git.c:662 | d0 | main | exit | | 0.001227 | | | code:0
6666+12:28:42.621259 trace2/tr2_tgt_perf.c:211 | d0 | main | atexit | | 0.001265 | | | code:0
6767+------------
6868+6969+`GIT_TR2_EVENT` (EVENT)::
7070+7171+ a JSON-based format of event data suitable for telemetry analysis.
7272++
7373+------------
7474+$ export GIT_TR2_EVENT=~/log.event
7575+$ git version
7676+git version 2.20.1.155.g426c96fcdb
7777+------------
7878++
7979+------------
8080+$ cat ~/log.event
8181+{"event":"version","sid":"1547659722619736-11614","thread":"main","time":"2019-01-16 17:28:42.620713","file":"common-main.c","line":38,"evt":"1","exe":"2.20.1.155.g426c96fcdb"}
8282+{"event":"start","sid":"1547659722619736-11614","thread":"main","time":"2019-01-16 17:28:42.621027","file":"common-main.c","line":39,"argv":["git","version"]}
8383+{"event":"cmd_name","sid":"1547659722619736-11614","thread":"main","time":"2019-01-16 17:28:42.621122","file":"git.c","line":432,"name":"version","hierarchy":"version"}
8484+{"event":"exit","sid":"1547659722619736-11614","thread":"main","time":"2019-01-16 17:28:42.621236","file":"git.c","line":662,"t_abs":0.001227,"code":0}
8585+{"event":"atexit","sid":"1547659722619736-11614","thread":"main","time":"2019-01-16 17:28:42.621268","file":"trace2/tr2_tgt_event.c","line":163,"t_abs":0.001265,"code":0}
8686+------------
8787+8888+== Enabling a Target
8989+9090+A Trace2 Target is enabled when the corresponding environment variable
9191+(`GIT_TR2`, `GIT_TR2_PERF`, or `GIT_TR2_EVENT`) is set. The following
9292+values are recognized.
9393+9494+`0`::
9595+`false`::
9696+9797+ Disables the target.
9898+9999+`1`::
100100+`true`::
101101+102102+ Enables the target and writes stream to `STDERR`.
103103+104104+`[2-9]`::
105105+106106+ Enables the target and writes to the already opened file descriptor.
107107+108108+`<absolute-pathname>`::
109109+110110+ Enables the target, opens and writes to the file in append mode.
111111+112112+`af_unix:[<socket_type>:]<absolute-pathname>`::
113113+114114+ Enables the target, opens and writes to a Unix Domain Socket
115115+ (on platforms that support them).
116116++
117117+Socket type can be either `stream` or `dgram`. If the socket type is
118118+omitted, Git will try both.
119119+120120+== Trace2 API
121121+122122+All public Trace2 functions and macros are defined in `trace2.h` and
123123+`trace2.c`. All public symbols are prefixed with `trace2_`.
124124+125125+There are no public Trace2 data structures.
126126+127127+The Trace2 code also defines a set of private functions and data types
128128+in the `trace2/` directory. These symbols are prefixed with `tr2_`
129129+and should only be used by functions in `trace2.c`.
130130+131131+== Conventions for Public Functions and Macros
132132+133133+The functions defined by the Trace2 API are declared and documented
134134+in `trace2.h`. It defines the API functions and wrapper macros for
135135+Trace2.
136136+137137+Some functions have a `_fl()` suffix to indicate that they take `file`
138138+and `line-number` arguments.
139139+140140+Some functions have a `_va_fl()` suffix to indicate that they also
141141+take a `va_list` argument.
142142+143143+Some functions have a `_printf_fl()` suffix to indicate that they also
144144+take a varargs argument.
145145+146146+There are CPP wrapper macros and ifdefs to hide most of these details.
147147+See `trace2.h` for more details. The following discussion will only
148148+describe the simplified forms.
149149+150150+== Public API
151151+152152+All Trace2 API functions send a messsage to all of the active
153153+Trace2 Targets. This section describes the set of available
154154+messages.
155155+156156+It helps to divide these functions into groups for discussion
157157+purposes.
158158+159159+=== Basic Command Messages
160160+161161+These are concerned with the lifetime of the overall git process.
162162+163163+`void trace2_initialize()`::
164164+165165+ Determines if any Trace2 Targets should be enabled and
166166+ initializes the Trace2 facility. This includes starting the
167167+ elapsed time clocks and thread local storage (TLS).
168168++
169169+This function emits a "version" message containing the version of git
170170+and the Trace2 protocol.
171171++
172172+This function should be called from `main()` as early as possible in
173173+the life of the process.
174174+175175+`int trace2_is_enabled()`::
176176+177177+ Returns 1 if Trace2 is enabled (at least one target is
178178+ active).
179179+180180+`void trace2_cmd_start(int argc, const char **argv)`::
181181+182182+ Emits a "start" message containing the process command line
183183+ arguments.
184184+185185+`int trace2_cmd_exit(int exit_code)`::
186186+187187+ Emits an "exit" message containing the process exit-code and
188188+ elapsed time.
189189++
190190+Returns the exit-code.
191191+192192+`void trace2_cmd_error(const char *fmt, va_list ap)`::
193193+194194+ Emits an "error" message containing a formatted error message.
195195+196196+`void trace2_cmd_path(const char *pathname)`::
197197+198198+ Emits a "cmd_path" message with the full pathname of the
199199+ current process.
200200+201201+=== Command Detail Messages
202202+203203+These are concerned with describing the specific Git command
204204+after the command line, config, and environment are inspected.
205205+206206+`void trace2_cmd_name(const char *name)`::
207207+208208+ Emits a "cmd_name" message with the canonical name of the
209209+ command, for example "status" or "checkout".
210210+211211+`void trace2_cmd_mode(const char *mode)`::
212212+213213+ Emits a "cmd_mode" message with a qualifier name to further
214214+ describe the current git command.
215215++
216216+This message is intended to be used with git commands having multiple
217217+major modes. For example, a "checkout" command can checkout a new
218218+branch or it can checkout a single file, so the checkout code could
219219+emit a cmd_mode message of "branch" or "file".
220220+221221+`void trace2_cmd_alias(const char *alias, const char **argv_expansion)`::
222222+223223+ Emits an "alias" message containing the alias used and the
224224+ argument expansion.
225225+226226+`void trace2_def_param(const char *parameter, const char *value)`::
227227+228228+ Emits a "def_param" message containing a key/value pair.
229229++
230230+This message is intended to report some global aspect of the current
231231+command, such as a configuration setting or command line switch that
232232+significantly affects program performance or behavior, such as
233233+`core.abbrev`, `status.showUntrackedFiles`, or `--no-ahead-behind`.
234234+235235+`void trace2_cmd_list_config()`::
236236+237237+ Emits a "def_param" messages for "important" configuration
238238+ settings.
239239++
240240+The environment variable `GIT_TR2_CONFIG_PARAMS` can be set to a
241241+list of patterns of important configuration settings, for example:
242242+`core.*,remote.*.url`. This function will iterate over all config
243243+settings and emit a "def_param" message for each match.
244244+245245+`void trace2_cmd_set_config(const char *key, const char *value)`::
246246+247247+ Emits a "def_param" message for a specific configuration
248248+ setting IFF it matches the `GIT_TR2_CONFIG_PARAMS` pattern.
249249++
250250+This is used to hook into `git_config_set()` and catch any
251251+configuration changes and update a value previously reported by
252252+`trace2_cmd_list_config()`.
253253+254254+`void trace2_def_repo(struct repository *repo)`::
255255+256256+ Registers a repository with the Trace2 layer. Assigns a
257257+ unique "repo-id" to `repo->trace2_repo_id`.
258258++
259259+Emits a "worktree" messages containing the repo-id and the worktree
260260+pathname.
261261++
262262+Region and data messages (described later) may refer to this repo-id.
263263++
264264+The main/top-level repository will have repo-id value 1 (aka "r1").
265265++
266266+The repo-id field is in anticipation of future in-proc submodule
267267+repositories.
268268+269269+=== Child Process Messages
270270+271271+These are concerned with the various spawned child processes,
272272+including shell scripts, git commands, editors, pagers, and hooks.
273273+274274+`void trace2_child_start(struct child_process *cmd)`::
275275+276276+ Emits a "child_start" message containing the "child-id",
277277+ "child-argv", and "child-classification".
278278++
279279+Before calling this, set `cmd->trace2_child_class` to a name
280280+describing the type of child process, for example "editor".
281281++
282282+This function assigns a unique "child-id" to `cmd->trace2_child_id`.
283283+This field is used later during the "child_exit" message to associate
284284+it with the "child_start" message.
285285++
286286+This function should be called before spawning the child process.
287287+288288+`void trace2_child_exit(struct child_proess *cmd, int child_exit_code)`::
289289+290290+ Emits a "child_exit" message containing the "child-id",
291291+ the child's elapsed time and exit-code.
292292++
293293+The reported elapsed time includes the process creation overhead and
294294+time spend waiting for it to exit, so it may be slightly longer than
295295+the time reported by the child itself.
296296++
297297+This function should be called after reaping the child process.
298298+299299+`int trace2_exec(const char *exe, const char **argv)`::
300300+301301+ Emits a "exec" message containing the "exec-id" and the
302302+ argv of the new process.
303303++
304304+This function should be called before calling one of the `exec()`
305305+variants, such as `execvp()`.
306306++
307307+This function returns a unique "exec-id". This value is used later
308308+if the exec() fails and a "exec-result" message is necessary.
309309+310310+`void trace2_exec_result(int exec_id, int error_code)`::
311311+312312+ Emits a "exec_result" message containing the "exec-id"
313313+ and the error code.
314314++
315315+On Unix-based systems, `exec()` does not return if successful.
316316+This message is used to indicate that the `exec()` failed and
317317+that the current program is continuing.
318318+319319+=== Git Thread Messages
320320+321321+These messages are concerned with Git thread usage.
322322+323323+`void trace2_thread_start(const char *thread_name)`::
324324+325325+ Emits a "thread_start" message.
326326++
327327+The `thread_name` field should be a descriptive name, such as the
328328+unique name of the thread-proc. A unique "thread-id" will be added
329329+to the name to uniquely identify thread instances.
330330++
331331+Region and data messages (described later) may refer to this thread
332332+name.
333333++
334334+This function must be called by the thread-proc of the new thread
335335+(so that TLS data is properly initialized) and not by the caller
336336+of `pthread_create()`.
337337+338338+`void trace2_thread_exit()`::
339339+340340+ Emits a "thread_exit" message containing the thread name
341341+ and the thread elapsed time.
342342++
343343+This function must be called by the thread-proc before it returns
344344+(so that the coorect TLS data is used and cleaned up. It should
345345+not be called by the caller of `pthread_join()`.
346346+347347+=== Region and Data Messages
348348+349349+These are concerned with recording performance data
350350+over regions or spans of code.
351351+352352+`void trace2_region_enter(const char *category, const char *label, const struct repository *repo)`::
353353+354354+`void trace2_region_enter_printf(const char *category, const char *label, const struct repository *repo, const char *fmt, ...)`::
355355+356356+`void trace2_region_enter_printf_va(const char *category, const char *label, const struct repository *repo, const char *fmt, va_list ap)`::
357357+358358+ Emits a thread-relative "region_enter" message with optional
359359+ printf string.
360360++
361361+This function pushes a new region nesting stack level on the current
362362+thread and starts a clock for the new stack frame.
363363++
364364+The `category` field is an arbitrary category name used to classify
365365+regions by feature area, such as "status" or "index". At this time
366366+it is only just printed along with the rest of the message. It may
367367+be used in the future to filter messages.
368368++
369369+The `label` field is an arbitrary label used to describe the activity
370370+being started, such as "read_recursive" or "do_read_index".
371371++
372372+The `repo` field, if set, will be used to get the "repo-id", so that
373373+recursive oerations can be attributed to the correct repository.
374374+375375+`void trace2_region_leave(const char *category, const char *label, const struct repository *repo)`::
376376+377377+`void trace2_region_leave_printf(const char *category, const char *label, const struct repository *repo, const char *fmt, ...)`::
378378+379379+`void trace2_region_leave_printf_va(const char *category, const char *label, const struct repository *repo, const char *fmt, va_list ap)`::
380380+381381+ Emits a thread-relative "region_leave" message with optional
382382+ printf string.
383383++
384384+This function pops the region nesting stack on the current thread
385385+and reports the elapsed time of the stack frame.
386386++
387387+The `category`, `label`, and `repo` fields are the same as above.
388388+The `category` and `label` do not need to match the correpsonding
389389+"region_enter" message, but it makes the data stream easier to
390390+understand.
391391+392392+`void trace2_data_string(const char *category, const struct repository *repo, const char *key, const char * value)`::
393393+394394+`void trace2_data_intmax(const char *category, const struct repository *repo, const char *key, intmax value)`::
395395+396396+`void trace2_data_json(const char *category, const struct repository *repo, const char *key, const struct json_writer *jw)`::
397397+398398+ Emits a region- and thread-relative "data" or "data_json" message.
399399++
400400+This is a key/value pair message containing information about the
401401+current thread, region stack, and repository. This could be used
402402+to print the number of files in a directory during a multi-threaded
403403+recursive tree walk.
404404+405405+`void trace2_printf(const char *fmt, ...)`::
406406+407407+`void trace2_printf_va(const char *fmt, va_list ap)`::
408408+409409+ Emits a region- and thread-relative "printf" message.
410410+411411+== Trace2 Target Formats
412412+413413+=== NORMAL Format
414414+415415+NORMAL format is enabled when the `GIT_TR2` environment variable is
416416+set.
417417+418418+Events are written as lines of the form:
419419+420420+------------
421421+[<time> SP <filename>:<line> SP+] <event-name> [[SP] <event-message>] LF
422422+------------
423423+424424+`<event-name>`::
425425+426426+ is the event name.
427427+428428+`<event-message>`::
429429+ is a free-form printf message intended for human consumption.
430430++
431431+Note that this may contain embedded LF or CRLF characters that are
432432+not escaped, so the event may spill across multiple lines.
433433+434434+If `GIT_TR2_BRIEF` is true, the `time`, `filename`, and `line` fields
435435+are omitted.
436436+437437+This target is intended to be more of a summary (like GIT_TRACE) and
438438+less detailed than the other targets. It ignores thread, region, and
439439+data messages, for example.
440440+441441+=== PERF Format
442442+443443+PERF format is enabled when the `GIT_TR2_PERF` environment variable
444444+is set.
445445+446446+Events are written as lines of the form:
447447+448448+------------
449449+[<time> SP <filename>:<line> SP+
450450+ BAR SP] d<depth> SP
451451+ BAR SP <thread-name> SP+
452452+ BAR SP <event-name> SP+
453453+ BAR SP [r<repo-id>] SP+
454454+ BAR SP [<t_abs>] SP+
455455+ BAR SP [<t_rel>] SP+
456456+ BAR SP [<category>] SP+
457457+ BAR SP DOTS* <perf-event-message>
458458+ LF
459459+------------
460460+461461+`<depth>`::
462462+ is the git process depth. This is the number of parent
463463+ git processes. A top-level git command has depth value "d0".
464464+ A child of it has depth value "d1". A second level child
465465+ has depth value "d2" and so on.
466466+467467+`<thread-name>`::
468468+ is a unique name for the thread. The primary thread
469469+ is called "main". Other thread names are of the form "th%d:%s"
470470+ and include a unique number and the name of the thread-proc.
471471+472472+`<event-name>`::
473473+ is the event name.
474474+475475+`<repo-id>`::
476476+ when present, is a number indicating the repository
477477+ in use. A `def_repo` event is emitted when a repository is
478478+ opened. This defines the repo-id and associated worktree.
479479+ Subsequent repo-specific events will reference this repo-id.
480480++
481481+Currently, this is always "r1" for the main repository.
482482+This field is in anticipation of in-proc submodules in the future.
483483+484484+`<t_abs>`::
485485+ when present, is the absolute time in seconds since the
486486+ program started.
487487+488488+`<t_rel>`::
489489+ when present, is time in seconds relative to the start of
490490+ the current region. For a thread-exit event, it is the elapsed
491491+ time of the thread.
492492+493493+`<category>`::
494494+ is present on region and data events and is used to
495495+ indicate a broad category, such as "index" or "status".
496496+497497+`<perf-event-message>`::
498498+ is a free-form printf message intended for human consumption.
499499+500500+------------
501501+15:33:33.532712 wt-status.c:2310 | d0 | main | region_enter | r1 | 0.126064 | | status | label:print
502502+15:33:33.532712 wt-status.c:2331 | d0 | main | region_leave | r1 | 0.127568 | 0.001504 | status | label:print
503503+------------
504504+505505+If `GIT_TR2_PERF_BRIEF` is true, the `time`, `file`, and `line`
506506+fields are omitted.
507507+508508+------------
509509+d0 | main | region_leave | r1 | 0.011717 | 0.009122 | index | label:preload
510510+------------
511511+512512+The PERF target is intended for interactive performance analysis
513513+during development and is quite noisy.
514514+515515+=== EVENT Format
516516+517517+EVENT format is enabled when the `GIT_TR2_EVENT` environment
518518+variable is set.
519519+520520+Each event is a JSON-object containing multiple key/value pairs
521521+written as a single line and followed by a LF.
522522+523523+------------
524524+'{' <key> ':' <value> [',' <key> ':' <value>]* '}' LF
525525+------------
526526+527527+Some key/value pairs are common to all events and some are
528528+event-specific.
529529+530530+==== Common Key/Value Pairs
531531+532532+The following key/value pairs are common to all events:
533533+534534+------------
535535+{
536536+ "event":"version",
537537+ "sid":"1547659722619736-11614",
538538+ "thread":"main",
539539+ "time":"2019-01-16 17:28:42.620713",
540540+ "file":"common-main.c",
541541+ "line":38,
542542+ ...
543543+}
544544+------------
545545+546546+`"event":<event>`::
547547+ is the event name.
548548+549549+`"sid":<sid>`::
550550+ is the session-id. This is a unique string to identify the
551551+ process instance to allow all events emitted by a process to
552552+ be identified. A session-id is used instead of a PID because
553553+ PIDs are recycled by the OS. For child git processes, the
554554+ session-id is prepended with the session-id of the parent git
555555+ process to allow parent-child relationships to be identified
556556+ during post-processing.
557557+558558+`"thread":<thread>`::
559559+ is the thread name.
560560+561561+`"time":<time>`::
562562+ is the UTC time of the event.
563563+564564+`"file":<filename>`::
565565+ is source file generating the event.
566566+567567+`"line":<line-number>`::
568568+ is the integer source line number generating the event.
569569+570570+`"repo":<repo-id>`::
571571+ when present, is the integer repo-id as described previously.
572572+573573+If `GIT_TR2_EVENT_BRIEF` is true, the `file` and `line` fields are omitted
574574+from all events and the `time` field is only present on the "start" and
575575+"atexit" events.
576576+577577+==== Event-Specific Key/Value Pairs
578578+579579+`"version"`::
580580+ This event gives the version of the executable and the EVENT format.
581581++
582582+------------
583583+{
584584+ "event":"version",
585585+ ...
586586+ "evt":"1", # EVENT format version
587587+ "exe":"2.20.1.155.g426c96fcdb" # git version
588588+}
589589+------------
590590+591591+`"start"`::
592592+ This event contains the complete argv received by main().
593593++
594594+------------
595595+{
596596+ "event":"start",
597597+ ...
598598+ "argv":["git","version"]
599599+}
600600+------------
601601+602602+`"exit"`::
603603+ This event is emitted when git calls `exit()`.
604604++
605605+------------
606606+{
607607+ "event":"exit",
608608+ ...
609609+ "t_abs":0.001227, # elapsed time in seconds
610610+ "code":0 # exit code
611611+}
612612+------------
613613+614614+`"atexit"`::
615615+ This event is emitted by the Trace2 `atexit` routine during
616616+ final shutdown. It should be the last event emitted by the
617617+ process.
618618++
619619+(The elapsed time reported here is greater than the time reported in
620620+the "exit" event because it runs after all other atexit tasks have
621621+completed.)
622622++
623623+------------
624624+{
625625+ "event":"atexit",
626626+ ...
627627+ "t_abs":0.001227, # elapsed time in seconds
628628+ "code":0 # exit code
629629+}
630630+------------
631631+632632+`"signal"`::
633633+ This event is emitted when the program is terminated by a user
634634+ signal. Depending on the platform, the signal event may
635635+ prevent the "atexit" event from being generated.
636636++
637637+------------
638638+{
639639+ "event":"signal",
640640+ ...
641641+ "t_abs":0.001227, # elapsed time in seconds
642642+ "signal":13 # SIGTERM, SIGINT, etc.
643643+}
644644+------------
645645+646646+`"error"`::
647647+ This event is emitted when one of the `error()`, `die()`,
648648+ or `usage()` functions are called.
649649++
650650+------------
651651+{
652652+ "event":"error",
653653+ ...
654654+ "msg":"invalid option: --cahced", # formatted error message
655655+ "fmt":"invalid option: %s" # error format string
656656+}
657657+------------
658658++
659659+The error event may be emitted more than once. The format string
660660+allows post-processors to group errors by type without worrying
661661+about specific error arguments.
662662+663663+`"cmd_path"`::
664664+ This event contains the discovered full path of the git
665665+ executable (on platforms that are configured to resolve it).
666666++
667667+------------
668668+{
669669+ "event":"cmd_path",
670670+ ...
671671+ "path":"C:/work/gfw/git.exe"
672672+}
673673+------------
674674+675675+`"cmd_name"`::
676676+ This event contains the command name for this git process
677677+ and the hierarchy of commands from parent git processes.
678678++
679679+------------
680680+{
681681+ "event":"cmd_name",
682682+ ...
683683+ "name":"pack-objects",
684684+ "hierarchy":"push/pack-objects"
685685+}
686686+------------
687687++
688688+Normally, the "name" field contains the canonical name of the
689689+command. When a canonical name is not available, one of
690690+these special values are used:
691691++
692692+------------
693693+"_query_" # "git --html-path"
694694+"_run_dashed_" # when "git foo" tries to run "git-foo"
695695+"_run_shell_alias_" # alias expansion to a shell command
696696+"_run_git_alias_" # alias expansion to a git command
697697+"_usage_" # usage error
698698+------------
699699+700700+`"cmd_mode"`::
701701+ This event, when present, describes the command variant This
702702+ event may be emitted more than once.
703703++
704704+------------
705705+{
706706+ "event":"cmd_mode",
707707+ ...
708708+ "name":"branch"
709709+}
710710+------------
711711++
712712+The "name" field is an arbitrary string to describe the command mode.
713713+For example, checkout can checkout a branch or an individual file.
714714+And these variations typically have different performance
715715+characteristics that are not comparable.
716716+717717+`"alias"`::
718718+ This event is present when an alias is expanded.
719719++
720720+------------
721721+{
722722+ "event":"alias",
723723+ ...
724724+ "alias":"l", # registered alias
725725+ "argv":["log","--graph"] # alias expansion
726726+}
727727+------------
728728+729729+`"child_start"`::
730730+ This event describes a child process that is about to be
731731+ spawned.
732732++
733733+------------
734734+{
735735+ "event":"child_start",
736736+ ...
737737+ "child_id":2,
738738+ "child_class":"?",
739739+ "use_shell":false,
740740+ "argv":["git","rev-list","--objects","--stdin","--not","--all","--quiet"]
741741+742742+ "hook_name":"<hook_name>" # present when child_class is "hook"
743743+ "cd":"<path>" # present when cd is required
744744+}
745745+------------
746746++
747747+The "child_id" field can be used to match this child_start with the
748748+corresponding child_exit event.
749749++
750750+The "child_class" field is a rough classification, such as "editor",
751751+"pager", "transport/*", and "hook". Unclassified children are classified
752752+with "?".
753753+754754+`"child_exit"`::
755755+ This event is generated after the current process has returned
756756+ from the waitpid() and collected the exit information from the
757757+ child.
758758++
759759+------------
760760+{
761761+ "event":"child_exit",
762762+ ...
763763+ "child_id":2,
764764+ "pid":14708, # child PID
765765+ "code":0, # child exit-code
766766+ "t_rel":0.110605 # observed run-time of child process
767767+}
768768+------------
769769++
770770+Note that the session-id of the child process is not available to
771771+the current/spawning process, so the child's PID is reported here as
772772+a hint for post-processing. (But it is only a hint because the child
773773+proces may be a shell script which doesn't have a session-id.)
774774++
775775+Note that the `t_rel` field contains the observed run time in seconds
776776+for the child process (starting before the fork/exec/spawn and
777777+stopping after the waitpid() and includes OS process creation overhead).
778778+So this time will be slightly larger than the atexit time reported by
779779+the child process itself.
780780+781781+`"exec"`::
782782+ This event is generated before git attempts to `exec()`
783783+ another command rather than starting a child process.
784784++
785785+------------
786786+{
787787+ "event":"exec",
788788+ ...
789789+ "exec_id":0,
790790+ "exe":"git",
791791+ "argv":["foo", "bar"]
792792+}
793793+------------
794794++
795795+The "exec_id" field is a command-unique id and is only useful if the
796796+`exec()` fails and a corresponding exec_result event is generated.
797797+798798+`"exec_result"`::
799799+ This event is generated if the `exec()` fails and control
800800+ returns to the current git command.
801801++
802802+------------
803803+{
804804+ "event":"exec_result",
805805+ ...
806806+ "exec_id":0,
807807+ "code":1 # error code (errno) from exec()
808808+}
809809+------------
810810+811811+`"thread_start"`::
812812+ This event is generated when a thread is started. It is
813813+ generated from *within* the new thread's thread-proc (for TLS
814814+ reasons).
815815++
816816+------------
817817+{
818818+ "event":"thread_start",
819819+ ...
820820+ "thread":"th02:preload_thread" # thread name
821821+}
822822+------------
823823+824824+`"thread_exit"`::
825825+ This event is generated when a thread exits. It is generated
826826+ from *within* the thread's thread-proc (for TLS reasons).
827827++
828828+------------
829829+{
830830+ "event":"thread_exit",
831831+ ...
832832+ "thread":"th02:preload_thread", # thread name
833833+ "t_rel":0.007328 # thread elapsed time
834834+}
835835+------------
836836+837837+`"def_param"`::
838838+ This event is generated to log a global parameter.
839839++
840840+------------
841841+{
842842+ "event":"def_param",
843843+ ...
844844+ "param":"core.abbrev",
845845+ "value":"7"
846846+}
847847+------------
848848+849849+`"def_repo"`::
850850+ This event defines a repo-id and associates it with the root
851851+ of the worktree.
852852++
853853+------------
854854+{
855855+ "event":"def_repo",
856856+ ...
857857+ "repo":1,
858858+ "worktree":"/Users/jeffhost/work/gfw"
859859+}
860860+------------
861861++
862862+As stated earlier, the repo-id is currently always 1, so there will
863863+only be one def_repo event. Later, if in-proc submodules are
864864+supported, a def_repo event should be emitted for each submodule
865865+visited.
866866+867867+`"region_enter"`::
868868+ This event is generated when entering a region.
869869++
870870+------------
871871+{
872872+ "event":"region_enter",
873873+ ...
874874+ "repo":1, # optional
875875+ "nesting":1, # current region stack depth
876876+ "category":"index", # optional
877877+ "label":"do_read_index", # optional
878878+ "msg":".git/index" # optional
879879+}
880880+------------
881881++
882882+The `category` field may be used in a future enhancement to
883883+do category-based filtering.
884884++
885885+The `GIT_TR2_EVENT_NESTING` environment variable can be used to
886886+filter deeply nested regions and data events. It defaults to "2".
887887+888888+`"region_leave"`::
889889+ This event is generated when leaving a region.
890890++
891891+------------
892892+{
893893+ "event":"region_leave",
894894+ ...
895895+ "repo":1, # optional
896896+ "t_rel":0.002876, # time spent in region in seconds
897897+ "nesting":1, # region stack depth
898898+ "category":"index", # optional
899899+ "label":"do_read_index", # optional
900900+ "msg":".git/index" # optional
901901+}
902902+------------
903903+904904+`"data"`::
905905+ This event is generated to log a thread- and region-local
906906+ key/value pair.
907907++
908908+------------
909909+{
910910+ "event":"data",
911911+ ...
912912+ "repo":1, # optional
913913+ "t_abs":0.024107, # absolute elapsed time
914914+ "t_rel":0.001031, # elapsed time in region/thread
915915+ "nesting":2, # region stack depth
916916+ "category":"index",
917917+ "key":"read/cache_nr",
918918+ "value":"3552"
919919+}
920920+------------
921921++
922922+The "value" field may be an integer or a string.
923923+924924+`"data-json"`::
925925+ This event is generated to log a pre-formatted JSON string
926926+ containing structured data.
927927++
928928+------------
929929+{
930930+ "event":"data_json",
931931+ ...
932932+ "repo":1, # optional
933933+ "t_abs":0.015905,
934934+ "t_rel":0.015905,
935935+ "nesting":1,
936936+ "category":"process",
937937+ "key":"windows/ancestry",
938938+ "value":["bash.exe","bash.exe"]
939939+}
940940+------------
941941+942942+== Example Trace2 API Usage
943943+944944+Here is a hypothetical usage of the Trace2 API showing the intended
945945+usage (without worrying about the actual Git details).
946946+947947+Initialization::
948948+949949+ Initialization happens in `main()`. Behind the scenes, an
950950+ `atexit` and `signal` handler are registered.
951951++
952952+----------------
953953+int main(int argc, const char **argv)
954954+{
955955+ int exit_code;
956956+957957+ trace2_initialize();
958958+ trace2_cmd_start(argv);
959959+960960+ exit_code = cmd_main(argc, argv);
961961+962962+ trace2_cmd_exit(exit_code);
963963+964964+ return exit_code;
965965+}
966966+----------------
967967+968968+Command Details::
969969+970970+ After the basics are established, additional command
971971+ information can be sent to Trace2 as it is discovered.
972972++
973973+----------------
974974+int cmd_checkout(int argc, const char **argv)
975975+{
976976+ trace2_cmd_name("checkout");
977977+ trace2_cmd_mode("branch");
978978+ trace2_def_repo(the_repository);
979979+980980+ // emit "def_param" messages for "interesting" config settings.
981981+ trace2_cmd_list_config();
982982+983983+ if (do_something())
984984+ trace2_cmd_error("Path '%s': cannot do something", path);
985985+986986+ return 0;
987987+}
988988+----------------
989989+990990+Child Processes::
991991+992992+ Wrap code spawning child processes.
993993++
994994+----------------
995995+void run_child(...)
996996+{
997997+ int child_exit_code;
998998+ struct child_process cmd = CHILD_PROCESS_INIT;
999999+ ...
10001000+ cmd.trace2_child_class = "editor";
10011001+10021002+ trace2_child_start(&cmd);
10031003+ child_exit_code = spawn_child_and_wait_for_it();
10041004+ trace2_child_exit(&cmd, child_exit_code);
10051005+}
10061006+----------------
10071007++
10081008+For example, the following fetch command spawned ssh, index-pack,
10091009+rev-list, and gc. This example also shows that fetch took
10101010+5.199 seconds and of that 4.932 was in ssh.
10111011++
10121012+----------------
10131013+$ export GIT_TR2_BRIEF=1
10141014+$ export GIT_TR2=~/log.normal
10151015+$ git fetch origin
10161016+...
10171017+----------------
10181018++
10191019+----------------
10201020+$ cat ~/log.normal
10211021+version 2.20.1.vfs.1.1.47.g534dbe1ad1
10221022+start git fetch origin
10231023+worktree /Users/jeffhost/work/gfw
10241024+cmd_name fetch (fetch)
10251025+child_start[0] ssh git@github.com ...
10261026+child_start[1] git index-pack ...
10271027+... (Trace2 events from child processes omitted)
10281028+child_exit[1] pid:14707 code:0 elapsed:0.076353
10291029+child_exit[0] pid:14706 code:0 elapsed:4.931869
10301030+child_start[2] git rev-list ...
10311031+... (Trace2 events from child process omitted)
10321032+child_exit[2] pid:14708 code:0 elapsed:0.110605
10331033+child_start[3] git gc --auto
10341034+... (Trace2 events from child process omitted)
10351035+child_exit[3] pid:14709 code:0 elapsed:0.006240
10361036+exit elapsed:5.198503 code:0
10371037+atexit elapsed:5.198541 code:0
10381038+----------------
10391039++
10401040+When a git process is a (direct or indirect) child of another
10411041+git process, it inherits Trace2 context information. This
10421042+allows the child to print the command hierarchy. This example
10431043+shows gc as child[3] of fetch. When the gc process reports
10441044+its name as "gc", it also reports the hierarchy as "fetch/gc".
10451045+(In this example, trace2 messages from the child process is
10461046+indented for clarity.)
10471047++
10481048+----------------
10491049+$ export GIT_TR2_BRIEF=1
10501050+$ export GIT_TR2=~/log.normal
10511051+$ git fetch origin
10521052+...
10531053+----------------
10541054++
10551055+----------------
10561056+$ cat ~/log.normal
10571057+version 2.20.1.160.g5676107ecd.dirty
10581058+start git fetch official
10591059+worktree /Users/jeffhost/work/gfw
10601060+cmd_name fetch (fetch)
10611061+...
10621062+child_start[3] git gc --auto
10631063+ version 2.20.1.160.g5676107ecd.dirty
10641064+ start /Users/jeffhost/work/gfw/git gc --auto
10651065+ worktree /Users/jeffhost/work/gfw
10661066+ cmd_name gc (fetch/gc)
10671067+ exit elapsed:0.001959 code:0
10681068+ atexit elapsed:0.001997 code:0
10691069+child_exit[3] pid:20303 code:0 elapsed:0.007564
10701070+exit elapsed:3.868938 code:0
10711071+atexit elapsed:3.868970 code:0
10721072+----------------
10731073+10741074+Regions::
10751075+10761076+ Regions can be use to time an interesting section of code.
10771077++
10781078+----------------
10791079+void wt_status_collect(struct wt_status *s)
10801080+{
10811081+ trace2_region_enter("status", "worktrees", s->repo);
10821082+ wt_status_collect_changes_worktree(s);
10831083+ trace2_region_leave("status", "worktrees", s->repo);
10841084+10851085+ trace2_region_enter("status", "index", s->repo);
10861086+ wt_status_collect_changes_index(s);
10871087+ trace2_region_leave("status", "index", s->repo);
10881088+10891089+ trace2_region_enter("status", "untracked", s->repo);
10901090+ wt_status_collect_untracked(s);
10911091+ trace2_region_leave("status", "untracked", s->repo);
10921092+}
10931093+10941094+void wt_status_print(struct wt_status *s)
10951095+{
10961096+ trace2_region_enter("status", "print", s->repo);
10971097+ switch (s->status_format) {
10981098+ ...
10991099+ }
11001100+ trace2_region_leave("status", "print", s->repo);
11011101+}
11021102+----------------
11031103++
11041104+In this example, scanning for untracked files ran from +0.012568 to
11051105++0.027149 (since the process started) and took 0.014581 seconds.
11061106++
11071107+----------------
11081108+$ export GIT_TR2_PERF_BRIEF=1
11091109+$ export GIT_TR2_PERF=~/log.perf
11101110+$ git status
11111111+...
11121112+11131113+$ cat ~/log.perf
11141114+d0 | main | version | | | | | 2.20.1.160.g5676107ecd.dirty
11151115+d0 | main | start | | | | | git status
11161116+d0 | main | def_repo | r1 | | | | worktree:/Users/jeffhost/work/gfw
11171117+d0 | main | cmd_name | | | | | status (status)
11181118+...
11191119+d0 | main | region_enter | r1 | 0.010988 | | status | label:worktrees
11201120+d0 | main | region_leave | r1 | 0.011236 | 0.000248 | status | label:worktrees
11211121+d0 | main | region_enter | r1 | 0.011260 | | status | label:index
11221122+d0 | main | region_leave | r1 | 0.012542 | 0.001282 | status | label:index
11231123+d0 | main | region_enter | r1 | 0.012568 | | status | label:untracked
11241124+d0 | main | region_leave | r1 | 0.027149 | 0.014581 | status | label:untracked
11251125+d0 | main | region_enter | r1 | 0.027411 | | status | label:print
11261126+d0 | main | region_leave | r1 | 0.028741 | 0.001330 | status | label:print
11271127+d0 | main | exit | | 0.028778 | | | code:0
11281128+d0 | main | atexit | | 0.028809 | | | code:0
11291129+----------------
11301130++
11311131+Regions may be nested. This causes messages to be indented in the
11321132+PERF target, for example.
11331133+Elapsed times are relative to the start of the correpsonding nesting
11341134+level as expected. For example, if we add region message to:
11351135++
11361136+----------------
11371137+static enum path_treatment read_directory_recursive(struct dir_struct *dir,
11381138+ struct index_state *istate, const char *base, int baselen,
11391139+ struct untracked_cache_dir *untracked, int check_only,
11401140+ int stop_at_first_file, const struct pathspec *pathspec)
11411141+{
11421142+ enum path_treatment state, subdir_state, dir_state = path_none;
11431143+11441144+ trace2_region_enter_printf("dir", "read_recursive", NULL, "%.*s", baselen, base);
11451145+ ...
11461146+ trace2_region_leave_printf("dir", "read_recursive", NULL, "%.*s", baselen, base);
11471147+ return dir_state;
11481148+}
11491149+----------------
11501150++
11511151+We can further investigate the time spent scanning for untracked files.
11521152++
11531153+----------------
11541154+$ export GIT_TR2_PERF_BRIEF=1
11551155+$ export GIT_TR2_PERF=~/log.perf
11561156+$ git status
11571157+...
11581158+$ cat ~/log.perf
11591159+d0 | main | version | | | | | 2.20.1.162.gb4ccea44db.dirty
11601160+d0 | main | start | | | | | git status
11611161+d0 | main | def_repo | r1 | | | | worktree:/Users/jeffhost/work/gfw
11621162+d0 | main | cmd_name | | | | | status (status)
11631163+...
11641164+d0 | main | region_enter | r1 | 0.015047 | | status | label:untracked
11651165+d0 | main | region_enter | | 0.015132 | | dir | ..label:read_recursive
11661166+d0 | main | region_enter | | 0.016341 | | dir | ....label:read_recursive vcs-svn/
11671167+d0 | main | region_leave | | 0.016422 | 0.000081 | dir | ....label:read_recursive vcs-svn/
11681168+d0 | main | region_enter | | 0.016446 | | dir | ....label:read_recursive xdiff/
11691169+d0 | main | region_leave | | 0.016522 | 0.000076 | dir | ....label:read_recursive xdiff/
11701170+d0 | main | region_enter | | 0.016612 | | dir | ....label:read_recursive git-gui/
11711171+d0 | main | region_enter | | 0.016698 | | dir | ......label:read_recursive git-gui/po/
11721172+d0 | main | region_enter | | 0.016810 | | dir | ........label:read_recursive git-gui/po/glossary/
11731173+d0 | main | region_leave | | 0.016863 | 0.000053 | dir | ........label:read_recursive git-gui/po/glossary/
11741174+...
11751175+d0 | main | region_enter | | 0.031876 | | dir | ....label:read_recursive builtin/
11761176+d0 | main | region_leave | | 0.032270 | 0.000394 | dir | ....label:read_recursive builtin/
11771177+d0 | main | region_leave | | 0.032414 | 0.017282 | dir | ..label:read_recursive
11781178+d0 | main | region_leave | r1 | 0.032454 | 0.017407 | status | label:untracked
11791179+...
11801180+d0 | main | exit | | 0.034279 | | | code:0
11811181+d0 | main | atexit | | 0.034322 | | | code:0
11821182+----------------
11831183++
11841184+Trace2 regions are similar to the existing trace_performance_enter()
11851185+and trace_performance_leave() routines, but are thread safe and
11861186+maintain per-thread stacks of timers.
11871187+11881188+Data Messages::
11891189+11901190+ Data messages added to a region.
11911191++
11921192+----------------
11931193+int read_index_from(struct index_state *istate, const char *path,
11941194+ const char *gitdir)
11951195+{
11961196+ trace2_region_enter_printf("index", "do_read_index", the_repository, "%s", path);
11971197+11981198+ ...
11991199+12001200+ trace2_data_intmax("index", the_repository, "read/version", istate->version);
12011201+ trace2_data_intmax("index", the_repository, "read/cache_nr", istate->cache_nr);
12021202+12031203+ trace2_region_leave_printf("index", "do_read_index", the_repository, "%s", path);
12041204+}
12051205+----------------
12061206++
12071207+This example shows that the index contained 3552 entries.
12081208++
12091209+----------------
12101210+$ export GIT_TR2_PERF_BRIEF=1
12111211+$ export GIT_TR2_PERF=~/log.perf
12121212+$ git status
12131213+...
12141214+$ cat ~/log.perf
12151215+d0 | main | version | | | | | 2.20.1.156.gf9916ae094.dirty
12161216+d0 | main | start | | | | | git status
12171217+d0 | main | def_repo | r1 | | | | worktree:/Users/jeffhost/work/gfw
12181218+d0 | main | cmd_name | | | | | status (status)
12191219+d0 | main | region_enter | r1 | 0.001791 | | index | label:do_read_index .git/index
12201220+d0 | main | data | r1 | 0.002494 | 0.000703 | index | ..read/version:2
12211221+d0 | main | data | r1 | 0.002520 | 0.000729 | index | ..read/cache_nr:3552
12221222+d0 | main | region_leave | r1 | 0.002539 | 0.000748 | index | label:do_read_index .git/index
12231223+...
12241224+----------------
12251225+12261226+Thread Events::
12271227+12281228+ Thread messages added to a thread-proc.
12291229++
12301230+For example, the multithreaded preload-index code can be
12311231+instrumented with a region around the thread pool and then
12321232+per-thread start and exit events within the threadproc.
12331233++
12341234+----------------
12351235+static void *preload_thread(void *_data)
12361236+{
12371237+ // start the per-thread clock and emit a message.
12381238+ trace2_thread_start("preload_thread");
12391239+12401240+ // report which chunk of the array this thread was assigned.
12411241+ trace2_data_intmax("index", the_repository, "offset", p->offset);
12421242+ trace2_data_intmax("index", the_repository, "count", nr);
12431243+12441244+ do {
12451245+ ...
12461246+ } while (--nr > 0);
12471247+ ...
12481248+12491249+ // report elapsed time taken by this thread.
12501250+ trace2_thread_exit();
12511251+ return NULL;
12521252+}
12531253+12541254+void preload_index(struct index_state *index,
12551255+ const struct pathspec *pathspec,
12561256+ unsigned int refresh_flags)
12571257+{
12581258+ trace2_region_enter("index", "preload", the_repository);
12591259+12601260+ for (i = 0; i < threads; i++) {
12611261+ ... /* create thread */
12621262+ }
12631263+12641264+ for (i = 0; i < threads; i++) {
12651265+ ... /* join thread */
12661266+ }
12671267+12681268+ trace2_region_leave("index", "preload", the_repository);
12691269+}
12701270+----------------
12711271++
12721272+In this example preload_index() was executed by the `main` thread
12731273+and started the `preload` region. Seven threads, named
12741274+`th01:preload_thread` through `th07:preload_thread`, were started.
12751275+Events from each thread are atomically appended to the shared target
12761276+stream as they occur so they may appear in random order with respect
12771277+other threads. Finally, the main thread waits for the threads to
12781278+finish and leaves the region.
12791279++
12801280+Data events are tagged with the active thread name. They are used
12811281+to report the per-thread parameters.
12821282++
12831283+----------------
12841284+$ export GIT_TR2_PERF_BRIEF=1
12851285+$ export GIT_TR2_PERF=~/log.perf
12861286+$ git status
12871287+...
12881288+$ cat ~/log.perf
12891289+...
12901290+d0 | main | region_enter | r1 | 0.002595 | | index | label:preload
12911291+d0 | th01:preload_thread | thread_start | | 0.002699 | | |
12921292+d0 | th02:preload_thread | thread_start | | 0.002721 | | |
12931293+d0 | th01:preload_thread | data | r1 | 0.002736 | 0.000037 | index | offset:0
12941294+d0 | th02:preload_thread | data | r1 | 0.002751 | 0.000030 | index | offset:2032
12951295+d0 | th03:preload_thread | thread_start | | 0.002711 | | |
12961296+d0 | th06:preload_thread | thread_start | | 0.002739 | | |
12971297+d0 | th01:preload_thread | data | r1 | 0.002766 | 0.000067 | index | count:508
12981298+d0 | th06:preload_thread | data | r1 | 0.002856 | 0.000117 | index | offset:2540
12991299+d0 | th03:preload_thread | data | r1 | 0.002824 | 0.000113 | index | offset:1016
13001300+d0 | th04:preload_thread | thread_start | | 0.002710 | | |
13011301+d0 | th02:preload_thread | data | r1 | 0.002779 | 0.000058 | index | count:508
13021302+d0 | th06:preload_thread | data | r1 | 0.002966 | 0.000227 | index | count:508
13031303+d0 | th07:preload_thread | thread_start | | 0.002741 | | |
13041304+d0 | th07:preload_thread | data | r1 | 0.003017 | 0.000276 | index | offset:3048
13051305+d0 | th05:preload_thread | thread_start | | 0.002712 | | |
13061306+d0 | th05:preload_thread | data | r1 | 0.003067 | 0.000355 | index | offset:1524
13071307+d0 | th05:preload_thread | data | r1 | 0.003090 | 0.000378 | index | count:508
13081308+d0 | th07:preload_thread | data | r1 | 0.003037 | 0.000296 | index | count:504
13091309+d0 | th03:preload_thread | data | r1 | 0.002971 | 0.000260 | index | count:508
13101310+d0 | th04:preload_thread | data | r1 | 0.002983 | 0.000273 | index | offset:508
13111311+d0 | th04:preload_thread | data | r1 | 0.007311 | 0.004601 | index | count:508
13121312+d0 | th05:preload_thread | thread_exit | | 0.008781 | 0.006069 | |
13131313+d0 | th01:preload_thread | thread_exit | | 0.009561 | 0.006862 | |
13141314+d0 | th03:preload_thread | thread_exit | | 0.009742 | 0.007031 | |
13151315+d0 | th06:preload_thread | thread_exit | | 0.009820 | 0.007081 | |
13161316+d0 | th02:preload_thread | thread_exit | | 0.010274 | 0.007553 | |
13171317+d0 | th07:preload_thread | thread_exit | | 0.010477 | 0.007736 | |
13181318+d0 | th04:preload_thread | thread_exit | | 0.011657 | 0.008947 | |
13191319+d0 | main | region_leave | r1 | 0.011717 | 0.009122 | index | label:preload
13201320+...
13211321+d0 | main | exit | | 0.029996 | | | code:0
13221322+d0 | main | atexit | | 0.030027 | | | code:0
13231323+----------------
13241324++
13251325+In this example, the preload region took 0.009122 seconds. The 7 threads
13261326+took between 0.006069 and 0.008947 seconds to work on their portion of
13271327+the index. Thread "th01" worked on 508 items at offset 0. Thread "th02"
13281328+worked on 508 items at offset 2032. Thread "th04" worked on 508 itemts
13291329+at offset 508.
13301330++
13311331+This example also shows that thread names are assigned in a racy manner
13321332+as each thread starts and allocates TLS storage.
13331333+13341334+== Future Work
13351335+13361336+=== Relationship to the Existing Trace Api (api-trace.txt)
13371337+13381338+There are a few issues to resolve before we can completely
13391339+switch to Trace2.
13401340+13411341+* Updating existing tests that assume GIT_TRACE format messages.
13421342+13431343+* How to best handle custom GIT_TRACE_<key> messages?
13441344+13451345+** The GIT_TRACE_<key> mechanism allows each <key> to write to a
13461346+different file (in addition to just stderr).
13471347+13481348+** Do we want to maintain that ability or simply write to the existing
13491349+Trace2 targets (and convert <key> to a "category").