0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

riscv如何实现自定义指令并用qemu运行详解

嵌入式IoT 来源:嵌入式IoT 作者:嵌入式IoT 2021-05-02 10:24 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

1.说明

riscv支持指令集自定义扩展,这大大增加了riscv的可玩性,同时对于一些实际应用中,自己通过一条指令来实现特定的功能,效率非常高,当然,前提是硬件平台需要对该指令的支持。

本文主要利用qemu模拟硬件平台,实现特定指令解析,同时写裸机代码来测试该指令的运行情况。当然,如果实现的很好,是需要修改riscv的gcc的,让自己的扩展指令加入。这里不做修改,后面会详细描述细节。

自定义指令实现完成后,用qemu对功能进行仿真,然后通过fgpa验证具体的行为,最后流片,一个完整的riscv,并支持自定义指令的芯片就可以完成了。

这里可以实现一个cube指令,并定义该指令的含义是将传入的值进行三次幂,得到最后的结果。

qemu模拟的硬件平台是sifive_u。

2.riscv扩展指令的添加

目的

实现cube指令,传入一个数,比如2,那么该指令返回的结果是8,如果是3,则返回3^3=27。

riscv指令的类型:

对于riscv,其指令按照特定的类型分为一下几种。

781de90c-9fec-11eb-8b86-12bb97331649.png

目前的实现只基于R-type。

其扩展指令集的格式如下

.insn r opcode, func3, func7, rd, rs1, rs2

按照其语法规则opcode表示操作码,目前是7位,对于非压缩指令来说,最后两位是1。所以自己可以定义一个操作码,当然有一些操作码已经使用了,具体可以查看下面的仓库。

https://github.com/riscv/riscv-opcodes

也可以在riscv官网上

的第Chapter 24 RV32/64G Instryction Set Listings查看目前riscv定义的指令码。

比如关于算数的指令集定义如下:

785b3e24-9fec-11eb-8b86-12bb97331649.png

自己设计一条指令要在这些标准指令之外的,比如操作码为0x7b。

内联汇编格式如下:

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 |

* |------------------------------------------------------|----------|

上图中,*表示的是任意值,所以该指令在翻译的时候,实际上就是取出rs1表示的是寄存器地址,然后返回的是rd,也是寄存器地址。最后,从寄存器中存放的地址取数据则得到相应的值。

3.裸机代码编译

下面一段非常简单的针对sifive_u的裸机代码,并在进入main函数后,直接调用custom_cube计算得到结果。

#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得到计算结果是否与a*a*a的值相等。

代码可以在下面的地址中找到

https://github.com/bigmagic123/riscv-hello-c

下载sifive的交叉编译工具链即可,不需要自己编译工具链,添加到系统环境变量,即可编译。

通过反汇编查看

riscv64-unknown-elf-objdump -D build/bin/rv64imac/qemu-sifive_u/hello 》 1.txt

可以看到如下的信息:

7868b46e-9fec-11eb-8b86-12bb97331649.png

可以看到gcc并不认识这条指令,没法翻译成伪代码,所以直接变成机器码了。

手动分析一下这个机器码

* func7 rs2 rs1 func3 rd opcode

* 31---------25--------19------15------12----------------6----------0

* | 0000110 | 00000 | 01111 | 110 | 01111 | 1111011 |

* |------------------------------------------------------|----------|

通过上述分析,主要关注传递的参数rs1与rd。其值都是01111,因为寄存器一共是32位,所以用五位来表示,此时使用了x15寄存器传递参数同时作为返回值。

4.qemu编译和指令的扩展

本机测试环境是Ubuntu20.04,首先需要从官方网站上下载最新的代码。

执行下面的命令,安装编译环境。

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

并进入qemu目录并创建build目录,进入build,输入下面语句开始编译。

。./configure --prefix=your_path/linux_qemu --target-list=riscv32-softmmu,riscv64-softmmu && make -j8 && make install

其中your_path/linux_qemu是自己存在的目录。编译完成后,qemu在该目录下。

4.1 添加扩展指令的decodetree

由于riscv指令格式具有一定的规律,所以有人根据语法规则写了一个通用的python脚本来生产对应指令解析函数,这也是非常值得学习。qemu是通过指令集解析的,目前只需在decodetree中增加一条cube指令的实现即可。

在target/riscv/insn32.decode中。

只需要按照规定的格式排版即可

78a4844e-9fec-11eb-8b86-12bb97331649.png

定义其格式

78e483d2-9fec-11eb-8b86-12bb97331649.png

4.2 添加扩展函数

在扩展函数实现上可以在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;

}

当指令集解析时,匹配上操作码后,可以执行该函数。

另外也需要在target/riscv/helper.h函数中添加函数定义

DEF_HELPER_1(cube, tl, tl)

其中第一个参数为名称,第二个是返回值,第三个是参数传递值。

4.3 解析函数实现

可以在target/riscv/op_helper.c中添加具体指令的实现。

target_ulong helper_cube(target_ulong rs1)

{

target_ulong val;

cpu_physical_memory_rw(rs1, &val, 4, 0);

return val*val*val;

}

由于该指令是实现立方乘法,所以返回乘法值即可。

5.功能测试与验证

qemu重新编译后,执行第二章节的代码。

78f5eece-9fec-11eb-8b86-12bb97331649.png

当指令执行正确会输出ok。

qemu-system-riscv64 -nographic -machine sifive_u -bios none -kernel build/bin/rv64imac/qemu-sifive_u/hello

实际执行效果如下:

791b9cdc-9fec-11eb-8b86-12bb97331649.png

