1.說明
riscv支持指令集自定義擴(kuò)展,這大大增加了riscv的可玩性,同時(shí)對于一些實(shí)際應(yīng)用中,自己通過一條指令來實(shí)現(xiàn)特定的功能,效率非常高,當(dāng)然,前提是硬件平臺需要對該指令的支持。
本文主要利用qemu模擬硬件平臺,實(shí)現(xiàn)特定指令解析,同時(shí)寫裸機(jī)代碼來測試該指令的運(yùn)行情況。當(dāng)然,如果實(shí)現(xiàn)的很好,是需要修改riscv的gcc的,讓自己的擴(kuò)展指令加入。這里不做修改,后面會(huì)詳細(xì)描述細(xì)節(jié)。
自定義指令實(shí)現(xiàn)完成后,用qemu對功能進(jìn)行仿真,然后通過fgpa驗(yàn)證具體的行為,最后流片,一個(gè)完整的riscv,并支持自定義指令的芯片就可以完成了。
這里可以實(shí)現(xiàn)一個(gè)cube指令,并定義該指令的含義是將傳入的值進(jìn)行三次冪,得到最后的結(jié)果。
qemu模擬的硬件平臺是sifive_u。
2.riscv擴(kuò)展指令的添加
目的:
實(shí)現(xiàn)cube指令,傳入一個(gè)數(shù),比如2,那么該指令返回的結(jié)果是8,如果是3,則返回3^3=27。
riscv指令的類型:
對于riscv,其指令按照特定的類型分為一下幾種。

目前的實(shí)現(xiàn)只基于R-type。
其擴(kuò)展指令集的格式如下
.insn r opcode, func3, func7, rd, rs1, rs2
按照其語法規(guī)則opcode表示操作碼,目前是7位,對于非壓縮指令來說,最后兩位是1。所以自己可以定義一個(gè)操作碼,當(dāng)然有一些操作碼已經(jīng)使用了,具體可以查看下面的倉庫。
https://github.com/riscv/riscv-opcodes
也可以在riscv官網(wǎng)上
的第Chapter 24 RV32/64G Instryction Set Listings查看目前riscv定義的指令碼。
比如關(guān)于算數(shù)的指令集定義如下:

自己設(shè)計(jì)一條指令要在這些標(biāo)準(zhǔn)指令之外的,比如操作碼為0x7b。
內(nèi)聯(lián)匯編格式如下:
asm volatile(“.insn r 0x7b, 6, 6, %0, %1, x0” : “=r”(cube) : “r”(addr));
于是,按照語法解析如下:
* func7 rs2 rs1 func3 rd opcode
* 31---------25--------19------15------12----------------6----------0
* | 000110 | 00000 | ***** | 110 | ***** | 1111011 |
* |------------------------------------------------------|----------|
上圖中,*表示的是任意值,所以該指令在翻譯的時(shí)候,實(shí)際上就是取出rs1表示的是寄存器地址,然后返回的是rd,也是寄存器地址。最后,從寄存器中存放的地址取數(shù)據(jù)則得到相應(yīng)的值。
3.裸機(jī)代碼編譯
下面一段非常簡單的針對sifive_u的裸機(jī)代碼,并在進(jìn)入main函數(shù)后,直接調(diào)用custom_cube計(jì)算得到結(jié)果。
#include 《stdio.h》
static int custom_cube(int addr)
{
int cube;
asm volatile (
“.insn r 0x7b, 6, 6, %0, %1, x0”
:“=r”(cube)
:“r”(addr)
);
return cube;
}
void main()
{
int a = 3;
int ret = 0;
ret = custom_cube((int)&a);
if(ret == a*a*a)
{
putchar(‘o’);
putchar(‘k’);
}
else
{
putchar(‘e’);
putchar(‘r’);
putchar(‘r’);
}
while(1);
}
程序非常簡單,就是判斷custom_cube得到計(jì)算結(jié)果是否與a*a*a的值相等。
代碼可以在下面的地址中找到
https://github.com/bigmagic123/riscv-hello-c
下載sifive的交叉編譯工具鏈即可,不需要自己編譯工具鏈,添加到系統(tǒng)環(huán)境變量,即可編譯。
通過反匯編查看
riscv64-unknown-elf-objdump -D build/bin/rv64imac/qemu-sifive_u/hello 》 1.txt
可以看到如下的信息:

