Advanced RISC-V Assembly Highlight for Vim and Neovim
Overview
This Vim plugin is designed to enhance your RISC-V assembly coding experience. It achieves this by offering comprehensive syntax highlighting and providing support for both standard and custom RISC-V extensions, including instructions and registers. Additionally, the plugin is configurable by using RISC-V ISA extension naming conventions, recognizes GNU Assembly (GAS) directives, and incorporates smart indent functionality.
Features
-
Syntax Highlighting:
- Full coverage of standard RISC-V instructions and registers for improved code readability.
- Clear and intuitive color coding to distinguish different elements in your RISC-V assembly code.
-
Customization:
- Add custom RISC-V extensions consist of instructions and CSRs to tailor the plugin to your specific needs.
- Follows RISC-V ISA Extension Naming Conventions to enable certain extensions, ensuring compatibility and standardization.
-
GAS Directives:
- Recognizes and highlights GNU Assembly (GAS) directives for a seamless coding experience with popular RISC-V assemblers.
-
Smart Indent:
- Provides smart indent functionality for improved code structure and alignment.
Installation
VimPlug
- Add the following text to the vim-plug section in your
.vimrc
:
Plug 'henry-hsieh/riscv-asm-vim'
- Reload
.vimrc
and run:PlugInstall
in Vim.
Vundle
- Add the following text to the vundle section in your
.vimrc
:
Plugin 'henry-hsieh/riscv-asm-vim'
-
There are two methods to install the plugin.
-
Reload
.vimrc
and run:PlugInstall
in Vim. -
Use command line:
-
vim +PluginInstall +qall
Pathogen
- Use command line:
cd ~/.vim/bundle
git clone https://github.com/henry-hsieh/riscv-asm-vim
Packer.nvim
- Add the following to your packer startup function:
use {
"henry-hsieh/riscv-asm-vim",
}
-
Install the plugin:
- Reload
init.lua
and run:PackerSync
in Neovim.
- Reload
lazy.nvim
- Add the following to your lazy.nvim startup function:
{
"henry-hsieh/riscv-asm-vim",
ft = { "riscv_asm" },
}
-
Install the plugin:
- Reload
init.lua
and run:Lazy install
in Neovim. - Or if you are using install missing plugins, just restart Neovim.
- Reload
Usage
-
Any file with a
.S
extension will be loaded as RISC-V assembly syntax. Note that this will override default asm syntax. -
Set filetype in Vim.
set ft=riscv_asm
- Add modeline in your files.
// vim: ft=riscv_asm
Options
To set the option of this plugin, you can use either Vimscript or Lua (Neovim only). Put the options in your Vim/Neovim startup file.
Highlight All Supported RISC-V ISA
Vimscript:
let g:riscv_asm_all_enable = 1
Neovim Lua:
vim.g.riscv_asm_all_enable = true
If the option is defined (set to any value), the plugin will ignore the value of g:riscv_asm_isa
and highlight all instructions and registers supported by the plugin. If an invalid sequence in g:riscv_asm_isa
is detected, the plugin will automatically highlight all instructions and registers, too.
Specify the Extensions You Want to Highlight
Vimscript:
let g:riscv_asm_isa = 'RV64GC'
Neovim Lua:
vim.g.riscv_asm_isa = 'RV64GC'
The value of g:riscv_asm_isa
aligns with RISC-V ISA extension naming conventions, allowing Vim to highlight specified RISC-V ISAs and extensions. By default, the plugin activates only the RV64GC extension, resulting in the highlighting of RV64I, M, A, F, D, C, Zicsr, and Zifencei instructions and registers.
While the version numbers of extensions can be parsed, the plugin currently does not utilize them. Here are examples of combinations of ISA extensions and version numbers:
rv32i2
(RV32I v2.0)rv32i2p2
(RV32I v2.2)rv32i2_p2
(RV32I v2.0, P v2.0)rv32i2p2p1
(RV32I v2.2, P v1.0)rv32i2p2_p1p2
(RV32I v2.2, P v1.2)
It's important to note that the ISA name is case insensitive, and underscores may provide hints to the parser but are not required in all situations.
Currently Supported Extensions
- Base Integer ISAs or Profiles
Base ISA / Profile | Full Name | Supported Version |
---|---|---|
RV32E |
Embedded 32-bit integer | 2.0 |
RV32I |
32-bit integer | 2.1 |
RV64E |
Embedded 64-bit integer | 2.0 |
RV64I |
64-bit integer | 2.1 |
RV128I |
128-bit integer | 1.7 |
RVI20U32 |
RVI20U32 Profile | |
RVI20U64 |
RVI20U64 Profile | |
RVA20U64 |
RVA20U64 Profile | |
RVA20S64 |
RVA20S64 Profile | |
RVA22U64 |
RVA22U64 Profile | |
RVA22S64 |
RVA22S64 Profile | |
RVA23U64 |
RVA23U64 Profile | |
RVA23S64 |
RVA23S64 Profile | |
RVB23U64 |
RVB23U64 Profile | |
RVB23S64 |
RVB23S64 Profile | |
RVM23U32 |
RVM23U32 Profile |
- Unprivileged Extensions
Extension | Full Name | Supported Version |
---|---|---|
M |
Integer Multiplication and Division | 2.0 |
A |
Atomics | 2.1 |
F |
Single-Precision Floating-Point | 2.2 |
D |
Double-Precision Floating-Point | 2.2 |
G |
General | |
Q |
Quad-Precision Floating-Point | 2.2 |
C |
16-bit Compressed Instructions | 2.0 |
V |
Vector Instructions | 1.1 |
- Privileged Extensions
Extension | Full Name | Supported Version |
---|---|---|
H |
Hypervisor Extension | 1.0 |
- Additional Unprivileged Extensions
Extension | Full Name | Supported Version |
---|---|---|
Zic64b |
Cache Block Size Is 64 Bytes | |
Zicbom |
Cache-Block Management Instructions | 1.0 |
Zicbop |
Cache-Block Prefetch Instructions | 1.0 |
Zicboz |
Cache-Block Zero Instructions | 1.0 |
Ziccamoa |
Main Memory Supports All Atomics in A | |
Ziccif |
Main Memory Supports Instruction Fetch With Atomicity Requirement | |
Zicclsm |
Main Memory Supports Misaligned Loads/Stores | |
Ziccrse |
Main Memory Supports Forward Progress on LR/SC Sequences | |
Zicfilp |
Landing Pad | 0.4 |
Zicfiss |
Shadow Stack | 0.4 |
Zicntr |
Base Counters and Timers | 2.0 |
Zicond |
Integer Conditional Operations Extension | 1.0 |
Zicsr |
Control and Status Register Access | 2.0 |
Zifencei |
Instruction-Fetch Fence | 2.0 |
Zihintntl |
Non-Temporal Locality Hints | 1.0 |
Zihintpause |
Pause Hint | 2.0 |
Zihpm |
Hardware Performance Counters | 2.0 |
Zimop |
May-Be-Operations Extension | 0.1 |
Zmmul |
Multiplication Without Division | 1.0 |
Za128rs |
Reservation Set Size of 128 Bytes | |
Za64rs |
Reservation Set Size of 64 Bytes | |
Zabha |
Byte and Halfword Atomic Memory Operations | 1.0 |
Zacas |
Atomic Compare-and-Swap Instructions | 1.0 |
Zawrs |
Wait-On-Reservation-Set Extension | 1.0 |
Zam |
Misaligned Atomic | 0.1 |
Zfa |
Additional Floating-Point Instructions | 1.0 |
Zfbfmin |
Scalar BF16 Converts | 1.0 |
Zfh |
Half-Precision Floating-Point | 1.0 |
Zfhmin |
Minimal Half-Precision Floating-Point | 1.0 |
Zfinx |
Single-Precision Floating-Point in Integer Registers | 1.0 |
Zdinx |
Double-Precision Floating-Point in Integer Registers | 1.0 |
Zhinx |
Half-Precision Floating-Point in Integer Registers | 1.0 |
Zhinxmin |
Minimal Half-Precision Floating-Point in Integer Registers | 1.0 |
Zca |
C Extension Instructions without Floating-Point Load/Stores | 1.0 |
Zcb |
Simple Code Size Saving Instructions | 1.0 |
Zcd |
Compressed Double-Precision Floating-Point Load/Stores Instructions | 1.0 |
Zce |
Compressed Instructions for Microcontrollers | 1.0 |
Zcf |
Compressed Single-Precision Floating-Point Load/Stores Instructions | 1.0 |
Zcmop |
Compressed May-Be-Operations Extension | 0.2 |
Zcmp |
Push/Pop and Double Move Instructions | 1.0 |
Zcmt |
Table Jump Instructions | 1.0 |
Zba |
Address Generation Instructions | 1.0 |
Zbb |
Basic Bit-Manipulation | 1.0 |
Zbc |
Carry-Less Multiplication | 1.0 |
Zbkb |
Bit-Manipulation Instructions for Cryptography | 1.0 |
Zbkc |
Carry-Less Multiplication Instructions for Cryptography | 1.0 |
Zbkx |
Crossbar Permutation Instructions for Cryptography | 1.0 |
Zbs |
Single-Bit Instructions | 1.0 |
Zk |
Standard Scalar Cryptography Extension | 1.0 |
Zkn |
NIST Algorithm Suite Extension | 1.0 |
Zknd |
AES Decryption Instructions | 1.0 |
Zkne |
AES Encryption Instructions | 1.0 |
Zknh |
SHA2 Hash Function Instructions | 1.0 |
Zkr |
Entropy Source Extension | 1.0 |
Zks |
ShangMi Algorithm Suite Extension | 1.0 |
Zksed |
SM4 Block Cipher Instructions | 1.0 |
Zksh |
SM3 Hash Function Instructions | 1.0 |
Zkt |
Data Independent Execution Latency Extension | 1.0 |
Zvamo |
Vector Atomic Instructions | |
Zvbb |
Vector Bit-Manipulation Instructions for Cryptography | 1.0 |
Zvbc |
Vector Carry-Less Multiplication | 1.0 |
Zve32f |
Vector Extension for Embedded 32-bit Single-Precision Floating-Point | 1.1 |
Zve32x |
Vector Extension for Embedded 32-bit Integer | 1.1 |
Zve64d |
Vector Extension for Embedded 64-bit Double-Precision Floating-Point | 1.1 |
Zve64f |
Vector Extension for Embedded 64-bit Single-Precision Floating-Point | 1.1 |
Zve64x |
Vector Extension for Embedded 64-bit Integer | 1.1 |
Zvediv |
Vector Divided Element Extension | |
Zvfbfmin |
Vector BF16 Converts | 1.0 |
Zvfbfwma |
Vector BF16 Widening Mul-Add | 1.0 |
Zvfh |
Vector Extension for Half-Precision Floating-Point | 1.1 |
Zvfhmin |
Vector Extension for Minimal Half-Precision Floating-Point | 1.1 |
Zvkb |
Vector Cryptography Bit-manipulation | 1.0 |
Zvkg |
Vector GCM/GMAC | 1.0 |
Zvkn |
Vector NIST Algorithm Suite Extension | 1.0 |
Zvknc |
Vector NIST Algorithm Suite with Carryless Multiply | 1.0 |
Zvkned |
Vector AES Block Cipher | 1.0 |
Zvkng |
Vector NIST Algorithm Suite with GCM | 1.0 |
Zvknha |
Vector SHA-2 Secure Hash for Embedded 32-bit | 1.0 |
Zvknhb |
Vector SHA-2 Secure Hash | 1.0 |
Zvks |
Vector ShangMi Algorithm Suite Extension | 1.0 |
Zvksc |
Vector ShangMi Algorithm Suite with Carryless Multiplication | 1.0 |
Zvksed |
Vector SM4 Block Cipher | 1.0 |
Zvksg |
Vector ShangMi Algorithm Suite with GCM | 1.0 |
Zvksh |
Vector SM3 Secure Hash | 1.0 |
Zvkt |
Vector Data Independent Execution Latency Extension | 1.0 |
Zvl32b |
Minimum 32-bit Vector Length | 1.1 |
Zvl64b |
Minimum 64-bit Vector Length | 1.1 |
Zvl128b |
Minimum 128-bit Vector Length | 1.1 |
Zvl256b |
Minimum 256-bit Vector Length | 1.1 |
Zvl512b |
Minimum 512-bit Vector Length | 1.1 |
Zvl1024b |
Minimum 1024-bit Vector Length | 1.1 |
Ztso |
Total Store Ordering | 1.0 |
- Additional Privileged Extensions
Extension | Full Name | Supported Version |
---|---|---|
Ss |
Supervisor-Level Extension | 1.12 |
Ssaia |
Supervisor-Level Advanced Interrupt Architecture Extension | 1.0 |
Ssccptr |
Main Memory Supports Page Table Reads | |
Sscofpmf |
Count Overflow and Mode-Based Filtering Extension | 1.0 |
Sscounterenw |
Support Writeable Enables for Any Supported Counter | |
Sscsrind |
Supervisor-Level Indirect CSR Access | 1.0 |
Sspmp |
Supervisor Memory Protection Extension | 0.8 |
Ssqosid |
Quality-of-Service Identifiers | 1.0 |
Ssstateen |
Supervisor-Level State Enable Extension | 1.0 |
Ssstrict |
No Non-Conforming Extensions Are Present | 1.0 |
Sstc |
Supervisor-Level Timer Comparison Extension | 1.0 |
Sstvala |
Stval Provides All Needed Values | |
Sstvecd |
Stvec Supports Direct Mode | |
Ssu64xl |
UXLEN=64 Must Be Supported | |
Sv32 |
Page-Based 32-bit Virtual-Memory Systems | |
Sv39 |
Page-Based 39-bit Virtual-Memory Systems | |
Sv48 |
Page-Based 48-bit Virtual-Memory Systems | |
Sv57 |
Page-Based 57-bit Virtual-Memory Systems | |
Svade |
Raise Exceptions on Improper A/D Bits | |
Svadu |
Hardware Updating of PTE A/D Bits | 1.0 |
Svbare |
Bare Mode Virtual-Memory Translation Supported | |
Svinval |
Fine-Grained Address-Translation Cache Invalidation | 1.0 |
Svnapot |
NAPOT Translation Contiguity | 1.0 |
Svpbmt |
Page-Based Memory Types | 1.0 |
Shcounterenw |
Support Writeable Enables for Any Supported Counter | |
Shtvala |
Htval Provides All Needed Values | |
Shvsatpa |
Vsatp Supports All Modes Supported by Satp | |
Shvstvala |
Vstval Provides All Needed Values | |
Shvstvecd |
Vstvec Supports Direct Mode | |
Shgatpa |
SvNNx4 Mode Supported for All Modes Supported by Satp, As Well As Bare | |
Sm |
Machine-Level Extension | 1.12 |
Smaia |
Machine-Level Advanced Interrupt Architecture Extension | 1.0 |
Smcntrpmf |
Cycle and Instret Privilege Mode Filtering | 1.0 |
Smcsrind |
Machine-Level Indirect CSR Access | 1.0 |
Smepmp |
PMP Enhancements for Memory Access and Execution Prevention on Machine Mode | 1.0 |
Smstateen |
Machine-Level State Enable Extension | 1.0 |
Sdext |
External Debug Extension | 1.0 |
Sdtrig |
Debug Trigger Extension | 1.0 |
Conflict Extensions
Extension A | Extension B |
---|---|
RV32E /RV64E |
H /G |
F /D /Q /V /Zfa /Zfh /Zcd /Zcf /Zfhmin /Zve32f /Zve64d /Zve64f /Zvfh /Zvfhmin |
Zfinx /Zdinx /Zhinx /Zhinxmin |
Zve32x /Zve32f |
Zvknhb |
Zcd |
RV128I /Zce /Zcmp /Zcmt |
C |
Zce /Zcmp /Zcmt |
Zcf |
RV64I /RV64E /RV128I |
Rules Should Be Followed
-
Ensure that you specify extensions in the sequence listed in Currently Supported Extensions. Always choose one of the base integer ISAs or profiles, and place the base ISA name at the beginning of the option. An exception is the general-purpose extension
G
, which can replace the RV32I and RV64I base ISAs. Do not specify more than one base ISA and profile, as this will result in a parser error. Extensions should be added in the following order: unprivileged, privileged, additional unprivileged, and additional privileged. Extensions within the same category should adhere to the order in the corresponding table. -
If the supported version of an extension is unspecified in the table, DO NOT including the version in the sequence.
-
Insert an underscore between two consecutive multi-letter extensions, i.e. additional extensions, for clarity.
-
Some extensions and privileged extensions cannot coexist; refer to Conflict Extensions. Avoid including conflicting extensions in the option.
Following table contains some examples of valid and invalid ISA extension combinations:
Sequence | Valid | Reason | Unresolved String |
---|---|---|---|
RV64IMCZicsr_Zifencei |
Yes | ||
RV32IAM |
No | Incorrect order | M |
RV32GMAF |
No | Incorrect order | MAF |
Zicsr |
No | No base ISA | Zicsr |
RV32IRV64IMAFC |
No | More than one base ISAs | RV64IMAFC |
RV64GZve64dZvl256b |
No | No underscore between two consecutive multi-letter extensions | Zve64dZvl256b |
RV32EMG |
No | Conflict extensions (RV32E & G ) |
G |
RV64GCVZvamo1p0 |
No | Specify version number on an unversioned extension | Zvamo1p0 |
Add Custom Extensions with Instructions and CSRs
Vimscript:
let g:riscv_asm_custom_isa = [{'name': 'Xcustom'}]
Neovim Lua:
vim.g.riscv_asm_custom_isa = {{name = 'Xcustom'}}
The custom ISA extensions are construct by a list of dictionaries, each representing a custom ISA extension. Each custom extension has following configuration options:
name
: String, mandatory. The custom extension name which starts with the letter 'X'.ver
: Float/Number, optional. The version of the custom extension.inst
: List of Strings, optional. Common instructions for allXLEN
.csr
: List of Strings, optional. Common CSRs for allXLEN
.inst32
: List of Strings, optional. Instructions forXLEN=32
.csr32
: List of Strings, optional. CSRs forXLEN=32
.inst64
: List of Strings, optional. Instructions forXLEN=64
.csr64
: List of Strings, optional. CSRs forXLEN=64
.inst128
: List of Strings, optional. Instructions forXLEN=128
.csr128
: List of Strings, optional. CSRs forXLEN=128
.
Highlight the Custom Extensions
Like the standard extensions, you can use g:riscv_asm_isa
to control enablement of the custom extensions, or use g:riscv_asm_all_enable
to highlight all of the standard extensions and the custom extensions. The syntax in g:riscv_asm_isa
should comply with the RISC-V ISA extension naming conventions:
-
The string is case insensitive.
-
The name of a custom extension should start with the letter 'X' followed by an alphabetical name and an optional version number.
-
The custom extensions must be listed after all standard extensions.
-
Like other multi-letter extensions, must be separated from other multi-letter extensions by an underscore.
-
If multiple custom extensions are listed, they should be ordered alphabetically.
If you did not specify the ver
of a custom extension defined in g:riscv_asm_custom_isa
, you should not list the custom extension with any version number. Following table listed some valid or invalid sequence with the two custom extensions in the examples.
Sequence | Valid | Reason | Unresolved String |
---|---|---|---|
RV64GCXargle1p0_Xbargle |
Yes | ||
RV64GCXargle1p0_Sv48 |
No | Incorrect order (standard extensions should be prior) | _Sv48 |
RV64GCXbargle_Xargle |
No | Incorrect order (violate the alphabetical order) | _Xargle |
RV64GCXargleXbargle |
No | No underscore between two consecutive multi-letter extensions | XargleXbargle |
RV64GC_Sv48_Xbargle1p0 |
No | Specify version number on an unversioned extension | _Xbargle1p0 |
Examples of Adding the Custom Extensions
Vim example for adding two custom extensions, Xangle
and Xbargle
:
let g:riscv_asm_custom_isa =
\ [
\ {
\ 'name': 'Xargle',
\ 'ver': 1.0,
\ 'inst': ['inst_1', 'inst_2'],
\ 'csr': ['csr_1', 'csr_2'],
\ 'inst32': ['inst32_1', 'inst32_2'],
\ 'csr32': ['csr32_1', 'csr32_2'],
\ 'inst64': ['inst64_1', 'inst64_2'],
\ 'csr64': ['csr64_1', 'csr64_2'],
\ },
\ {
\ 'name': 'Xbargle',
\ },
\ ]
Equivalent example of Neovim Lua:
vim.g.riscv_asm_custom_isa = {
{
name = 'Xargle',
ver = 1.0,
inst = {'inst_1', 'inst_2'},
csr = {'csr_1', 'csr_2'},
inst32 = {'inst32_1', 'inst32_2'},
csr32 = {'csr32_1', 'csr32_2'},
inst64 = {'inst64_1', 'inst64_2'},
csr64 = {'csr64_1', 'csr64_2'},
},
{
name = 'Xbargle',
},
}
Display Debug Information
Vimscript:
let g:riscv_asm_debug = 1
Neovim Lua:
vim.g.riscv_asm_debug = true
If the option is defined (set to any value), the plugin will display more information while it's parsing the value of g:riscv_asm_isa
or the custom extensions.
Other VIM plugins
Tagbar
Tagbar is a Vim plugin that provides an easy way to browse the tags of the current file and get an overview of its structure. Follow steps in Tagbar Wiki to apply it.
Contributing
Contributions are welcome! If you encounter issues or have suggestions for improvements, please open an issue or submit a pull request on the riscv-asm-vim.
License
This plugin is licensed under the MIT License.