Memory trace samples from DynamoRIO's drmemtrace tracer for its trace analysis framework.
The memory address tracer we use is part of the drmemtrace open-source tracing and analysis framework, which is part of the DynamoRIO dynamic binary instrumentation framework. Here we summarize the tracing format. See the tracing and analysis framework documentation for further information.
A trace contains a sequence of user-mode instruction and memory fetches for each thread in a target application. The insruction encodings are also included. Each 32KB block of thread data has a timestamp and records which cpu it executed on, allowing reconstructing the thread interleaving at that granularity.
The trace format is described further at https://dynamorio.org/sec_drcachesim_format.html. A simple example trace analysis tool that reads the trace format is basic_counts.cpp.
A human-readable view of a sample trace highlighting thread switches and a signal handler:
--------------------------------------------------
$ bin64/drrun -t drmemtrace -indir drmemtrace.threadsig.[0-9]*.dir -tool view -view_syntax intel 2>&1 | less
Output format:
<--record#-> <--instr#->: <---tid---> <record details>
------------------------------------------------------------
           1           0:     2765542 <marker: version 5>
           2           0:     2765542 <marker: filetype 0xe40>
           3           0:     2765542 <marker: cache line size 64>
           4           0:     2765542 <marker: chunk instruction count 10000000>
           5           0:     2765542 <marker: page size 4096>
<...>
       47444       37769:     2765543 ifetch       6 byte(s) @ 0x0000000000404a78 0f 84 5b 01 00 00    jz     $0x0000000000404bd9 (untaken)
       47445       37770:     2765543 ifetch       5 byte(s) @ 0x0000000000404a7e 48 8b 5c 24 08       mov    0x08(%rsp)[8byte] -> %rbx
       47446       37770:     2765543 read         8 byte(s) @ 0x00007fdf64e2adc8 by PC 0x0000000000404a7e
       47447       37771:     2765543 ifetch       5 byte(s) @ 0x0000000000404a83 be 18 00 00 00       mov    $0x00000018 -> %esi
       47448       37772:     2765543 ifetch       5 byte(s) @ 0x0000000000404a88 b8 11 01 00 00       mov    $0x00000111 -> %eax
       47449       37773:     2765543 ifetch       7 byte(s) @ 0x0000000000404a8d 48 8d bb e0 02 00 00 lea    0x000002e0(%rbx) -> %rdi
       47450       37774:     2765543 ifetch       2 byte(s) @ 0x0000000000404a94 0f 05                syscall  -> %rcx %r11
       47451       37774:     2765543 <marker: system call 273>
------------------------------------------------------------
       47452       37774:     2765542 <marker: timestamp 13334804716609227>
       47453       37774:     2765542 <marker: tid 2765542 on core 7>
       47454       37775:     2765542 ifetch       1 byte(s) @ 0x000000000040257d 55                   push   %rbp %rsp -> %rsp 0xfffffff8(%rsp)[8byte]
       47455       37775:     2765542 write        8 byte(s) @ 0x00007ffd90c13950 by PC 0x000000000040257d
       47456       37776:     2765542 ifetch       3 byte(s) @ 0x000000000040257e 48 89 e5             mov    %rsp -> %rbp
       47457       37777:     2765542 ifetch       3 byte(s) @ 0x0000000000402581 89 7d fc             mov    %edi -> 0xfffffffc(%rbp)[4byte]
       47458       37777:     2765542 write        4 byte(s) @ 0x00007ffd90c1394c by PC 0x0000000000402581
