Skip to content

Commit

Permalink
Merge pull request #7375 from BradleyWood/x86docs
Browse files Browse the repository at this point in the history
x86: Update code generator documentation
  • Loading branch information
0xdaryl authored Oct 4, 2024
2 parents d4f0498 + 5388e41 commit 27f3631
Showing 1 changed file with 72 additions and 4 deletions.
76 changes: 72 additions & 4 deletions doc/compiler/x/OpCodeEncoding.md
Original file line number Diff line number Diff line change
Expand Up @@ -57,10 +57,13 @@ enum TR_OpCodeVEX_L : uint8_t
VEX_L256 = 0x1,
VEX_L512 = 0x2,
VEX_L___ = 0x3, // Instruction does not support VEX encoding
EVEX_L128 = 0x4,
EVEX_L256 = 0x5,
EVEX_L512 = 0x6,
};
enum TR_OpCodeVEX_v : uint8_t
{
VEX_vNONE = 0x0,
VEX_vNONE = 0x0, // typical of SIMD instructions with a single source operand
VEX_vReg_ = 0x1,
};
enum TR_InstructionREX_W : uint8_t
Expand All @@ -74,6 +77,8 @@ enum TR_OpCodePrefix : uint8_t
PREFIX_66 = 0x1,
PREFIX_F3 = 0x2,
PREFIX_F2 = 0x3,
PREFIX_66_F2 = 0x4,
PREFIX_66_F3 = 0x5,
};
enum TR_OpCodeEscape : uint8_t
{
Expand Down Expand Up @@ -115,8 +120,71 @@ enum TR_OpCodeImmediate : uint8_t
2.1 Convert the 3-byte VEX to 2-byte VEX if possible
3. Write the VEX prefix

## Working with SIMD instructions

With the support of AVX, AVX2, and AVX-512, OMR supports generation
of vector instructions for 128/256/512-bit vectors. SIMD instructions
can also be generated using VEX, EVEX, and legacy SSE encoding methods.
Each vector length can use the encoding methods shown below.

```
128 - SSE, VEX_L128, EVEX_L128
256 - VEX_L256, EVEX_L256
512 - EVEX_L512
```

### Generating SIMD instructions

Generating SIMD instructions is simple. You may optionally specify the method by which to encode
the instruction. Shown below are a list of possible encoding methods.

1. Legacy implies SSE instruction
2. Default (or not specified) uses information already specified on the instruction

```c++
typedef enum
{
VEX_L128 = 0x0,
VEX_L256 = 0x1,
Default = 0x2,
Legacy = 0x3,
EVEX_L128 = 0x4,
EVEX_L256 = 0x5,
EVEX_L512 = 0x6,
Bad = 0x7
} Encoding;
```
#### 128-Bit vmovdqu example
```c++
generateRegRegInstruction(TR::InstOpCode::MOVDQURegReg, node, resultReg, valueReg, cg, OMR::X86::VEX_L128);
```

By not specifying an opcode encoding method, the code generator will use the method specified
on the opcode's definition. If the instruction is labeled as VEX_L128, and AVX is not supported,
legacy SSE encoding will be used.
```c++
generateRegRegInstruction(TR::InstOpCode::MOVDQURegReg, node, resultReg, valueReg, cg);
```
#### Dynamically determining the best encoding method
To dynamically find the best encoding method you may [call](https://github.com/eclipse/omr/blob/e0dde228728f4dd7cf48e407971e01b0fcc130fe/compiler/x/codegen/OMRInstOpCode.hpp#L504)
```OMR::InstOpCode::getSIMDEncoding(&cpu, vl)```.
This query uses flags stored on the instruction to determine the best encoding method for
the given instruction, CPU, and vector length.
For example, this method will return ```OMR::X86::Bad``` if the opcode is not supported at the given
vector length and will also throw an assertion failure if the instruction is missing
CPU feature requirements flags.
```c++
movOpcode = TR::InstOpCode::MOVDQURegReg;
OMR::X86::Encoding movEncoding = movOpcode.getSIMDEncoding(&cg->comp()->target().cpu, vl);
```

# FUTURE WORK

## Generate AVX-512 Instruction
1. Obtain REX prefix from operand and set REX.W according to rex_w field.
2. Setup EVEX structure and write it.
## Generate APX Instructions
1. REX 2 Prefix
2. Enhanced EVEX prefix

0 comments on commit 27f3631

Please sign in to comment.