可以看到gcc并不認(rèn)識這條指令,沒法翻譯成偽代碼,所以直接變成機(jī)器碼了。
手動(dòng)分析一下這個(gè)機(jī)器碼
* func7 rs2 rs1 func3 rd opcode
* 31---------25--------19------15------12----------------6----------0
* | 0000110 | 00000 | 01111 | 110 | 01111 | 1111011 |
* |------------------------------------------------------|----------|
通過上述分析,主要關(guān)注傳遞的參數(shù)rs1與rd。其值都是01111,因?yàn)榧拇嫫饕还彩?2位,所以用五位來表示,此時(shí)使用了x15寄存器傳遞參數(shù)同時(shí)作為返回值。
4.qemu編譯和指令的擴(kuò)展
本機(jī)測試環(huán)境是Ubuntu20.04,首先需要從官方網(wǎng)站上下載最新的代碼。
執(zhí)行下面的命令,安裝編譯環(huán)境。
sudo apt-get install -y git build-essential pkg-config zlib1g-dev libglib2.0-0 libglib2.0-dev libsdl1.2-dev libpixman-1-dev libfdt-dev autoconf automake libtool librbd-dev libaio-dev flex bison make
sudo apt-get install ninja-build
并進(jìn)入qemu目錄并創(chuàng)建build目錄,進(jìn)入build,輸入下面語句開始編譯。
。./configure --prefix=your_path/linux_qemu --target-list=riscv32-softmmu,riscv64-softmmu && make -j8 && make install
其中your_path/linux_qemu是自己存在的目錄。編譯完成后,qemu在該目錄下。
4.1 添加擴(kuò)展指令的decodetree
由于riscv指令格式具有一定的規(guī)律,所以有人根據(jù)語法規(guī)則寫了一個(gè)通用的python腳本來生產(chǎn)對應(yīng)指令解析函數(shù),這也是非常值得學(xué)習(xí)。qemu是通過指令集解析的,目前只需在decodetree中增加一條cube指令的實(shí)現(xiàn)即可。
在target/riscv/insn32.decode中。
只需要按照規(guī)定的格式排版即可

定義其格式

4.2 添加擴(kuò)展函數(shù)
在擴(kuò)展函數(shù)實(shí)現(xiàn)上可以在target/riscv/insn_trans/trans_rvi.c.inc中添加
static bool trans_cube(DisasContext *ctx, arg_cube *a)
{
gen_helper_cube(cpu_gpr[a-》rd], cpu_gpr[a-》rs1]);
return true;
}
當(dāng)指令集解析時(shí),匹配上操作碼后,可以執(zhí)行該函數(shù)。
另外也需要在target/riscv/helper.h函數(shù)中添加函數(shù)定義
DEF_HELPER_1(cube, tl, tl)
其中第一個(gè)參數(shù)為名稱,第二個(gè)是返回值,第三個(gè)是參數(shù)傳遞值。
4.3 解析函數(shù)實(shí)現(xiàn)
可以在target/riscv/op_helper.c中添加具體指令的實(shí)現(xiàn)。
target_ulong helper_cube(target_ulong rs1)
{
target_ulong val;
cpu_physical_memory_rw(rs1, &val, 4, 0);
return val*val*val;
}
由于該指令是實(shí)現(xiàn)立方乘法,所以返回乘法值即可。
5.功能測試與驗(yàn)證
qemu重新編譯后,執(zhí)行第二章節(jié)的代碼。

當(dāng)指令執(zhí)行正確會(huì)輸出ok。
qemu-system-riscv64 -nographic -machine sifive_u -bios none -kernel build/bin/rv64imac/qemu-sifive_u/hello
實(shí)際執(zhí)行效果如下:

此時(shí),可以正常的執(zhí)行成功。
編輯:lyn
-
CUBE
+關(guān)注
關(guān)注
0文章
11瀏覽量
9983 -
RISC-V
+關(guān)注
關(guān)注
48文章
2887瀏覽量
52936 -
qemu
+關(guān)注
關(guān)注
0文章
57瀏覽量
5939
原文標(biāo)題:riscv實(shí)現(xiàn)自定義指令并用qemu運(yùn)行
文章出處:【微信號:Embeded_IoT,微信公眾號:嵌入式IoT】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
無圖形界面模式下自定義檢查工具的應(yīng)用
易靈思FPGA RISC-V自定義指令的使用方法
riscv實(shí)現(xiàn)自定義指令并用qemu運(yùn)行
關(guān)于協(xié)處理器自定義指令的實(shí)現(xiàn)
軟硬件協(xié)同技術(shù)分享 - 任務(wù)劃分 + 自定義指令集
采用匯編指示符來使用自定義指令
NucleiStudio如何生成.verilog文件和.dasm文件,以及對.dasm文件中自定義指令反匯編結(jié)果分析
e203自定義指令硬件模塊設(shè)計(jì)不工作是怎么回事?
nuclei studio e203自定義指令不識別怎么解決?
零代碼實(shí)現(xiàn)茶吧機(jī)自定義語音控制定制
KiCad 中的自定義規(guī)則(KiCon 演講)
riscv如何實(shí)現(xiàn)自定義指令并用qemu運(yùn)行詳解
評論