此时,可以正常的执行成功。
编辑:lyn

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • CUBE
    +关注

    关注

    0

    文章

    11

    浏览量

    9906
  • RISC-V
    +关注

    关注

    48

    文章

    2797

    浏览量

    51929
  • qemu
    +关注

    关注

    0

    文章

    57

    浏览量

    5878

原文标题:riscv实现自定义指令并用qemu运行

文章出处:【微信号:Embeded_IoT,微信公众号:嵌入式IoT】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    无图形界面模式下自定义检查工具的应用

    此前文章已介绍 ANSA 中的自定义检查工具。本文将探讨该功能在无图形界面(No-GUI)模式下的应用,旨在满足标准化工作流程的需求,适用于需要高度自动化的前处理场景。通过集成自定义检查,用户可实现工作流程的高效自动化
    的头像 发表于 11-30 14:13 305次阅读
    无图形界面模式下<b class='flag-5'>自定义</b>检查工具的应用

    易灵思FPGA RISC-V自定义指令的使用方法

    功耗设备到高性能计算。易灵思提供完整的RISC-V 解决方案,致力于让开发者和硬件设计人员可以根据需要自定义和扩展指令集,且无需担心专有技术的限制。
    的头像 发表于 11-24 11:36 3753次阅读
    易灵思FPGA RISC-V<b class='flag-5'>自定义</b><b class='flag-5'>指令</b>的使用方法

    riscv实现自定义指令并用qemu运行

    利用qemu模拟硬件平台,实现特定指令解析,同时写裸机代码来测试该指令运行情况。当然,如果实现
    发表于 10-31 07:37

    关于协处理器自定义指令实现

    位置1,在上述链接的指令格式中也可见其对应关系: 后面一个6就是对应指令编号了,在31位到25位的地方,不同的指令编号即表示不同的指令,所以每种custom类型都可以
    发表于 10-31 06:36

    软硬件协同技术分享 - 任务划分 + 自定义指令

    利用定时器中断,率先判断该FIFO的值不为空,并且保证一次读取一个帧长(即10个周期)的数据,能够实现该FIFO内数据即写即读,数据写入不久便能取走的效果。 自定义指令集设计 E203
    发表于 10-28 08:03

    采用汇编指示符来使用自定义指令

    具体实现 1、采用.word .half .dword等汇编指示符直接插入自定义指令,这种方法需要自己指定寄存器。其中.word为插入一个字的数据即32位,.half为插入半字即16位
    发表于 10-28 06:02

    NucleiStudio如何生成.verilog文件和.dasm文件,以及对.dasm文件中自定义指令反汇编结果分析

    硬件设计需要.verilog文件来运行加NICE后的tb仿真、软件组需要.dasm来确定自定义指令有无被正确编译。今天,我们来分享一下NucleiStudio如何生成.verilog文件和.dasm
    发表于 10-24 06:33

    e203自定义指令硬件模块设计不工作是怎么回事?

    设计了自定义指令,用软件跑了verilog二进制文件,激励进去。但自定义的硬件模块不工作,都是零,是我指令没给进去吗?还是逻辑有问题?有遇到过类似问题的小伙伴吗?求解答
    发表于 10-20 06:50

    HarmonyOS应用自定义键盘解决方案

    自定义键盘是一种替换系统默认键盘的解决方案,可实现键盘个性化交互。允许用户结合业务需求与操作习惯,对按键布局进行可视化重构、设置多功能组合键位,使输入更加便捷和舒适。在安全防护层面,自定义键盘可以
    的头像 发表于 06-05 14:19 1594次阅读

    LabVIEW运动控制(三):EtherCAT运动控制器的高效加工指令自定义封装

    LabVIEW高效加工指令自定义封装
    的头像 发表于 04-08 13:49 3270次阅读
    LabVIEW运动控制(三):EtherCAT运动控制器的高效加工<b class='flag-5'>指令</b><b class='flag-5'>自定义</b>封装

    如何添加自定义单板

    在开发过程中,用户有时需要创建自定义板配置。本节将通过一个实例讲解用户如何创建属于自己的machine,下面以g2l-test.conf为例进行说明。
    的头像 发表于 03-12 14:43 1093次阅读

    e203在vivado硬件里自定义指令识别为非法指令怎么解决?

    e203自定义指令硬件模块设计,修改内核,综合没错误,软件也修改工具链通过并产生verilog文件,但在vivado硬件里自定义指令识别为非法指令
    发表于 03-07 07:34

    Altium Designer 15.0自定义元件设计

    电子发烧友网站提供《Altium Designer 15.0自定义元件设计.pdf》资料免费下载
    发表于 01-21 15:04 0次下载
    Altium Designer 15.0<b class='flag-5'>自定义</b>元件设计

    自定义Modbus通信块指令的搭建教程

    和欧姆龙E5EZ的智能温控仪表作为从站为例,为大家分享如何使用自定义的Modbus RTU的通信程序块,来实现快速高效的Modbus通信程序设计。同时在本文的结尾会为大家提供参考程序和打包好的Modbus RTU通信块的库指令
    的头像 发表于 01-20 10:37 1867次阅读
    <b class='flag-5'>自定义</b>Modbus通信块<b class='flag-5'>指令</b>的搭建教程

    think-cell:自定义think-cell(四)

    C.5 设置默认议程幻灯片布局 think-cell 议程可以在演示文稿中使用特定的自定义布局来定义议程、位置和议程幻灯片上的其他形状,例如标题或图片。通过将此自定义布局添加到模板,您可以为整个组织
    的头像 发表于 01-13 10:37 885次阅读
    think-cell:<b class='flag-5'>自定义</b>think-cell(四)