diff --git a/README.md b/README.md index 762745ba..4758aaa5 100644 --- a/README.md +++ b/README.md @@ -5,6 +5,7 @@ This repository enumerates standard RISC-V instruction opcodes and control/statu Artifacts like `encoding.h`, `latex-tables`, etc., from this repo are used in tools and projects such as Spike, PK, and the RISC-V Manual. ## Table of Contents + 1. [Project Structure](#project-structure) 2. [File Naming Policy](#file-naming-policy) 3. [Encoding Syntax](#encoding-syntax) @@ -15,6 +16,7 @@ Artifacts like `encoding.h`, `latex-tables`, etc., from this repo are used in to 8. [Contributing](#contributing) --- + ## Project Structure ```bash @@ -27,16 +29,18 @@ Artifacts like `encoding.h`, `latex-tables`, etc., from this repo are used in to ├── rv* # instruction opcode files └── unratified # contains unratified instruction opcode files ``` + --- + ## File Naming Policy This project follows a specific file naming convention for instruction encodings: -* **`rv_x`**: Instructions common to both 32-bit and 64-bit modes of extension `X`. -* **`rv32_x`**: Instructions specific to `rv32x` (e.g., `brev8`). -* **`rv64_x`**: Instructions specific to `rv64x` (e.g., `addw`). -* **`rv_x_y`**: Instructions valid when both extensions `X` and `Y` are enabled. Canonical ordering as specified by the RISC-V spec should be followed. -* **`unratified/`**: Contains instruction encodings that are not ratified yet, following the same policy. +- **`rv_x`**: Instructions common to both 32-bit and 64-bit modes of extension `X`. +- **`rv32_x`**: Instructions specific to `rv32x` (e.g., `brev8`). +- **`rv64_x`**: Instructions specific to `rv64x` (e.g., `addw`). +- **`rv_x_y`**: Instructions valid when both extensions `X` and `Y` are enabled. Canonical ordering as specified by the RISC-V spec should be followed. +- **`unratified/`**: Contains instruction encodings that are not ratified yet, following the same policy. For instructions present in multiple extensions where the spec is vague, the encoding should be placed in the canonically ordered first extension and imported into others using `$import`. @@ -46,48 +50,53 @@ For instructions present in multiple extensions where the spec is vague, the enc Instruction encoding files in this project use the following syntax: -* **Keywords**: `$import` and `$pseudo_op` are keywords used to indicate special operations. -* **Operators**: `::` defines relationships between extensions and instructions; `..` defines bit ranges. -* **Comments**: Use `#` for comments. Inline comments are not supported. +- **Keywords**: `$import` and `$pseudo_op` are keywords used to indicate special operations. +- **Operators**: `::` defines relationships between extensions and instructions; `..` defines bit ranges. +- **Comments**: Use `#` for comments. Inline comments are not supported. ### Instruction Categories 1. **Regular Instructions**: Instructions with unique opcodes. - - **Syntax**: ` ` - - **Example**: - ```plaintext - lui rd imm20 6..2=0x0D 1..0=3 - beq bimm12hi rs1 rs2 bimm12lo 14..12=0 6..2=0x18 1..0=3 - ``` - - **Bit Encoding Types**: - - *Single Bit Assignment*: `=` - - *Range Assignment*: `..=` + - **Syntax**: ` ` + - **Example**: + + ```plaintext + lui rd imm20 6..2=0x0D 1..0=3 + beq bimm12hi rs1 rs2 bimm12lo 14..12=0 6..2=0x18 1..0=3 + ``` + + - **Bit Encoding Types**: + - _Single Bit Assignment_: `=` + - _Range Assignment_: `..=` 2. **Pseudo Instructions** (`$pseudo_op`): Aliases for regular instructions with restricted bit encodings. - - **Syntax**: `$pseudo_op :: ` - - ``: Specifies the extension which contains the base instruction. - - ``: Indicates the name of the instruction this pseudo-instruction is an alias of. - - The remaining fields are the same as the regular instruction syntax, where all the arguments and the fields of the pseudo instruction are specified. - - **Example**: - ```plaintext - $pseudo_op rv_zicsr::csrrs frflags rd 19..15=0 31..20=0x001 14..12=2 6..2=0x1C 1..0=3 - ``` - - **Recommendation**: If a ratified instruction is a `$pseudo_op` of a regular unratified instruction, it is recommended to maintain this `$pseudo_op` relationship. Define the new instruction as a `$pseudo_op` of the unratified regular instruction to avoid overlapping opcodes for users experimenting with unratified extensions. + - **Syntax**: `$pseudo_op :: ` + - ``: Specifies the extension which contains the base instruction. + - ``: Indicates the name of the instruction this pseudo-instruction is an alias of. + - The remaining fields are the same as the regular instruction syntax, where all the arguments and the fields of the pseudo instruction are specified. + - **Example**: + + ```plaintext + $pseudo_op rv_zicsr::csrrs frflags rd 19..15=0 31..20=0x001 14..12=2 6..2=0x1C 1..0=3 + ``` + + - **Recommendation**: If a ratified instruction is a `$pseudo_op` of a regular unratified instruction, it is recommended to maintain this `$pseudo_op` relationship. Define the new instruction as a `$pseudo_op` of the unratified regular instruction to avoid overlapping opcodes for users experimenting with unratified extensions. 3. **Imported Instructions** (`$import`): Instructions borrowed from another extension. - - These are instructions borrowed from an extension into a new or different extension/sub-extension. Only regular instructions can be imported. Pseudo-ops or already imported instructions cannot be imported. - - **Syntax**: `$import :: ` - - **Example**: - ```plaintext - $import rv32_zkne::aes32esmi - ``` + - These are instructions borrowed from an extension into a new or different extension/sub-extension. Only regular instructions can be imported. Pseudo-ops or already imported instructions cannot be imported. + - **Syntax**: `$import :: ` + - **Example**: + ```plaintext + $import rv32_zkne::aes32esmi + ``` ### Restrictions -* Pseudo-ops or already imported instructions cannot be imported again. -* A base instruction for a pseudo-op cannot be a pseudo-op itself. +- Pseudo-ops or already imported instructions cannot be imported again. +- A base instruction for a pseudo-op cannot be a pseudo-op itself. + --- ## Flow for parse.py @@ -95,18 +104,22 @@ Instruction encoding files in this project use the following syntax: The `parse.py` Python file is used to perform checks on the current set of instruction encodings and also generates multiple artifacts: LaTeX tables, `encoding.h` header file, etc. This section provides a brief overview of the flow within the Python file. 1. **Initial Setup**: + - `parse.py` creates a list of all `rv*` files currently checked into the repo (including those inside the `unratified` directory). - It starts parsing each file line by line. 2. **First Pass - Regular Instructions**: + - Capture only regular instructions and ignore imported or pseudo instructions. - **Checks performed**: - - For range-assignment syntax, the *msb* (most significant bit) position must be higher than the *lsb* (least significant bit) position. - - The value of the range must be representable in the space identified by *msb* and *lsb*. + + - For range-assignment syntax, the _msb_ (most significant bit) position must be higher than the _lsb_ (least significant bit) position. + - The value of the range must be representable in the space identified by _msb_ and _lsb_. - Values for the same bit positions should not be defined multiple times. - All bit positions must be accounted for (either as arguments or constant value fields). - **Dictionary Creation**: + - Create a dictionary for each instruction with the following fields: - `encoding`: A 32-bit string defining the encoding of the instruction. `-` is used to represent instruction argument fields. - `extension`: String indicating which extension/filename this instruction was picked from. @@ -117,31 +130,33 @@ The `parse.py` Python file is used to perform checks on the current set of instr - Add the dictionary elements to a main `instr_dict` dictionary under the instruction node. This process continues until all regular instructions have been processed. 3. **Second Pass - Pseudo Instructions**: + - Process `$pseudo_op` instructions. - **Checks performed**: - - Verify if the *base-instruction* of the pseudo instruction exists in the relevant extension/filename. + - Verify if the _base-instruction_ of the pseudo instruction exists in the relevant extension/filename. - The remaining part of the syntax undergoes the same checks as above. - - If the checks pass and the *base-instruction* is not already added to the main `instr_dict`, then add the pseudo-instruction to the list. + - If the checks pass and the _base-instruction_ is not already added to the main `instr_dict`, then add the pseudo-instruction to the list. 4. **Third Pass - Imported Instructions**: + - Process imported instructions. 5. **Special Case**: - - If the *base-instruction* for a pseudo-instruction is not present in the main `instr_dict` after the first pass, it may be due to processing only a subset of extensions where the *base-instruction* is not included. + - If the _base-instruction_ for a pseudo-instruction is not present in the main `instr_dict` after the first pass, it may be due to processing only a subset of extensions where the _base-instruction_ is not included. ## Artifact Generation and Usage The `parse.py` script can generate the following artifacts: -* **`instr_dict.yaml`**: Contains the main dictionary `instr_dict` in YAML format. Note that dots in instruction names are replaced with underscores in this YAML file. -* **`encoding.out.h`**: A header file used by tools such as Spike, PK, etc. -* **`instr-table.tex`**: LaTeX table of instructions for the RISC-V unprivileged specification. -* **`priv-instr-table.tex`**: LaTeX table of instructions for the RISC-V privileged specification. -* **`inst.chisel`**: Chisel code for decoding instructions. -* **`inst.sverilog`**: SystemVerilog code for decoding instructions. -* **`inst.rs`**: Rust code containing mask and match variables for all instructions. -* **`inst.spinalhdl`**: SpinalHDL code for decoding instructions. -* **`inst.go`**: Go code for decoding instructions. +- **`instr_dict.yaml`**: Contains the main dictionary `instr_dict` in YAML format. Note that dots in instruction names are replaced with underscores in this YAML file. +- **`encoding.out.h`**: A header file used by tools such as Spike, PK, etc. +- **`instr-table.tex`**: LaTeX table of instructions for the RISC-V unprivileged specification. +- **`priv-instr-table.tex`**: LaTeX table of instructions for the RISC-V privileged specification. +- **`inst.chisel`**: Chisel code for decoding instructions. +- **`inst.sverilog`**: SystemVerilog code for decoding instructions. +- **`inst.rs`**: Rust code containing mask and match variables for all instructions. +- **`inst.spinalhdl`**: SpinalHDL code for decoding instructions. +- **`inst.go`**: Go code for decoding instructions. ### Prerequisites @@ -151,22 +166,27 @@ Ensure you have the required Python dependencies installed. Run the following co sudo apt-get install python3-pip pip3 install -r requirements.txt ``` + ### Generating Artifacts + To generate all artifacts for all instructions currently checked in, run make from the root directory. This will produce the following output: - ```plaintext - Running with args : ['./parse.py', '-c', '-go', '-chisel', '-sverilog', '-rust', '-latex', '-spinalhdl', 'rv*', 'unratified/rv*'] - Extensions selected : ['rv*', 'unratified/rv*'] - INFO:: encoding.out.h generated successfully - INFO:: inst.chisel generated successfully - INFO:: inst.spinalhdl generated successfully - INFO:: inst.sverilog generated successfully - INFO:: inst.rs generated successfully - INFO:: inst.go generated successfully - INFO:: instr-table.tex generated successfully - INFO:: priv-instr-table.tex generated successfully - ``` +```plaintext +Running with args : ['./parse.py', '-c', '-go', '-chisel', '-sverilog', '-rust', '-latex', '-spinalhdl', 'rv*', 'unratified/rv*'] +Extensions selected : ['rv*', 'unratified/rv*'] +INFO:: encoding.out.h generated successfully +INFO:: inst.chisel generated successfully +INFO:: inst.spinalhdl generated successfully +INFO:: inst.sverilog generated successfully +INFO:: inst.rs generated successfully +INFO:: inst.go generated successfully +INFO:: instr-table.tex generated successfully +INFO:: priv-instr-table.tex generated successfully + +``` + ### Selecting Specific Extensions + By default, all extensions are enabled. To select a subset of extensions, modify the EXTENSIONS variable in the Makefile to include only the filenames of interest. For example, to include only the I and M extensions: For example if you want only the I and M extensions you can do the following: @@ -186,15 +206,18 @@ This will produce the following output: INFO:: instr-table.tex generated successfully INFO:: priv-instr-table.tex generated successfully ``` + ### Generating Specific Artifacts To generate specific artifacts, use one or more of the following targets: -* `c` -* `rust` -* `chisel` -* `sverilog` -* `latex` +- `inst.c`,`rs`,`chisel`,`sverilog`, `instr-table.tex` + +For example, if you want to generate using Chisel and Rust, run the following command: + +```bash + make inst.chisel inst.rs +``` ### Cleaning Up @@ -203,35 +226,37 @@ To remove all generated artifacts, use the `clean` target: ```bash make clean ``` + --- ## Adding a New Extension To add a new extension of instructions, follow these steps: -1. **Create the Extension File**: - - Create a new `rv*` file according to the policy defined in the [File Structure](#file-naming-policy). +1. **Create the Extension File**: + +- Create a new `rv*` file according to the policy defined in the [File Structure](#file-naming-policy). 2. **Run Checks and Generate Artifacts**: - - From the root directory, run the `make` command to ensure that all checks pass and that all artifacts are generated correctly. - - A successful run will produce the following output: - - ```plaintext - Running with args : ['./parse.py', '-c', '-chisel', '-sverilog', '-rust', '-latex', 'rv*', 'unratified/rv*'] - Extensions selected : ['rv*', 'unratified/rv*'] - INFO:: encoding.out.h generated successfully - INFO:: inst.chisel generated successfully - INFO:: inst.sverilog generated successfully - INFO:: inst.rs generated successfully - INFO:: instr-table.tex generated successfully - INFO:: priv-instr-table.tex generated successfully - ``` + +- From the root directory, run the `make` command to ensure that all checks pass and that all artifacts are generated correctly. +- A successful run will produce the following output: + + ```plaintext + Running with args : ['./parse.py', '-c', '-chisel', '-sverilog', '-rust', '-latex', 'rv*', 'unratified/rv*'] + Extensions selected : ['rv*', 'unratified/rv*'] + INFO:: encoding.out.h generated successfully + INFO:: inst.chisel generated successfully + INFO:: inst.sverilog generated successfully + INFO:: inst.rs generated successfully + INFO:: instr-table.tex generated successfully + INFO:: priv-instr-table.tex generated successfully + ``` 3. **Submit for Review**: - Create a pull request (PR) to submit your changes for review. -Ensure you follow these steps carefully to integrate the new extension properly. ---- +## Ensure you follow these steps carefully to integrate the new extension properly. ## How do I find where an instruction is defined? @@ -242,31 +267,42 @@ You can locate the definition of an instruction using one of the following metho grep "^\s*" rv* unratified/rv* ``` 2. **Using `make`**: - - Run make to generate the instr_dict.yaml file. - - Open instr_dict.yaml and search for the instruction. - - The extension field in the file will indicate which file the instruction was picked from. + +- Run make to generate the instr_dict.yaml file. +- Open instr_dict.yaml and search for the instruction. +- The extension field in the file will indicate which file the instruction was picked from. + --- + ## Debugging + To enable debug logs in parse.py: 1. Modify the logging level in parse.py: + ```python level=logging.INFO ``` + Change it to: + ```python level=logging.DEBUG ``` + 2. Example debug output: - ```bash - DEBUG:: Parsing File: ./rv_i - DEBUG:: Processing line: lui rd imm20 6..2=0x0D 1..0=3 - ``` + +```bash +DEBUG:: Parsing File: ./rv_i +DEBUG:: Processing line: lui rd imm20 6..2=0x0D 1..0=3 +``` + --- + ## Contributing - If you wish to contribute to this project: - - Open a pull request (PR) or issue. - - Ensure that all tests pass. - - Follow the repository’s coding guidelines. +If you wish to contribute to this project: +- Open a pull request (PR) or issue. +- Ensure that all tests pass. +- Follow the repository’s coding guidelines.