1 | //===-- xray_fdr_log_records.h -------------------------------------------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // This file is a part of XRay, a function call tracing system. |
10 | // |
11 | //===----------------------------------------------------------------------===// |
12 | #ifndef XRAY_XRAY_FDR_LOG_RECORDS_H |
13 | #define XRAY_XRAY_FDR_LOG_RECORDS_H |
14 | #include <cstdint> |
15 | |
16 | namespace __xray { |
17 | |
18 | enum class RecordType : uint8_t { Function, Metadata }; |
19 | |
20 | // A MetadataRecord encodes the kind of record in its first byte, and have 15 |
21 | // additional bytes in the end to hold free-form data. |
22 | struct alignas(16) MetadataRecord { |
23 | // A MetadataRecord must always have a type of 1. |
24 | /* RecordType */ uint8_t Type : 1; |
25 | |
26 | // Each kind of record is represented as a 7-bit value (even though we use an |
27 | // unsigned 8-bit enum class to do so). |
28 | enum class RecordKinds : uint8_t { |
29 | NewBuffer, |
30 | EndOfBuffer, |
31 | NewCPUId, |
32 | TSCWrap, |
33 | WalltimeMarker, |
34 | CustomEventMarker, |
35 | CallArgument, |
36 | BufferExtents, |
37 | TypedEventMarker, |
38 | Pid, |
39 | }; |
40 | |
41 | // Use 7 bits to identify this record type. |
42 | /* RecordKinds */ uint8_t RecordKind : 7; |
43 | char Data[15]; |
44 | } __attribute__((packed)); |
45 | |
46 | static_assert(sizeof(MetadataRecord) == 16, "Wrong size for MetadataRecord." ); |
47 | |
48 | struct alignas(8) FunctionRecord { |
49 | // A FunctionRecord must always have a type of 0. |
50 | /* RecordType */ uint8_t Type : 1; |
51 | enum class RecordKinds { |
52 | FunctionEnter = 0x00, |
53 | FunctionExit = 0x01, |
54 | FunctionTailExit = 0x02, |
55 | }; |
56 | /* RecordKinds */ uint8_t RecordKind : 3; |
57 | |
58 | // We only use 28 bits of the function ID, so that we can use as few bytes as |
59 | // possible. This means we only support 2^28 (268,435,456) unique function ids |
60 | // in a single binary. |
61 | int FuncId : 28; |
62 | |
63 | // We use another 4 bytes to hold the delta between the previous entry's TSC. |
64 | // In case we've found that the distance is greater than the allowable 32 bits |
65 | // (either because we are running in a different CPU and the TSC might be |
66 | // different then), we should use a MetadataRecord before this FunctionRecord |
67 | // that will contain the full TSC for that CPU, and keep this to 0. |
68 | uint32_t TSCDelta; |
69 | } __attribute__((packed)); |
70 | |
71 | static_assert(sizeof(FunctionRecord) == 8, "Wrong size for FunctionRecord." ); |
72 | |
73 | } // namespace __xray |
74 | |
75 | #endif // XRAY_XRAY_FDR_LOG_RECORDS_H |
76 | |