<...>
       47564       37842:     2765542 ifetch       5 byte(s) @ 0x0000000000402bcf e8 da 01 00 00       call   $0x0000000000402dae %rsp -> %rsp 0xfffffff8(%rsp)[8byte]
       47565       37842:     2765542 write        8 byte(s) @ 0x00007ffd90c14638 by PC 0x0000000000402bcf
       47566       37842:     2765542 <marker: kernel xfer from 0x402dae to handler>
       47567       37842:     2765542 <marker: timestamp 13334804716613173>
       47568       37842:     2765542 <marker: tid 2765542 on core 7>
       47569       37843:     2765542 ifetch       1 byte(s) @ 0x000000000040257d 55                   push   %rbp %rsp -> %rsp 0xfffffff8(%rsp)[8byte]
       47570       37843:     2765542 write        8 byte(s) @ 0x00007ffd90c13930 by PC 0x000000000040257d
       47571       37844:     2765542 ifetch       3 byte(s) @ 0x000000000040257e 48 89 e5             mov    %rsp -> %rbp
       47572       37845:     2765542 ifetch       3 byte(s) @ 0x0000000000402581 89 7d fc             mov    %edi -> 0xfffffffc(%rbp)[4byte]
       47573       37845:     2765542 write        4 byte(s) @ 0x00007ffd90c1392c by PC 0x0000000000402581
       47574       37846:     2765542 ifetch       4 byte(s) @ 0x0000000000402584 48 89 75 f0          mov    %rsi -> 0xfffffff0(%rbp)[8byte]
       47575       37846:     2765542 write        8 byte(s) @ 0x00007ffd90c13920 by PC 0x0000000000402584
       47576       37847:     2765542 ifetch       4 byte(s) @ 0x0000000000402588 48 89 55 e8          mov    %rdx -> 0xffffffe8(%rbp)[8byte]
       47577       37847:     2765542 write        8 byte(s) @ 0x00007ffd90c13918 by PC 0x0000000000402588
       47578       37848:     2765542 ifetch       4 byte(s) @ 0x000000000040258c 83 7d fc 1a          cmp    0xfffffffc(%rbp)[4byte] $0x0000001a
       47579       37848:     2765542 read         4 byte(s) @ 0x00007ffd90c1392c by PC 0x000000000040258c
       47580       37849:     2765542 ifetch       2 byte(s) @ 0x0000000000402590 75 0f                jnz    $0x00000000004025a1 (untaken)
       47581       37850:     2765542 ifetch       6 byte(s) @ 0x0000000000402592 8b 05 5c 0f 0e 00    mov    <rel> 0x00000000004e34f4[4byte] -> %eax
       47582       37850:     2765542 read         4 byte(s) @ 0x00000000004e34f4 by PC 0x0000000000402592
       47583       37851:     2765542 ifetch       3 byte(s) @ 0x0000000000402598 83 c0 01             add    $0x00000001 %eax -> %eax
       47584       37852:     2765542 ifetch       6 byte(s) @ 0x000000000040259b 89 05 53 0f 0e 00    mov    %eax -> <rel> 0x00000000004e34f4[4byte]
       47585       37852:     2765542 write        4 byte(s) @ 0x00000000004e34f4 by PC 0x000000000040259b
       47586       37853:     2765542 ifetch       1 byte(s) @ 0x00000000004025a1 90                   nop
       47587       37854:     2765542 ifetch       1 byte(s) @ 0x00000000004025a2 5d                   pop    %rsp (%rsp)[8byte] -> %rbp %rsp
       47588       37854:     2765542 read         8 byte(s) @ 0x00007ffd90c13930 by PC 0x00000000004025a2
       47589       37855:     2765542 ifetch       1 byte(s) @ 0x00000000004025a3 c3                   ret    %rsp (%rsp)[8byte] -> %rsp (target 0x407bb0)
       47590       37855:     2765542 read         8 byte(s) @ 0x00007ffd90c13938 by PC 0x00000000004025a3
       47591       37856:     2765542 ifetch       7 byte(s) @ 0x0000000000407bb0 48 c7 c0 0f 00 00 00 mov    $0x000000000000000f -> %rax
       47592       37857:     2765542 ifetch       2 byte(s) @ 0x0000000000407bb7 0f 05                syscall  -> %rcx %r11
       47593       37857:     2765542 <marker: system call 15>
       47594       37857:     2765542 <marker: timestamp 13334804716613183>
       47595       37857:     2765542 <marker: tid 2765542 on core 7>
       47596       37857:     2765542 <marker: syscall xfer from 0x407bb9>
       47597       37857:     2765542 <marker: timestamp 13334804716613190>
       47598       37857:     2765542 <marker: tid 2765542 on core 7>
       47599       37858:     2765542 ifetch       1 byte(s) @ 0x0000000000402dae 55                   push   %rbp %rsp -> %rsp 0xfffffff8(%rsp)[8byte]
       47600       37858:     2765542 write        8 byte(s) @ 0x00007ffd90c14630 by PC 0x0000000000402dae
<...>
--------------------------------------------------
It is a series of user-mode instruction fetch, data fetch, and metadata entries. The fetches contain addresses and sizes and, for instruction fetches, instruction encodings. The addresses are all virtual (it is possible to gather physical addresses in some circumstances). The metadata "markers" indicate things like which core a thread executed on, timestamps, an arriving signal causing a PC discontinuity, etc.
For using a trace in a core simulator, the provided instruction encodings can be decoded into opcodes and operands as is done with the "view" tool above. This can be done with DynamoRIO's decoder, as in the provided sample tool opcode_mix.cpp. Other aspects of the trace which help core simulation are discussed in our documentation.