1# Compiled Code Metainfo 2 3## Overview 4 5Metainfo is an information that aims to provide reg-to-stack mapping for virtual registers. It is needed for stack 6unwinding process to restore CFrame at specific PC. 7 8When native code calls runtime or another code that can call runtime, we must provide approach to restore virtual 9registers which are live during this call instruction. Since all virtual regisetrs should be saved on the stack before 10we call the runtime, we can save information in which stack slot specific vreg is live. 11 12Metainfo is placed together with compiled code: 13``` 14 +-------------+ 15 | CodePrefix | 16 | +-------------------+ 17 | | magic | 18 | | code_info_offset | 19 | | code_info_size | 20 +-------------+-------------------+ 21 | | <-- Method::CompiledCodeEntrypoint 22 | Code | 23 | | 24 +-------------+-----------------+ 25 | CodeInfo | CodeInfoHeader | 26 | |-----------------+----------------------+ 27 | | | StackMap | 28 | | | InlineInfo | 29 | | | Roots Reg Mask | 30 | | | Roots Stack Mask | 31 | | Bit Tables | Method indexes | 32 | | | VRegs mask | 33 | | | VRegs map | 34 | | | VRegs catalogue | 35 | | | Implicit Nullchecks | 36 | | | Constants | 37 |-------------+-----------------+----------------------+ 38``` 39 40## Bit table 41 42Columns width is in a bits. 43 44First row is a BitTable's header, that describe rows count and columns width. 45 46``` 47+------------+----------------+----------------+----------------+ 48| Rows count | Column 0 width | . . . | Column N width | 49+------------+----------------+----------------+----------------+ 50``` 51Header is followed by data, which is a rows with fixed length. Row length is equal to sum of columns width. 52 53Column width can't be greater than 32 bits, because `BitTableBuilder` class, that aims to build bit tables, 54use `uint32_t` for a single table element. 55 56Example: 57``` 58 Rows 59 count Columns 60+-----+-----+-----+-------+-----+ 61| 5 | 2 | 0 | 15 | 8 | ; Header 62+-----+-----+-----+-------+-----+ 63 | 2 | - | 31547 | 23 | ; 0 row 64 | 1 | - | 12 | 241 | ; 1 row 65 | 1 | - | 128 | 1 | ; 2 row 66 | 2 | - | 0 | 24 | ; 3 row 67 | 0 | - | 4587 | 0 | ; 4 row 68 +-----+-----+-------+-----+ 69``` 70Here, we have 4 columns and 5 rows. Number of rows is defeined in the first digit in a header, 71number of columns is determined in compile time. 72 73Row size is 25 bits, that is sum of columns width: 2 + 0 + 15 + 8. 74Column width is determined by maximum value in a table, e.g. for 2th column it is zero row, that has value 31547. This value fits in 15 bits. 75 76So, the size of this table's data is 25 * 5 = 125 bits = 15.625 bytes. 77 78## Bitmap table 79 80Bitmap table is a Bit table with one column, that doesn't have 32-bits limitation for the width. 81 82## Numbers packing 83 84For number compressisng following approach is used. 85 86The first four bits determine the variable length of the encoded number: 87- Values 0..11 represent the result as-is, with no further following bits. 88- Values 12..15 mean the result is in the next 8/16/24/32-bits respectively. 89 90Example for numbers (2, 0, 15, 254874): 91``` 92+---------+---------+---------+---------+---------+---------+ 93| 0 byte | 1 byte | 2 byte | 3 byte | 4 byte | 5 byte | 94+---------+---------+---------+---------+---------+---------+ 95| 2 | 0 | 12 | 14 | 15 | 254874 | 96+---------+---------+---------+---------+---------+---------+ 97``` 98 99## Code header 100 101Code headers describes structure of the compiled code and metainfo. 102 103| Field | Description | 104|-----|----| 105| PROPERTIES | Properties of the code info (e.g. frame size) | 106| CALLEE_REG_MASK | Specifies registers mask that is saved in the method | 107| CALLEE_FP_REG_MASK | Specifies fp registers mask that is saved in the method | 108| TABLE_MASK | Bit mask of existing bit tables | 109| VREGS_COUNT | Number of virtual registers in reg map | 110 111## Tables 112 113### 1. StackMap 114 115| Field | Description | 116|-----|----| 117| PROPERTIES | Define properties of the stackmap, currently, it contains only one flag: is_osr | 118| NATIVE_PC | Native address to which this stackmap corresponds | 119| BYTECODE_PC | Bytecode address to which this stackmap corresponds | 120| ROOTS_REG_MASK_INDEX | Mask of the CPU registers that hold managed objects | 121| ROOTS_STACK_MASK_INDEX | Mask of the stack slots that hold managed objects | 122| INLINE_INFO_INDEX | Inline information for the stackmap | 123| VREG_MASK_INDEX | Mask of the virtual registers, that are modified from the last stackmap to the current one | 124| VREG_MAP_INDEX | Map of the virtual registers, that are modified from the last stackmap to the current one | 125 126> NOTE: fields with `_INDEX` in the end of the name contain only index of the record in the corresponding table. 127 128### 2. InlineInfo 129 130| Field | Description | 131|-----|----| 132| IS_LAST | Whether this inlined method is a last, i.e. is it a leaf | 133| BYTECODE_PC | Bytecode address to which this inline info corresponds | 134| METHOD_ID_INDEX | Index of the method id in the Methods table | 135| METHOD_HI | Hi 32-bit part of the method pointer (actual only for jit) | 136| METHOD_LOW | Low 32-bit part of the method pointer (actual only for jit) | 137| VREGS_COUNT | Number of virtual registers, that belongs to this inlined method | 138 139Stackmap and all its inlined infos use same Virtual registers map, but separate it via `VREGS_COUNT` field of inline infoes. 140 141Example for inline chain `method 0` -> `inlines 1` -> `inlines 3`: 142``` 143------------- 144vreg 0 145vreg 1 ; method 0: CodeInfoHeader::VREGS_COUNT=3 146vreg 2 147--------------- 148vreg 3 ; method 1: InlineInfo::VREGS_COUNT=1 149--------------- 150vreg 4 151vreg 5 ; method 3: InlineInfo::VREGS_COUNT=2 152``` 153 154### 3. Roots Reg Mask 155 156This is a Bitmap table, where column is a bit mask, that determines which CPU register holds a managed object. 157 158### 4. Roots Stack Mask 159 160This is a Bitmap table, where column is a bit mask, that determines which stack slot holds a managed object. 161 162### 5. Method indexes 163 164Holds single column - method index within a pandafile. It was moved to the separate table due to better deduplication. 165 166### 6. VRegs mask 167 168This is a Bitmap table, where column is a bit mask, that determines which virtual register is modified in the StackMap. 169 170### 7. VRegs map 171 172Holds single column - index of the VReg description in the `VRegs catalogue` table. 173 174### 8. VRegs catalogue 175 176| Field | Description | 177|-----|----| 178| INFO | Virtual register description | 179| VALUE | Virtual register value | 180 181Virtual register description has the following fields: 182- **Location** - where vreg is stored: stack slot, CPU register or constant. 183- **Type** - type of the value: OBJECT, INT32, INT64, FLOAT32, FLOAT64, BOOL. 184- **IsAccumulator** - whethre vreg is accumulator. 185- **Index** - index of the virtual register. 186 187Value of the `VALUE` field depends on the value of `Location` field: 188- CPU register: number of CPU register 189- stack slot: number of the slot within a frame 190- constant: index of a row within `Constants` table 191 192### 9. Implicit nullchecks 193 194This is a table with offsets: removed NullCheck instruction position and corresponding SlowPath position. 195The table helps the signal handler to find the SlowPath address to continue execution after a segmentation fault. 196 197| Field | Description | 198|-----|----| 199| INST_NATIVE_PC | NullCheck instruction position. | 200| SLOW_PATH_NATIVE_PC | NullCheck SlowPath position. | 201 202### 10. Constants 203 204This table contains only one column - constant value.