CoCalc provides the best real-time collaborative environment for Jupyter Notebooks, LaTeX documents, and SageMath, scalable from individual users to large groups and classes!
CoCalc provides the best real-time collaborative environment for Jupyter Notebooks, LaTeX documents, and SageMath, scalable from individual users to large groups and classes!
Path: blob/master/ext/riscv-disas.cpp
Views: 1401
/*1* RISC-V Disassembler2*3* Copyright (c) 2016-2017 Michael Clark <[email protected]>4* Copyright (c) 2017-2018 SiFive, Inc.5*6* Permission is hereby granted, free of charge, to any person obtaining a copy7* of this software and associated documentation files (the "Software"), to deal8* in the Software without restriction, including without limitation the rights9* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell10* copies of the Software, and to permit persons to whom the Software is11* furnished to do so, subject to the following conditions:12*13* The above copyright notice and this permission notice shall be included in14* all copies or substantial portions of the Software.15*16* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR17* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,18* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL19* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER20* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,21* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN22* THE SOFTWARE.23*/2425#define __STDC_FORMAT_MACROS 126#include <cstdlib>27#include <cstdio>28#include <cmath>29#include <cinttypes>30#include "riscv-disas.h"3132typedef struct {33const int op;34const rvc_constraint *constraints;35} rv_comp_data;3637enum {38rvcd_imm_nz = 0x1,39rvcd_imm_nz_hint = 0x240};4142typedef struct {43const char * const name;44const rv_codec codec;45const char * const format;46const rv_comp_data *pseudo;47const short decomp_rv32;48const short decomp_rv64;49const short decomp_rv128;50const short decomp_data;51} rv_opcode_data;5253/* register names */5455static const char rv_ireg_name_sym[32][5] = {56"zero", "ra", "sp", "gp", "tp", "t0", "t1", "t2",57"s0", "s1", "a0", "a1", "a2", "a3", "a4", "a5",58"a6", "a7", "s2", "s3", "s4", "s5", "s6", "s7",59"s8", "s9", "s10", "s11", "t3", "t4", "t5", "t6",60};6162static const char rv_freg_name_sym[32][5] = {63"ft0", "ft1", "ft2", "ft3", "ft4", "ft5", "ft6", "ft7",64"fs0", "fs1", "fa0", "fa1", "fa2", "fa3", "fa4", "fa5",65"fa6", "fa7", "fs2", "fs3", "fs4", "fs5", "fs6", "fs7",66"fs8", "fs9", "fs10", "fs11", "ft8", "ft9", "ft10", "ft11",67};6869/* instruction formats */7071static const char rv_fmt_none[] = "O\t";72static const char rv_fmt_rs1[] = "O\t1";73static const char rv_fmt_offset[] = "O\to";74static const char rv_fmt_pred_succ[] = "O\tp,s";75static const char rv_fmt_rs1_rs2[] = "O\t1,2";76static const char rv_fmt_rd_imm[] = "O\t0,i";77static const char rv_fmt_rd_offset[] = "O\t0,o";78static const char rv_fmt_rd_rs1_rs2[] = "O\t0,1,2";79static const char rv_fmt_frd_rs1[] = "O\t3,1";80static const char rv_fmt_rd_frs1[] = "O\t0,4";81static const char rv_fmt_rd_frs1_frs2[] = "O\t0,4,5";82static const char rv_fmt_frd_frs1[] = "O\t3,4";83static const char rv_fmt_frd_frs1_frs2[] = "O\t3,4,5";84static const char rv_fmt_rm_frd_frs1[] = "O\tr,3,4";85static const char rv_fmt_rm_frd_rs1[] = "O\tr,3,1";86static const char rv_fmt_rm_rd_frs1[] = "O\tr,0,4";87static const char rv_fmt_rm_frd_frs1_frs2[] = "O\tr,3,4,5";88static const char rv_fmt_rm_frd_frs1_frs2_frs3[] = "O\tr,3,4,5,6";89static const char rv_fmt_rd_rs1_imm[] = "O\t0,1,i";90static const char rv_fmt_rd_rs1_offset[] = "O\t0,1,i";91static const char rv_fmt_rd_offset_rs1[] = "O\t0,i(1)";92static const char rv_fmt_frd_offset_rs1[] = "O\t3,i(1)";93static const char rv_fmt_rd_csr_rs1[] = "O\t0,c,1";94static const char rv_fmt_rd_csr_zimm[] = "O\t0,c,7";95static const char rv_fmt_rs2_offset_rs1[] = "O\t2,i(1)";96static const char rv_fmt_frs2_offset_rs1[] = "O\t5,i(1)";97static const char rv_fmt_rs1_rs2_offset[] = "O\t1,2,o";98static const char rv_fmt_rs2_rs1_offset[] = "O\t2,1,o";99static const char rv_fmt_aqrl_rd_rs2_rs1[] = "OAR\t0,2,(1)";100static const char rv_fmt_aqrl_rd_rs1[] = "OAR\t0,(1)";101static const char rv_fmt_rd[] = "O\t0";102static const char rv_fmt_rd_zimm[] = "O\t0,7";103static const char rv_fmt_rd_rs1[] = "O\t0,1";104static const char rv_fmt_rd_rs2[] = "O\t0,2";105static const char rv_fmt_rs1_offset[] = "O\t1,o";106static const char rv_fmt_rs2_offset[] = "O\t2,o";107108/* pseudo-instruction constraints */109110static const rvc_constraint rvcc_last[] = { rvc_end };111static const rvc_constraint rvcc_imm_eq_zero[] = { rvc_imm_eq_zero, rvc_end };112static const rvc_constraint rvcc_imm_eq_n1[] = { rvc_imm_eq_n1, rvc_end };113static const rvc_constraint rvcc_imm_eq_p1[] = { rvc_imm_eq_p1, rvc_end };114static const rvc_constraint rvcc_rs1_eq_x0[] = { rvc_rs1_eq_x0, rvc_end };115static const rvc_constraint rvcc_rs2_eq_x0[] = { rvc_rs2_eq_x0, rvc_end };116static const rvc_constraint rvcc_rs2_eq_rs1[] = { rvc_rs2_eq_rs1, rvc_end };117static const rvc_constraint rvcc_jal_j[] = { rvc_rd_eq_x0, rvc_end };118static const rvc_constraint rvcc_jal_jal[] = { rvc_rd_eq_ra, rvc_end };119static const rvc_constraint rvcc_jalr_jr[] = { rvc_rd_eq_x0, rvc_imm_eq_zero, rvc_end };120static const rvc_constraint rvcc_jalr_jalr[] = { rvc_rd_eq_ra, rvc_imm_eq_zero, rvc_end };121static const rvc_constraint rvcc_jalr_ret[] = { rvc_rd_eq_x0, rvc_rs1_eq_ra, rvc_imm_eq_zero, rvc_end };122static const rvc_constraint rvcc_addi_nop[] = { rvc_rd_eq_x0, rvc_rs1_eq_x0, rvc_imm_eq_zero, rvc_end };123static const rvc_constraint rvcc_rdcycle[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc00, rvc_end };124static const rvc_constraint rvcc_rdtime[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc01, rvc_end };125static const rvc_constraint rvcc_rdinstret[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc02, rvc_end };126static const rvc_constraint rvcc_rdcycleh[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc80, rvc_end };127static const rvc_constraint rvcc_rdtimeh[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc81, rvc_end };128static const rvc_constraint rvcc_rdinstreth[] = { rvc_rs1_eq_x0, rvc_csr_eq_0xc82, rvc_end };129static const rvc_constraint rvcc_frcsr[] = { rvc_rs1_eq_x0, rvc_csr_eq_0x003, rvc_end };130static const rvc_constraint rvcc_frrm[] = { rvc_rs1_eq_x0, rvc_csr_eq_0x002, rvc_end };131static const rvc_constraint rvcc_frflags[] = { rvc_rs1_eq_x0, rvc_csr_eq_0x001, rvc_end };132static const rvc_constraint rvcc_fscsr[] = { rvc_csr_eq_0x003, rvc_end };133static const rvc_constraint rvcc_fsrm[] = { rvc_csr_eq_0x002, rvc_end };134static const rvc_constraint rvcc_fsflags[] = { rvc_csr_eq_0x001, rvc_end };135static const rvc_constraint rvcc_fsrmi[] = { rvc_csr_eq_0x002, rvc_end };136static const rvc_constraint rvcc_fsflagsi[] = { rvc_csr_eq_0x001, rvc_end };137138/* pseudo-instruction metadata */139140static const rv_comp_data rvcp_jal[] = {141{ rv_op_j, rvcc_jal_j },142{ rv_op_jal, rvcc_jal_jal },143{ rv_op_illegal, NULL }144};145146static const rv_comp_data rvcp_jalr[] = {147{ rv_op_ret, rvcc_jalr_ret },148{ rv_op_jr, rvcc_jalr_jr },149{ rv_op_jalr, rvcc_jalr_jalr },150{ rv_op_illegal, NULL }151};152153static const rv_comp_data rvcp_beq[] = {154{ rv_op_beqz, rvcc_rs2_eq_x0 },155{ rv_op_illegal, NULL }156};157158static const rv_comp_data rvcp_bne[] = {159{ rv_op_bnez, rvcc_rs2_eq_x0 },160{ rv_op_illegal, NULL }161};162163static const rv_comp_data rvcp_blt[] = {164{ rv_op_bltz, rvcc_rs2_eq_x0 },165{ rv_op_bgtz, rvcc_rs1_eq_x0 },166{ rv_op_bgt, rvcc_last },167{ rv_op_illegal, NULL }168};169170static const rv_comp_data rvcp_bge[] = {171{ rv_op_bgez, rvcc_rs2_eq_x0 },172{ rv_op_blez, rvcc_rs1_eq_x0 },173{ rv_op_ble, rvcc_last },174{ rv_op_illegal, NULL }175};176177static const rv_comp_data rvcp_bltu[] = {178{ rv_op_bgtu, rvcc_last },179{ rv_op_illegal, NULL }180};181182static const rv_comp_data rvcp_bgeu[] = {183{ rv_op_bleu, rvcc_last },184{ rv_op_illegal, NULL }185};186187static const rv_comp_data rvcp_addi[] = {188{ rv_op_nop, rvcc_addi_nop },189{ rv_op_mv, rvcc_imm_eq_zero },190{ rv_op_illegal, NULL }191};192193static const rv_comp_data rvcp_sltiu[] = {194{ rv_op_seqz, rvcc_imm_eq_p1 },195{ rv_op_illegal, NULL }196};197198static const rv_comp_data rvcp_xori[] = {199{ rv_op_not, rvcc_imm_eq_n1 },200{ rv_op_illegal, NULL }201};202203static const rv_comp_data rvcp_sub[] = {204{ rv_op_neg, rvcc_rs1_eq_x0 },205{ rv_op_illegal, NULL }206};207208static const rv_comp_data rvcp_slt[] = {209{ rv_op_sltz, rvcc_rs2_eq_x0 },210{ rv_op_sgtz, rvcc_rs1_eq_x0 },211{ rv_op_illegal, NULL }212};213214static const rv_comp_data rvcp_sltu[] = {215{ rv_op_snez, rvcc_rs1_eq_x0 },216{ rv_op_illegal, NULL }217};218219static const rv_comp_data rvcp_addiw[] = {220{ rv_op_sext_w, rvcc_imm_eq_zero },221{ rv_op_illegal, NULL }222};223224static const rv_comp_data rvcp_subw[] = {225{ rv_op_negw, rvcc_rs1_eq_x0 },226{ rv_op_illegal, NULL }227};228229static const rv_comp_data rvcp_csrrw[] = {230{ rv_op_fscsr, rvcc_fscsr },231{ rv_op_fsrm, rvcc_fsrm },232{ rv_op_fsflags, rvcc_fsflags },233{ rv_op_illegal, NULL }234};235236static const rv_comp_data rvcp_csrrs[] = {237{ rv_op_rdcycle, rvcc_rdcycle },238{ rv_op_rdtime, rvcc_rdtime },239{ rv_op_rdinstret, rvcc_rdinstret },240{ rv_op_rdcycleh, rvcc_rdcycleh },241{ rv_op_rdtimeh, rvcc_rdtimeh },242{ rv_op_rdinstreth, rvcc_rdinstreth },243{ rv_op_frcsr, rvcc_frcsr },244{ rv_op_frrm, rvcc_frrm },245{ rv_op_frflags, rvcc_frflags },246{ rv_op_illegal, NULL }247};248249static const rv_comp_data rvcp_csrrwi[] = {250{ rv_op_fsrmi, rvcc_fsrmi },251{ rv_op_fsflagsi, rvcc_fsflagsi },252{ rv_op_illegal, NULL }253};254255static const rv_comp_data rvcp_fsgnj_h[] = {256{ rv_op_fmv_h, rvcc_rs2_eq_rs1 },257{ rv_op_illegal, NULL }258};259260static const rv_comp_data rvcp_fsgnjn_h[] = {261{ rv_op_fneg_h, rvcc_rs2_eq_rs1 },262{ rv_op_illegal, NULL }263};264265static const rv_comp_data rvcp_fsgnjx_h[] = {266{ rv_op_fabs_h, rvcc_rs2_eq_rs1 },267{ rv_op_illegal, NULL }268};269270static const rv_comp_data rvcp_fsgnj_s[] = {271{ rv_op_fmv_s, rvcc_rs2_eq_rs1 },272{ rv_op_illegal, NULL }273};274275static const rv_comp_data rvcp_fsgnjn_s[] = {276{ rv_op_fneg_s, rvcc_rs2_eq_rs1 },277{ rv_op_illegal, NULL }278};279280static const rv_comp_data rvcp_fsgnjx_s[] = {281{ rv_op_fabs_s, rvcc_rs2_eq_rs1 },282{ rv_op_illegal, NULL }283};284285static const rv_comp_data rvcp_fsgnj_d[] = {286{ rv_op_fmv_d, rvcc_rs2_eq_rs1 },287{ rv_op_illegal, NULL }288};289290static const rv_comp_data rvcp_fsgnjn_d[] = {291{ rv_op_fneg_d, rvcc_rs2_eq_rs1 },292{ rv_op_illegal, NULL }293};294295static const rv_comp_data rvcp_fsgnjx_d[] = {296{ rv_op_fabs_d, rvcc_rs2_eq_rs1 },297{ rv_op_illegal, NULL }298};299300static const rv_comp_data rvcp_fsgnj_q[] = {301{ rv_op_fmv_q, rvcc_rs2_eq_rs1 },302{ rv_op_illegal, NULL }303};304305static const rv_comp_data rvcp_fsgnjn_q[] = {306{ rv_op_fneg_q, rvcc_rs2_eq_rs1 },307{ rv_op_illegal, NULL }308};309310static const rv_comp_data rvcp_fsgnjx_q[] = {311{ rv_op_fabs_q, rvcc_rs2_eq_rs1 },312{ rv_op_illegal, NULL }313};314315/* instruction metadata */316317const rv_opcode_data opcode_data[] = {318{ "illegal", rv_codec_illegal, rv_fmt_none, NULL, 0, 0, 0 },319{ "lui", rv_codec_u, rv_fmt_rd_imm, NULL, 0, 0, 0 },320{ "auipc", rv_codec_u, rv_fmt_rd_offset, NULL, 0, 0, 0 },321{ "jal", rv_codec_uj, rv_fmt_rd_offset, rvcp_jal, 0, 0, 0 },322{ "jalr", rv_codec_i, rv_fmt_rd_rs1_offset, rvcp_jalr, 0, 0, 0 },323{ "beq", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_beq, 0, 0, 0 },324{ "bne", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_bne, 0, 0, 0 },325{ "blt", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_blt, 0, 0, 0 },326{ "bge", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_bge, 0, 0, 0 },327{ "bltu", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_bltu, 0, 0, 0 },328{ "bgeu", rv_codec_sb, rv_fmt_rs1_rs2_offset, rvcp_bgeu, 0, 0, 0 },329{ "lb", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },330{ "lh", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },331{ "lw", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },332{ "lbu", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },333{ "lhu", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },334{ "sb", rv_codec_s, rv_fmt_rs2_offset_rs1, NULL, 0, 0, 0 },335{ "sh", rv_codec_s, rv_fmt_rs2_offset_rs1, NULL, 0, 0, 0 },336{ "sw", rv_codec_s, rv_fmt_rs2_offset_rs1, NULL, 0, 0, 0 },337{ "addi", rv_codec_i, rv_fmt_rd_rs1_imm, rvcp_addi, 0, 0, 0 },338{ "slti", rv_codec_i, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },339{ "sltiu", rv_codec_i, rv_fmt_rd_rs1_imm, rvcp_sltiu, 0, 0, 0 },340{ "xori", rv_codec_i, rv_fmt_rd_rs1_imm, rvcp_xori, 0, 0, 0 },341{ "ori", rv_codec_i, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },342{ "andi", rv_codec_i, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },343{ "slli", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },344{ "srli", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },345{ "srai", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },346{ "add", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },347{ "sub", rv_codec_r, rv_fmt_rd_rs1_rs2, rvcp_sub, 0, 0, 0 },348{ "sll", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },349{ "slt", rv_codec_r, rv_fmt_rd_rs1_rs2, rvcp_slt, 0, 0, 0 },350{ "sltu", rv_codec_r, rv_fmt_rd_rs1_rs2, rvcp_sltu, 0, 0, 0 },351{ "xor", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },352{ "srl", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },353{ "sra", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },354{ "or", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },355{ "and", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },356{ "fence", rv_codec_r_f, rv_fmt_pred_succ, NULL, 0, 0, 0 },357{ "fence.i", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },358{ "lwu", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },359{ "ld", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },360{ "sd", rv_codec_s, rv_fmt_rs2_offset_rs1, NULL, 0, 0, 0 },361{ "addiw", rv_codec_i, rv_fmt_rd_rs1_imm, rvcp_addiw, 0, 0, 0 },362{ "slliw", rv_codec_i_sh5, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },363{ "srliw", rv_codec_i_sh5, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },364{ "sraiw", rv_codec_i_sh5, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },365{ "addw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },366{ "subw", rv_codec_r, rv_fmt_rd_rs1_rs2, rvcp_subw, 0, 0, 0 },367{ "sllw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },368{ "srlw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },369{ "sraw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },370{ "ldu", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },371{ "lq", rv_codec_i, rv_fmt_rd_offset_rs1, NULL, 0, 0, 0 },372{ "sq", rv_codec_s, rv_fmt_rs2_offset_rs1, NULL, 0, 0, 0 },373{ "addid", rv_codec_i, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },374{ "sllid", rv_codec_i_sh6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },375{ "srlid", rv_codec_i_sh6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },376{ "sraid", rv_codec_i_sh6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },377{ "addd", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },378{ "subd", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },379{ "slld", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },380{ "srld", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },381{ "srad", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },382{ "mul", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },383{ "mulh", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },384{ "mulhsu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },385{ "mulhu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },386{ "div", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },387{ "divu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },388{ "rem", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },389{ "remu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },390{ "mulw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },391{ "divw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },392{ "divuw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },393{ "remw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },394{ "remuw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },395{ "muld", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },396{ "divd", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },397{ "divud", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },398{ "remd", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },399{ "remud", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },400{ "lr.w", rv_codec_r_l, rv_fmt_aqrl_rd_rs1, NULL, 0, 0, 0 },401{ "sc.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },402{ "amoswap.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },403{ "amoadd.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },404{ "amoxor.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },405{ "amoor.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },406{ "amoand.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },407{ "amomin.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },408{ "amomax.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },409{ "amominu.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },410{ "amomaxu.w", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },411{ "lr.d", rv_codec_r_l, rv_fmt_aqrl_rd_rs1, NULL, 0, 0, 0 },412{ "sc.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },413{ "amoswap.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },414{ "amoadd.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },415{ "amoxor.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },416{ "amoor.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },417{ "amoand.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },418{ "amomin.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },419{ "amomax.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },420{ "amominu.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },421{ "amomaxu.d", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },422{ "lr.q", rv_codec_r_l, rv_fmt_aqrl_rd_rs1, NULL, 0, 0, 0 },423{ "sc.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },424{ "amoswap.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },425{ "amoadd.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },426{ "amoxor.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },427{ "amoor.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },428{ "amoand.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },429{ "amomin.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },430{ "amomax.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },431{ "amominu.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },432{ "amomaxu.q", rv_codec_r_a, rv_fmt_aqrl_rd_rs2_rs1, NULL, 0, 0, 0 },433{ "ecall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },434{ "ebreak", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },435{ "uret", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },436{ "sret", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },437{ "hret", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },438{ "mret", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },439{ "dret", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },440{ "sfence.vm", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },441{ "sfence.vma", rv_codec_r, rv_fmt_rs1_rs2, NULL, 0, 0, 0 },442{ "wfi", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },443{ "csrrw", rv_codec_i_csr, rv_fmt_rd_csr_rs1, rvcp_csrrw, 0, 0, 0 },444{ "csrrs", rv_codec_i_csr, rv_fmt_rd_csr_rs1, rvcp_csrrs, 0, 0, 0 },445{ "csrrc", rv_codec_i_csr, rv_fmt_rd_csr_rs1, NULL, 0, 0, 0 },446{ "csrrwi", rv_codec_i_csr, rv_fmt_rd_csr_zimm, rvcp_csrrwi, 0, 0, 0 },447{ "csrrsi", rv_codec_i_csr, rv_fmt_rd_csr_zimm, NULL, 0, 0, 0 },448{ "csrrci", rv_codec_i_csr, rv_fmt_rd_csr_zimm, NULL, 0, 0, 0 },449{ "flh", rv_codec_i, rv_fmt_frd_offset_rs1, NULL, 0, 0, 0 },450{ "fsh", rv_codec_s, rv_fmt_frs2_offset_rs1, NULL, 0, 0, 0 },451{ "fmadd.h", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },452{ "fmsub.h", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },453{ "fnmsub.h", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },454{ "fnmadd.h", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },455{ "fadd.h", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },456{ "fsub.h", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },457{ "fmul.h", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },458{ "fdiv.h", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },459{ "fsgnj.h", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnj_h, 0, 0, 0 },460{ "fsgnjn.h", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjn_h, 0, 0, 0 },461{ "fsgnjx.h", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjx_h, 0, 0, 0 },462{ "fmin.h", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },463{ "fmax.h", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },464{ "fsqrt.h", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },465{ "fle.h", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },466{ "flt.h", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },467{ "feq.h", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },468{ "fcvt.w.h", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },469{ "fcvt.wu.h", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },470{ "fcvt.h.w", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },471{ "fcvt.h.wu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },472{ "fclass.h", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },473{ "fcvt.l.h", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },474{ "fcvt.lu.h", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },475{ "fmv.x.h", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },476{ "fcvt.h.l", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },477{ "fcvt.h.lu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },478{ "fmv.h.x", rv_codec_r, rv_fmt_frd_rs1, NULL, 0, 0, 0 },479{ "fcvt.s.h", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },480{ "fcvt.h.s", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },481{ "fcvt.d.h", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },482{ "fcvt.h.d", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },483{ "fcvt.q.h", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },484{ "fcvt.h.q", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },485{ "fmv.h", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },486{ "fabs.h", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },487{ "fneg.h", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },488{ "flw", rv_codec_i, rv_fmt_frd_offset_rs1, NULL, 0, 0, 0 },489{ "fsw", rv_codec_s, rv_fmt_frs2_offset_rs1, NULL, 0, 0, 0 },490{ "fmadd.s", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },491{ "fmsub.s", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },492{ "fnmsub.s", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },493{ "fnmadd.s", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },494{ "fadd.s", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },495{ "fsub.s", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },496{ "fmul.s", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },497{ "fdiv.s", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },498{ "fsgnj.s", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnj_s, 0, 0, 0 },499{ "fsgnjn.s", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjn_s, 0, 0, 0 },500{ "fsgnjx.s", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjx_s, 0, 0, 0 },501{ "fmin.s", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },502{ "fmax.s", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },503{ "fsqrt.s", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },504{ "fle.s", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },505{ "flt.s", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },506{ "feq.s", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },507{ "fcvt.w.s", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },508{ "fcvt.wu.s", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },509{ "fcvt.s.w", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },510{ "fcvt.s.wu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },511{ "fmv.x.s", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },512{ "fclass.s", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },513{ "fmv.s.x", rv_codec_r, rv_fmt_frd_rs1, NULL, 0, 0, 0 },514{ "fcvt.l.s", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },515{ "fcvt.lu.s", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },516{ "fcvt.s.l", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },517{ "fcvt.s.lu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },518{ "fld", rv_codec_i, rv_fmt_frd_offset_rs1, NULL, 0, 0, 0 },519{ "fsd", rv_codec_s, rv_fmt_frs2_offset_rs1, NULL, 0, 0, 0 },520{ "fmadd.d", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },521{ "fmsub.d", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },522{ "fnmsub.d", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },523{ "fnmadd.d", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },524{ "fadd.d", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },525{ "fsub.d", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },526{ "fmul.d", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },527{ "fdiv.d", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },528{ "fsgnj.d", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnj_d, 0, 0, 0 },529{ "fsgnjn.d", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjn_d, 0, 0, 0 },530{ "fsgnjx.d", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjx_d, 0, 0, 0 },531{ "fmin.d", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },532{ "fmax.d", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },533{ "fcvt.s.d", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },534{ "fcvt.d.s", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },535{ "fsqrt.d", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },536{ "fle.d", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },537{ "flt.d", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },538{ "feq.d", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },539{ "fcvt.w.d", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },540{ "fcvt.wu.d", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },541{ "fcvt.d.w", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },542{ "fcvt.d.wu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },543{ "fclass.d", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },544{ "fcvt.l.d", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },545{ "fcvt.lu.d", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },546{ "fmv.x.d", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },547{ "fcvt.d.l", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },548{ "fcvt.d.lu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },549{ "fmv.d.x", rv_codec_r, rv_fmt_frd_rs1, NULL, 0, 0, 0 },550{ "flq", rv_codec_i, rv_fmt_frd_offset_rs1, NULL, 0, 0, 0 },551{ "fsq", rv_codec_s, rv_fmt_frs2_offset_rs1, NULL, 0, 0, 0 },552{ "fmadd.q", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },553{ "fmsub.q", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },554{ "fnmsub.q", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },555{ "fnmadd.q", rv_codec_r4_m, rv_fmt_rm_frd_frs1_frs2_frs3, NULL, 0, 0, 0 },556{ "fadd.q", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },557{ "fsub.q", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },558{ "fmul.q", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },559{ "fdiv.q", rv_codec_r_m, rv_fmt_rm_frd_frs1_frs2, NULL, 0, 0, 0 },560{ "fsgnj.q", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnj_q, 0, 0, 0 },561{ "fsgnjn.q", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjn_q, 0, 0, 0 },562{ "fsgnjx.q", rv_codec_r, rv_fmt_frd_frs1_frs2, rvcp_fsgnjx_q, 0, 0, 0 },563{ "fmin.q", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },564{ "fmax.q", rv_codec_r, rv_fmt_frd_frs1_frs2, NULL, 0, 0, 0 },565{ "fcvt.s.q", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },566{ "fcvt.q.s", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },567{ "fcvt.d.q", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },568{ "fcvt.q.d", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },569{ "fsqrt.q", rv_codec_r_m, rv_fmt_rm_frd_frs1, NULL, 0, 0, 0 },570{ "fle.q", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },571{ "flt.q", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },572{ "feq.q", rv_codec_r, rv_fmt_rd_frs1_frs2, NULL, 0, 0, 0 },573{ "fcvt.w.q", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },574{ "fcvt.wu.q", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },575{ "fcvt.q.w", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },576{ "fcvt.q.wu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },577{ "fclass.q", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },578{ "fcvt.l.q", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },579{ "fcvt.lu.q", rv_codec_r_m, rv_fmt_rm_rd_frs1, NULL, 0, 0, 0 },580{ "fcvt.q.l", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },581{ "fcvt.q.lu", rv_codec_r_m, rv_fmt_rm_frd_rs1, NULL, 0, 0, 0 },582{ "fmv.x.q", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },583{ "fmv.q.x", rv_codec_r, rv_fmt_frd_rs1, NULL, 0, 0, 0 },584{ "c.addi4spn", rv_codec_ciw_4spn, rv_fmt_rd_rs1_imm, NULL, rv_op_addi, rv_op_addi, rv_op_addi, rvcd_imm_nz },585{ "c.fld", rv_codec_cl_ld, rv_fmt_frd_offset_rs1, NULL, rv_op_fld, rv_op_fld, 0 },586{ "c.lw", rv_codec_cl_lw, rv_fmt_rd_offset_rs1, NULL, rv_op_lw, rv_op_lw, rv_op_lw },587{ "c.flw", rv_codec_cl_lw, rv_fmt_frd_offset_rs1, NULL, rv_op_flw, 0, 0 },588{ "c.fsd", rv_codec_cs_sd, rv_fmt_frs2_offset_rs1, NULL, rv_op_fsd, rv_op_fsd, 0 },589{ "c.sw", rv_codec_cs_sw, rv_fmt_rs2_offset_rs1, NULL, rv_op_sw, rv_op_sw, rv_op_sw },590{ "c.fsw", rv_codec_cs_sw, rv_fmt_frs2_offset_rs1, NULL, rv_op_fsw, 0, 0 },591{ "c.nop", rv_codec_ci_none, rv_fmt_none, NULL, rv_op_addi, rv_op_addi, rv_op_addi },592{ "c.addi", rv_codec_ci, rv_fmt_rd_rs1_imm, NULL, rv_op_addi, rv_op_addi, rv_op_addi, rvcd_imm_nz_hint },593{ "c.jal", rv_codec_cj_jal, rv_fmt_rd_offset, NULL, rv_op_jal, 0, 0 },594{ "c.li", rv_codec_ci_li, rv_fmt_rd_rs1_imm, NULL, rv_op_addi, rv_op_addi, rv_op_addi },595{ "c.addi16sp", rv_codec_ci_16sp, rv_fmt_rd_rs1_imm, NULL, rv_op_addi, rv_op_addi, rv_op_addi, rvcd_imm_nz },596{ "c.lui", rv_codec_ci_lui, rv_fmt_rd_imm, NULL, rv_op_lui, rv_op_lui, rv_op_lui, rvcd_imm_nz },597{ "c.srli", rv_codec_cb_sh6, rv_fmt_rd_rs1_imm, NULL, rv_op_srli, rv_op_srli, rv_op_srli, rvcd_imm_nz },598{ "c.srai", rv_codec_cb_sh6, rv_fmt_rd_rs1_imm, NULL, rv_op_srai, rv_op_srai, rv_op_srai, rvcd_imm_nz },599{ "c.andi", rv_codec_cb_imm, rv_fmt_rd_rs1_imm, NULL, rv_op_andi, rv_op_andi, rv_op_andi, rvcd_imm_nz },600{ "c.sub", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_sub, rv_op_sub, rv_op_sub },601{ "c.xor", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_xor, rv_op_xor, rv_op_xor },602{ "c.or", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_or, rv_op_or, rv_op_or },603{ "c.and", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_and, rv_op_and, rv_op_and },604{ "c.subw", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_subw, rv_op_subw, rv_op_subw },605{ "c.addw", rv_codec_cs, rv_fmt_rd_rs1_rs2, NULL, rv_op_addw, rv_op_addw, rv_op_addw },606{ "c.j", rv_codec_cj, rv_fmt_rd_offset, NULL, rv_op_jal, rv_op_jal, rv_op_jal },607{ "c.beqz", rv_codec_cb, rv_fmt_rs1_rs2_offset, NULL, rv_op_beq, rv_op_beq, rv_op_beq },608{ "c.bnez", rv_codec_cb, rv_fmt_rs1_rs2_offset, NULL, rv_op_bne, rv_op_bne, rv_op_bne },609{ "c.slli", rv_codec_ci_sh6, rv_fmt_rd_rs1_imm, NULL, rv_op_slli, rv_op_slli, rv_op_slli, rvcd_imm_nz },610{ "c.fldsp", rv_codec_ci_ldsp, rv_fmt_frd_offset_rs1, NULL, rv_op_fld, rv_op_fld, rv_op_fld },611{ "c.lwsp", rv_codec_ci_lwsp, rv_fmt_rd_offset_rs1, NULL, rv_op_lw, rv_op_lw, rv_op_lw },612{ "c.flwsp", rv_codec_ci_lwsp, rv_fmt_frd_offset_rs1, NULL, rv_op_flw, 0, 0 },613{ "c.jr", rv_codec_cr_jr, rv_fmt_rd_rs1_offset, NULL, rv_op_jalr, rv_op_jalr, rv_op_jalr },614{ "c.mv", rv_codec_cr_mv, rv_fmt_rd_rs1_rs2, NULL, rv_op_addi, rv_op_addi, rv_op_addi },615{ "c.ebreak", rv_codec_ci_none, rv_fmt_none, NULL, rv_op_ebreak, rv_op_ebreak, rv_op_ebreak },616{ "c.jalr", rv_codec_cr_jalr, rv_fmt_rd_rs1_offset, NULL, rv_op_jalr, rv_op_jalr, rv_op_jalr },617{ "c.add", rv_codec_cr, rv_fmt_rd_rs1_rs2, NULL, rv_op_add, rv_op_add, rv_op_add },618{ "c.fsdsp", rv_codec_css_sdsp, rv_fmt_frs2_offset_rs1, NULL, rv_op_fsd, rv_op_fsd, rv_op_fsd },619{ "c.swsp", rv_codec_css_swsp, rv_fmt_rs2_offset_rs1, NULL, rv_op_sw, rv_op_sw, rv_op_sw },620{ "c.fswsp", rv_codec_css_swsp, rv_fmt_frs2_offset_rs1, NULL, rv_op_fsw, 0, 0 },621{ "c.ld", rv_codec_cl_ld, rv_fmt_rd_offset_rs1, NULL, 0, rv_op_ld, rv_op_ld },622{ "c.sd", rv_codec_cs_sd, rv_fmt_rs2_offset_rs1, NULL, 0, rv_op_sd, rv_op_sd },623{ "c.addiw", rv_codec_ci, rv_fmt_rd_rs1_imm, NULL, 0, rv_op_addiw, rv_op_addiw },624{ "c.ldsp", rv_codec_ci_ldsp, rv_fmt_rd_offset_rs1, NULL, 0, rv_op_ld, rv_op_ld },625{ "c.sdsp", rv_codec_css_sdsp, rv_fmt_rs2_offset_rs1, NULL, 0, rv_op_sd, rv_op_sd },626{ "c.lq", rv_codec_cl_lq, rv_fmt_rd_offset_rs1, NULL, 0, 0, rv_op_lq },627{ "c.sq", rv_codec_cs_sq, rv_fmt_rs2_offset_rs1, NULL, 0, 0, rv_op_sq },628{ "c.lqsp", rv_codec_ci_lqsp, rv_fmt_rd_offset_rs1, NULL, 0, 0, rv_op_lq },629{ "c.sqsp", rv_codec_css_sqsp, rv_fmt_rs2_offset_rs1, NULL, 0, 0, rv_op_sq },630{ "nop", rv_codec_i, rv_fmt_none, NULL, 0, 0, 0 },631{ "mv", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },632{ "not", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },633{ "neg", rv_codec_r, rv_fmt_rd_rs2, NULL, 0, 0, 0 },634{ "negw", rv_codec_r, rv_fmt_rd_rs2, NULL, 0, 0, 0 },635{ "sext.w", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },636{ "seqz", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },637{ "snez", rv_codec_r, rv_fmt_rd_rs2, NULL, 0, 0, 0 },638{ "sltz", rv_codec_r, rv_fmt_rd_rs1, NULL, 0, 0, 0 },639{ "sgtz", rv_codec_r, rv_fmt_rd_rs2, NULL, 0, 0, 0 },640{ "fmv.s", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },641{ "fabs.s", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },642{ "fneg.s", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },643{ "fmv.d", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },644{ "fabs.d", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },645{ "fneg.d", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },646{ "fmv.q", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },647{ "fabs.q", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },648{ "fneg.q", rv_codec_r, rv_fmt_frd_frs1, NULL, 0, 0, 0 },649{ "beqz", rv_codec_sb, rv_fmt_rs1_offset, NULL, 0, 0, 0 },650{ "bnez", rv_codec_sb, rv_fmt_rs1_offset, NULL, 0, 0, 0 },651{ "blez", rv_codec_sb, rv_fmt_rs2_offset, NULL, 0, 0, 0 },652{ "bgez", rv_codec_sb, rv_fmt_rs1_offset, NULL, 0, 0, 0 },653{ "bltz", rv_codec_sb, rv_fmt_rs1_offset, NULL, 0, 0, 0 },654{ "bgtz", rv_codec_sb, rv_fmt_rs2_offset, NULL, 0, 0, 0 },655{ "ble", rv_codec_sb, rv_fmt_rs2_rs1_offset, NULL, 0, 0, 0 },656{ "bleu", rv_codec_sb, rv_fmt_rs2_rs1_offset, NULL, 0, 0, 0 },657{ "bgt", rv_codec_sb, rv_fmt_rs2_rs1_offset, NULL, 0, 0, 0 },658{ "bgtu", rv_codec_sb, rv_fmt_rs2_rs1_offset, NULL, 0, 0, 0 },659{ "j", rv_codec_uj, rv_fmt_offset, NULL, 0, 0, 0 },660{ "ret", rv_codec_i, rv_fmt_none, NULL, 0, 0, 0 },661{ "jr", rv_codec_i, rv_fmt_rs1, NULL, 0, 0, 0 },662{ "rdcycle", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },663{ "rdtime", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },664{ "rdinstret", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },665{ "rdcycleh", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },666{ "rdtimeh", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },667{ "rdinstreth", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },668{ "frcsr", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },669{ "frrm", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },670{ "frflags", rv_codec_i_csr, rv_fmt_rd, NULL, 0, 0, 0 },671{ "fscsr", rv_codec_i_csr, rv_fmt_rd_rs1, NULL, 0, 0, 0 },672{ "fsrm", rv_codec_i_csr, rv_fmt_rd_rs1, NULL, 0, 0, 0 },673{ "fsflags", rv_codec_i_csr, rv_fmt_rd_rs1, NULL, 0, 0, 0 },674{ "fsrmi", rv_codec_i_csr, rv_fmt_rd_zimm, NULL, 0, 0, 0 },675{ "fsflagsi", rv_codec_i_csr, rv_fmt_rd_zimm, NULL, 0, 0, 0 },676{ "add.uw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },677{ "andn", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },678{ "bclr", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },679{ "bclri", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },680{ "bext", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },681{ "bexti", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },682{ "binv", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },683{ "binvi", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },684{ "bset", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },685{ "bseti", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },686{ "clmul", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },687{ "clmulh", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },688{ "clmulr", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },689{ "clz", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },690{ "clzw", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },691{ "cpop", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },692{ "cpopw", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },693{ "ctz", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },694{ "ctzw", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },695{ "max", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },696{ "maxu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },697{ "min", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },698{ "minu", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },699{ "orc.b", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },700{ "orn", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },701{ "rev8", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },702{ "rol", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },703{ "rolw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },704{ "ror", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },705{ "rori", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },706{ "roriw", rv_codec_i_sh7, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },707{ "rorw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },708{ "sext.b", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },709{ "sext.h", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },710{ "sh1add", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },711{ "sh1add.uw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },712{ "sh2add", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },713{ "sh2add.uw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },714{ "sh3add", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },715{ "sh3add.uw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },716{ "slli.uw", rv_codec_i_sh6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },717{ "xnor", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },718{ "zext.h", rv_codec_i, rv_fmt_rd_rs1, NULL, 0, 0, 0 },719};720721/* CSR names */722723static const char *csr_name(int csrno)724{725switch (csrno) {726case 0x0000: return "ustatus";727case 0x0001: return "fflags";728case 0x0002: return "frm";729case 0x0003: return "fcsr";730case 0x0004: return "uie";731case 0x0005: return "utvec";732case 0x0007: return "utvt";733case 0x0008: return "vstart";734case 0x0009: return "vxsat";735case 0x000a: return "vxrm";736case 0x000f: return "vcsr";737case 0x0040: return "uscratch";738case 0x0041: return "uepc";739case 0x0042: return "ucause";740case 0x0043: return "utval";741case 0x0044: return "uip";742case 0x0045: return "unxti";743case 0x0046: return "uintstatus";744case 0x0048: return "uscratchcsw";745case 0x0049: return "uscratchcswl";746case 0x0100: return "sstatus";747case 0x0102: return "sedeleg";748case 0x0103: return "sideleg";749case 0x0104: return "sie";750case 0x0105: return "stvec";751case 0x0106: return "scounteren";752case 0x0107: return "stvt";753case 0x0140: return "sscratch";754case 0x0141: return "sepc";755case 0x0142: return "scause";756case 0x0143: return "stval";757case 0x0144: return "sip";758case 0x0145: return "snxti";759case 0x0146: return "sintstatus";760case 0x0148: return "sscratchcsw";761case 0x0149: return "sscratchcswl";762case 0x0180: return "satp";763case 0x0200: return "vsstatus";764case 0x0204: return "vsie";765case 0x0205: return "vstvec";766case 0x0240: return "vsscratch";767case 0x0241: return "vsepc";768case 0x0242: return "vscause";769case 0x0243: return "vstval";770case 0x0244: return "vsip";771case 0x0280: return "vsatp";772case 0x0300: return "mstatus";773case 0x0301: return "misa";774case 0x0302: return "medeleg";775case 0x0303: return "mideleg";776case 0x0304: return "mie";777case 0x0305: return "mtvec";778case 0x0306: return "mcounteren";779case 0x0307: return "mtvt";780case 0x0310: return "mstatush";781case 0x0320: return "mcountinhibit";782case 0x0323: return "mhpmevent3";783case 0x0324: return "mhpmevent4";784case 0x0325: return "mhpmevent5";785case 0x0326: return "mhpmevent6";786case 0x0327: return "mhpmevent7";787case 0x0328: return "mhpmevent8";788case 0x0329: return "mhpmevent9";789case 0x032a: return "mhpmevent10";790case 0x032b: return "mhpmevent11";791case 0x032c: return "mhpmevent12";792case 0x032d: return "mhpmevent13";793case 0x032e: return "mhpmevent14";794case 0x032f: return "mhpmevent15";795case 0x0330: return "mhpmevent16";796case 0x0331: return "mhpmevent17";797case 0x0332: return "mhpmevent18";798case 0x0333: return "mhpmevent19";799case 0x0334: return "mhpmevent20";800case 0x0335: return "mhpmevent21";801case 0x0336: return "mhpmevent22";802case 0x0337: return "mhpmevent23";803case 0x0338: return "mhpmevent24";804case 0x0339: return "mhpmevent25";805case 0x033a: return "mhpmevent26";806case 0x033b: return "mhpmevent27";807case 0x033c: return "mhpmevent28";808case 0x033d: return "mhpmevent29";809case 0x033e: return "mhpmevent30";810case 0x033f: return "mhpmevent31";811case 0x0340: return "mscratch";812case 0x0341: return "mepc";813case 0x0342: return "mcause";814case 0x0343: return "mtval";815case 0x0344: return "mip";816case 0x0345: return "mnxti";817case 0x0346: return "mintstatus";818case 0x0348: return "mscratchcsw";819case 0x0349: return "mscratchcswl";820case 0x034a: return "mtinst";821case 0x034b: return "mtval2";822case 0x03a0: return "pmpcfg0";823case 0x03a1: return "pmpcfg1";824case 0x03a2: return "pmpcfg2";825case 0x03a3: return "pmpcfg3";826case 0x03b0: return "pmpaddr0";827case 0x03b1: return "pmpaddr1";828case 0x03b2: return "pmpaddr2";829case 0x03b3: return "pmpaddr3";830case 0x03b4: return "pmpaddr4";831case 0x03b5: return "pmpaddr5";832case 0x03b6: return "pmpaddr6";833case 0x03b7: return "pmpaddr7";834case 0x03b8: return "pmpaddr8";835case 0x03b9: return "pmpaddr9";836case 0x03ba: return "pmpaddr10";837case 0x03bb: return "pmpaddr11";838case 0x03bc: return "pmpaddr12";839case 0x03bd: return "pmpaddr13";840case 0x03be: return "pmpaddr14";841case 0x03bf: return "pmpaddr15";842case 0x0600: return "hstatus";843case 0x0602: return "hedeleg";844case 0x0603: return "hideleg";845case 0x0604: return "hie";846case 0x0605: return "htimedelta";847case 0x0606: return "hcounteren";848case 0x0607: return "hgeie";849case 0x0615: return "htimedeltah";850case 0x0643: return "htval";851case 0x0644: return "hip";852case 0x0645: return "hvip";853case 0x064a: return "htinst";854case 0x0680: return "hgatp";855case 0x07a0: return "tselect";856case 0x07a1: return "tdata1";857case 0x07a2: return "tdata2";858case 0x07a3: return "tdata3";859case 0x07a4: return "tinfo";860case 0x07a5: return "tcontrol";861case 0x07a8: return "mcontext";862case 0x07a9: return "mnoise";863case 0x07aa: return "scontext";864case 0x07b0: return "dcsr";865case 0x07b1: return "dpc";866case 0x07b2: return "dscratch0";867case 0x07b3: return "dscratch1";868case 0x0b00: return "mcycle";869case 0x0b02: return "minstret";870case 0x0b03: return "mhpmcounter3";871case 0x0b04: return "mhpmcounter4";872case 0x0b05: return "mhpmcounter5";873case 0x0b06: return "mhpmcounter6";874case 0x0b07: return "mhpmcounter7";875case 0x0b08: return "mhpmcounter8";876case 0x0b09: return "mhpmcounter9";877case 0x0b0a: return "mhpmcounter10";878case 0x0b0b: return "mhpmcounter11";879case 0x0b0c: return "mhpmcounter12";880case 0x0b0d: return "mhpmcounter13";881case 0x0b0e: return "mhpmcounter14";882case 0x0b0f: return "mhpmcounter15";883case 0x0b10: return "mhpmcounter16";884case 0x0b11: return "mhpmcounter17";885case 0x0b12: return "mhpmcounter18";886case 0x0b13: return "mhpmcounter19";887case 0x0b14: return "mhpmcounter20";888case 0x0b15: return "mhpmcounter21";889case 0x0b16: return "mhpmcounter22";890case 0x0b17: return "mhpmcounter23";891case 0x0b18: return "mhpmcounter24";892case 0x0b19: return "mhpmcounter25";893case 0x0b1a: return "mhpmcounter26";894case 0x0b1b: return "mhpmcounter27";895case 0x0b1c: return "mhpmcounter28";896case 0x0b1d: return "mhpmcounter29";897case 0x0b1e: return "mhpmcounter30";898case 0x0b1f: return "mhpmcounter31";899case 0x0b80: return "mcycleh";900case 0x0b82: return "minstreth";901case 0x0b83: return "mhpmcounter3h";902case 0x0b84: return "mhpmcounter4h";903case 0x0b85: return "mhpmcounter5h";904case 0x0b86: return "mhpmcounter6h";905case 0x0b87: return "mhpmcounter7h";906case 0x0b88: return "mhpmcounter8h";907case 0x0b89: return "mhpmcounter9h";908case 0x0b8a: return "mhpmcounter10h";909case 0x0b8b: return "mhpmcounter11h";910case 0x0b8c: return "mhpmcounter12h";911case 0x0b8d: return "mhpmcounter13h";912case 0x0b8e: return "mhpmcounter14h";913case 0x0b8f: return "mhpmcounter15h";914case 0x0b90: return "mhpmcounter16h";915case 0x0b91: return "mhpmcounter17h";916case 0x0b92: return "mhpmcounter18h";917case 0x0b93: return "mhpmcounter19h";918case 0x0b94: return "mhpmcounter20h";919case 0x0b95: return "mhpmcounter21h";920case 0x0b96: return "mhpmcounter22h";921case 0x0b97: return "mhpmcounter23h";922case 0x0b98: return "mhpmcounter24h";923case 0x0b99: return "mhpmcounter25h";924case 0x0b9a: return "mhpmcounter26h";925case 0x0b9b: return "mhpmcounter27h";926case 0x0b9c: return "mhpmcounter28h";927case 0x0b9d: return "mhpmcounter29h";928case 0x0b9e: return "mhpmcounter30h";929case 0x0b9f: return "mhpmcounter31h";930case 0x0c00: return "cycle";931case 0x0c01: return "time";932case 0x0c02: return "instret";933case 0x0c03: return "hpmcounter3";934case 0x0c04: return "hpmcounter4";935case 0x0c05: return "hpmcounter5";936case 0x0c06: return "hpmcounter6";937case 0x0c07: return "hpmcounter7";938case 0x0c08: return "hpmcounter8";939case 0x0c09: return "hpmcounter9";940case 0x0c0a: return "hpmcounter10";941case 0x0c0b: return "hpmcounter11";942case 0x0c0c: return "hpmcounter12";943case 0x0c0d: return "hpmcounter13";944case 0x0c0e: return "hpmcounter14";945case 0x0c0f: return "hpmcounter15";946case 0x0c10: return "hpmcounter16";947case 0x0c11: return "hpmcounter17";948case 0x0c12: return "hpmcounter18";949case 0x0c13: return "hpmcounter19";950case 0x0c14: return "hpmcounter20";951case 0x0c15: return "hpmcounter21";952case 0x0c16: return "hpmcounter22";953case 0x0c17: return "hpmcounter23";954case 0x0c18: return "hpmcounter24";955case 0x0c19: return "hpmcounter25";956case 0x0c1a: return "hpmcounter26";957case 0x0c1b: return "hpmcounter27";958case 0x0c1c: return "hpmcounter28";959case 0x0c1d: return "hpmcounter29";960case 0x0c1e: return "hpmcounter30";961case 0x0c1f: return "hpmcounter31";962case 0x0c20: return "vl";963case 0x0c21: return "vtype";964case 0x0c22: return "vlenb";965case 0x0c80: return "cycleh";966case 0x0c81: return "timeh";967case 0x0c82: return "instreth";968case 0x0c83: return "hpmcounter3h";969case 0x0c84: return "hpmcounter4h";970case 0x0c85: return "hpmcounter5h";971case 0x0c86: return "hpmcounter6h";972case 0x0c87: return "hpmcounter7h";973case 0x0c88: return "hpmcounter8h";974case 0x0c89: return "hpmcounter9h";975case 0x0c8a: return "hpmcounter10h";976case 0x0c8b: return "hpmcounter11h";977case 0x0c8c: return "hpmcounter12h";978case 0x0c8d: return "hpmcounter13h";979case 0x0c8e: return "hpmcounter14h";980case 0x0c8f: return "hpmcounter15h";981case 0x0c90: return "hpmcounter16h";982case 0x0c91: return "hpmcounter17h";983case 0x0c92: return "hpmcounter18h";984case 0x0c93: return "hpmcounter19h";985case 0x0c94: return "hpmcounter20h";986case 0x0c95: return "hpmcounter21h";987case 0x0c96: return "hpmcounter22h";988case 0x0c97: return "hpmcounter23h";989case 0x0c98: return "hpmcounter24h";990case 0x0c99: return "hpmcounter25h";991case 0x0c9a: return "hpmcounter26h";992case 0x0c9b: return "hpmcounter27h";993case 0x0c9c: return "hpmcounter28h";994case 0x0c9d: return "hpmcounter29h";995case 0x0c9e: return "hpmcounter30h";996case 0x0c9f: return "hpmcounter31h";997case 0x0e12: return "hgeip";998case 0x0f11: return "mvendorid";999case 0x0f12: return "marchid";1000case 0x0f13: return "mimpid";1001case 0x0f14: return "mhartid";1002case 0x0f15: return "mentropy";1003default: return NULL;1004}1005}10061007/* decode opcode */10081009static void decode_inst_opcode(rv_decode *dec, rv_isa isa)1010{1011rv_inst inst = dec->inst;1012rv_opcode op = rv_op_illegal;1013switch (((inst >> 0) & 0b11)) {1014case 0:1015switch (((inst >> 13) & 0b111)) {1016case 0: op = rv_op_c_addi4spn; break;1017case 1: op = (isa == rv128) ? rv_op_c_lq : rv_op_c_fld; break;1018case 2: op = rv_op_c_lw; break;1019case 3: op = (isa == rv32) ? rv_op_c_flw : rv_op_c_ld; break;1020case 5: op = (isa == rv128) ? rv_op_c_sq : rv_op_c_fsd; break;1021case 6: op = rv_op_c_sw; break;1022case 7: op = (isa == rv32) ? rv_op_c_fsw : rv_op_c_sd; break;1023}1024break;1025case 1:1026switch (((inst >> 13) & 0b111)) {1027case 0:1028switch (((inst >> 2) & 0b11111111111)) {1029case 0: op = rv_op_c_nop; break;1030default: op = rv_op_c_addi; break;1031}1032break;1033case 1: op = (isa == rv32) ? rv_op_c_jal : rv_op_c_addiw; break;1034case 2: op = rv_op_c_li; break;1035case 3:1036switch (((inst >> 7) & 0b11111)) {1037case 2: op = rv_op_c_addi16sp; break;1038default: op = rv_op_c_lui; break;1039}1040break;1041case 4:1042switch (((inst >> 10) & 0b11)) {1043case 0:1044op = rv_op_c_srli;1045break;1046case 1:1047op = rv_op_c_srai;1048break;1049case 2: op = rv_op_c_andi; break;1050case 3:1051switch (((inst >> 10) & 0b100) | ((inst >> 5) & 0b011)) {1052case 0: op = rv_op_c_sub; break;1053case 1: op = rv_op_c_xor; break;1054case 2: op = rv_op_c_or; break;1055case 3: op = rv_op_c_and; break;1056case 4: op = rv_op_c_subw; break;1057case 5: op = rv_op_c_addw; break;1058}1059break;1060}1061break;1062case 5: op = rv_op_c_j; break;1063case 6: op = rv_op_c_beqz; break;1064case 7: op = rv_op_c_bnez; break;1065}1066break;1067case 2:1068switch (((inst >> 13) & 0b111)) {1069case 0:1070op = rv_op_c_slli;1071break;1072case 1: op = (isa == rv128) ? rv_op_c_lqsp : rv_op_c_fldsp; break;1073case 2: op = rv_op_c_lwsp; break;1074case 3: op = (isa == rv32) ? rv_op_c_flwsp : rv_op_c_ldsp; break;1075case 4:1076switch (((inst >> 12) & 0b1)) {1077case 0:1078switch (((inst >> 2) & 0b11111)) {1079case 0: op = rv_op_c_jr; break;1080default: op = rv_op_c_mv; break;1081}1082break;1083case 1:1084switch (((inst >> 2) & 0b11111)) {1085case 0:1086switch (((inst >> 7) & 0b11111)) {1087case 0: op = rv_op_c_ebreak; break;1088default: op = rv_op_c_jalr; break;1089}1090break;1091default: op = rv_op_c_add; break;1092}1093break;1094}1095break;1096case 5: op = (isa == rv128) ? rv_op_c_sqsp : rv_op_c_fsdsp; break;1097case 6: op = rv_op_c_swsp; break;1098case 7: op = (isa == rv32) ? rv_op_c_fswsp : rv_op_c_sdsp; break;1099}1100break;1101case 3:1102switch (((inst >> 2) & 0b11111)) {1103case 0:1104switch (((inst >> 12) & 0b111)) {1105case 0: op = rv_op_lb; break;1106case 1: op = rv_op_lh; break;1107case 2: op = rv_op_lw; break;1108case 3: op = rv_op_ld; break;1109case 4: op = rv_op_lbu; break;1110case 5: op = rv_op_lhu; break;1111case 6: op = rv_op_lwu; break;1112case 7: op = rv_op_ldu; break;1113}1114break;1115case 1:1116switch (((inst >> 12) & 0b111)) {1117case 1: op = rv_op_flh; break;1118case 2: op = rv_op_flw; break;1119case 3: op = rv_op_fld; break;1120case 4: op = rv_op_flq; break;1121}1122break;1123case 3:1124switch (((inst >> 12) & 0b111)) {1125case 0: op = rv_op_fence; break;1126case 1: op = rv_op_fence_i; break;1127case 2: op = rv_op_lq; break;1128}1129break;1130case 4:1131switch (((inst >> 12) & 0b111)) {1132case 0: op = rv_op_addi; break;1133case 1:1134switch (((inst >> 27) & 0b11111)) {1135case 0: op = rv_op_slli; break;1136case 5: op = rv_op_bseti; break;1137case 9: op = rv_op_bclri; break;1138case 12:1139switch (((inst >> 20) & 0b11111)) {1140case 0: op = rv_op_clz; break;1141case 1: op = rv_op_ctz; break;1142case 2: op = rv_op_cpop; break;1143case 4: op = rv_op_sext_b; break;1144case 5: op = rv_op_sext_h; break;1145}1146break;1147case 13: op = rv_op_binvi; break;1148}1149break;1150case 2: op = rv_op_slti; break;1151case 3: op = rv_op_sltiu; break;1152case 4: op = rv_op_xori; break;1153case 5:1154switch (((inst >> 27) & 0b11111)) {1155case 0: op = rv_op_srli; break;1156case 5:1157switch (((inst >> 20) & 0b1111111)) {1158case 7: op = rv_op_orc_b; break;1159}1160break;1161case 8: op = rv_op_srai; break;1162case 9: op = rv_op_bexti; break;1163case 12: op = rv_op_rori; break;1164case 13:1165switch (((inst >> 20) & 0b1111111)) {1166case 24: if (isa == rv32) op = rv_op_rev8; break;1167case 56: if (isa == rv64) op = rv_op_rev8; break;1168}1169break;1170}1171break;1172case 6: op = rv_op_ori; break;1173case 7: op = rv_op_andi; break;1174}1175break;1176case 5: op = rv_op_auipc; break;1177case 6:1178switch (((inst >> 12) & 0b111)) {1179case 0: op = rv_op_addiw; break;1180case 1:1181switch (((inst >> 25) & 0b1111111)) {1182case 0: op = rv_op_slliw; break;1183case 4: op = rv_op_slli_uw; break;1184case 5: op = rv_op_slli_uw; break;1185case 48:1186switch (((inst >> 20) & 0b11111)) {1187case 0: op = rv_op_clzw; break;1188case 1: op = rv_op_ctzw; break;1189case 2: op = rv_op_cpopw; break;1190}1191break;1192}1193break;1194case 5:1195switch (((inst >> 25) & 0b1111111)) {1196case 0: op = rv_op_srliw; break;1197case 32: op = rv_op_sraiw; break;1198case 48: op = rv_op_roriw; break;1199}1200break;1201}1202break;1203case 8:1204switch (((inst >> 12) & 0b111)) {1205case 0: op = rv_op_sb; break;1206case 1: op = rv_op_sh; break;1207case 2: op = rv_op_sw; break;1208case 3: op = rv_op_sd; break;1209case 4: op = rv_op_sq; break;1210}1211break;1212case 9:1213switch (((inst >> 12) & 0b111)) {1214case 1: op = rv_op_fsh; break;1215case 2: op = rv_op_fsw; break;1216case 3: op = rv_op_fsd; break;1217case 4: op = rv_op_fsq; break;1218}1219break;1220case 11:1221switch (((inst >> 24) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1222case 2: op = rv_op_amoadd_w; break;1223case 3: op = rv_op_amoadd_d; break;1224case 4: op = rv_op_amoadd_q; break;1225case 10: op = rv_op_amoswap_w; break;1226case 11: op = rv_op_amoswap_d; break;1227case 12: op = rv_op_amoswap_q; break;1228case 18:1229switch (((inst >> 20) & 0b11111)) {1230case 0: op = rv_op_lr_w; break;1231}1232break;1233case 19:1234switch (((inst >> 20) & 0b11111)) {1235case 0: op = rv_op_lr_d; break;1236}1237break;1238case 20:1239switch (((inst >> 20) & 0b11111)) {1240case 0: op = rv_op_lr_q; break;1241}1242break;1243case 26: op = rv_op_sc_w; break;1244case 27: op = rv_op_sc_d; break;1245case 28: op = rv_op_sc_q; break;1246case 34: op = rv_op_amoxor_w; break;1247case 35: op = rv_op_amoxor_d; break;1248case 36: op = rv_op_amoxor_q; break;1249case 66: op = rv_op_amoor_w; break;1250case 67: op = rv_op_amoor_d; break;1251case 68: op = rv_op_amoor_q; break;1252case 98: op = rv_op_amoand_w; break;1253case 99: op = rv_op_amoand_d; break;1254case 100: op = rv_op_amoand_q; break;1255case 130: op = rv_op_amomin_w; break;1256case 131: op = rv_op_amomin_d; break;1257case 132: op = rv_op_amomin_q; break;1258case 162: op = rv_op_amomax_w; break;1259case 163: op = rv_op_amomax_d; break;1260case 164: op = rv_op_amomax_q; break;1261case 194: op = rv_op_amominu_w; break;1262case 195: op = rv_op_amominu_d; break;1263case 196: op = rv_op_amominu_q; break;1264case 226: op = rv_op_amomaxu_w; break;1265case 227: op = rv_op_amomaxu_d; break;1266case 228: op = rv_op_amomaxu_q; break;1267}1268break;1269case 12:1270switch (((inst >> 22) & 0b1111111000) | ((inst >> 12) & 0b0000000111)) {1271case 0: op = rv_op_add; break;1272case 1: op = rv_op_sll; break;1273case 2: op = rv_op_slt; break;1274case 3: op = rv_op_sltu; break;1275case 4: op = rv_op_xor; break;1276case 5: op = rv_op_srl; break;1277case 6: op = rv_op_or; break;1278case 7: op = rv_op_and; break;1279case 8: op = rv_op_mul; break;1280case 9: op = rv_op_mulh; break;1281case 10: op = rv_op_mulhsu; break;1282case 11: op = rv_op_mulhu; break;1283case 12: op = rv_op_div; break;1284case 13: op = rv_op_divu; break;1285case 14: op = rv_op_rem; break;1286case 15: op = rv_op_remu; break;1287case 36:1288switch (((inst >> 20) & 0b11111)) {1289case 0: if (isa == rv32) op = rv_op_zext_h; break;1290}1291break;1292case 41: op = rv_op_clmul; break;1293case 42: op = rv_op_clmulr; break;1294case 43: op = rv_op_clmulh; break;1295case 44: op = rv_op_min; break;1296case 45: op = rv_op_minu; break;1297case 46: op = rv_op_max; break;1298case 47: op = rv_op_maxu; break;1299case 130: op = rv_op_sh1add; break;1300case 132: op = rv_op_sh2add; break;1301case 134: op = rv_op_sh3add; break;1302case 161: op = rv_op_bset; break;1303case 256: op = rv_op_sub; break;1304case 260: op = rv_op_xnor; break;1305case 261: op = rv_op_sra; break;1306case 262: op = rv_op_orn; break;1307case 263: op = rv_op_andn; break;1308case 289: op = rv_op_bclr; break;1309case 293: op = rv_op_bext; break;1310case 385: op = rv_op_rol; break;1311case 389: op = rv_op_ror; break;1312case 417: op = rv_op_binv; break;1313}1314break;1315case 13: op = rv_op_lui; break;1316case 14:1317switch (((inst >> 22) & 0b1111111000) | ((inst >> 12) & 0b0000000111)) {1318case 0: op = rv_op_addw; break;1319case 1: op = rv_op_sllw; break;1320case 5: op = rv_op_srlw; break;1321case 8: op = rv_op_mulw; break;1322case 12: op = rv_op_divw; break;1323case 13: op = rv_op_divuw; break;1324case 14: op = rv_op_remw; break;1325case 15: op = rv_op_remuw; break;1326case 32: op = rv_op_add_uw; break;1327case 36:1328switch (((inst >> 20) & 0b11111)) {1329case 0: if (isa == rv64) op = rv_op_zext_h; break;1330}1331break;1332case 130: op = rv_op_sh1add_uw; break;1333case 132: op = rv_op_sh2add_uw; break;1334case 134: op = rv_op_sh3add_uw; break;1335case 256: op = rv_op_subw; break;1336case 261: op = rv_op_sraw; break;1337case 385: op = rv_op_rolw; break;1338case 389: op = rv_op_rorw; break;1339}1340break;1341case 16:1342switch (((inst >> 25) & 0b11)) {1343case 0: op = rv_op_fmadd_s; break;1344case 1: op = rv_op_fmadd_d; break;1345case 2: op = rv_op_fmadd_h; break;1346case 3: op = rv_op_fmadd_q; break;1347}1348break;1349case 17:1350switch (((inst >> 25) & 0b11)) {1351case 0: op = rv_op_fmsub_s; break;1352case 1: op = rv_op_fmsub_d; break;1353case 2: op = rv_op_fmsub_h; break;1354case 3: op = rv_op_fmsub_q; break;1355}1356break;1357case 18:1358switch (((inst >> 25) & 0b11)) {1359case 0: op = rv_op_fnmsub_s; break;1360case 1: op = rv_op_fnmsub_d; break;1361case 2: op = rv_op_fnmsub_h; break;1362case 3: op = rv_op_fnmsub_q; break;1363}1364break;1365case 19:1366switch (((inst >> 25) & 0b11)) {1367case 0: op = rv_op_fnmadd_s; break;1368case 1: op = rv_op_fnmadd_d; break;1369case 2: op = rv_op_fnmadd_h; break;1370case 3: op = rv_op_fnmadd_q; break;1371}1372break;1373case 20:1374switch (((inst >> 25) & 0b1111111)) {1375case 0: op = rv_op_fadd_s; break;1376case 1: op = rv_op_fadd_d; break;1377case 2: op = rv_op_fadd_h; break;1378case 3: op = rv_op_fadd_q; break;1379case 4: op = rv_op_fsub_s; break;1380case 5: op = rv_op_fsub_d; break;1381case 6: op = rv_op_fsub_h; break;1382case 7: op = rv_op_fsub_q; break;1383case 8: op = rv_op_fmul_s; break;1384case 9: op = rv_op_fmul_d; break;1385case 10: op = rv_op_fmul_h; break;1386case 11: op = rv_op_fmul_q; break;1387case 12: op = rv_op_fdiv_s; break;1388case 13: op = rv_op_fdiv_d; break;1389case 14: op = rv_op_fdiv_h; break;1390case 15: op = rv_op_fdiv_q; break;1391case 16:1392switch (((inst >> 12) & 0b111)) {1393case 0: op = rv_op_fsgnj_s; break;1394case 1: op = rv_op_fsgnjn_s; break;1395case 2: op = rv_op_fsgnjx_s; break;1396}1397break;1398case 17:1399switch (((inst >> 12) & 0b111)) {1400case 0: op = rv_op_fsgnj_d; break;1401case 1: op = rv_op_fsgnjn_d; break;1402case 2: op = rv_op_fsgnjx_d; break;1403}1404break;1405case 18:1406switch (((inst >> 12) & 0b111)) {1407case 0: op = rv_op_fsgnj_h; break;1408case 1: op = rv_op_fsgnjn_h; break;1409case 2: op = rv_op_fsgnjx_h; break;1410}1411break;1412case 19:1413switch (((inst >> 12) & 0b111)) {1414case 0: op = rv_op_fsgnj_q; break;1415case 1: op = rv_op_fsgnjn_q; break;1416case 2: op = rv_op_fsgnjx_q; break;1417}1418break;1419case 20:1420switch (((inst >> 12) & 0b111)) {1421case 0: op = rv_op_fmin_s; break;1422case 1: op = rv_op_fmax_s; break;1423}1424break;1425case 21:1426switch (((inst >> 12) & 0b111)) {1427case 0: op = rv_op_fmin_d; break;1428case 1: op = rv_op_fmax_d; break;1429}1430break;1431case 22:1432switch (((inst >> 12) & 0b111)) {1433case 0: op = rv_op_fmin_h; break;1434case 1: op = rv_op_fmax_h; break;1435}1436break;1437case 23:1438switch (((inst >> 12) & 0b111)) {1439case 0: op = rv_op_fmin_q; break;1440case 1: op = rv_op_fmax_q; break;1441}1442break;1443case 32:1444switch (((inst >> 20) & 0b11111)) {1445case 1: op = rv_op_fcvt_s_d; break;1446case 2: op = rv_op_fcvt_s_h; break;1447case 3: op = rv_op_fcvt_s_q; break;1448}1449break;1450case 33:1451switch (((inst >> 20) & 0b11111)) {1452case 0: op = rv_op_fcvt_d_s; break;1453case 2: op = rv_op_fcvt_d_h; break;1454case 3: op = rv_op_fcvt_d_q; break;1455}1456break;1457case 34:1458switch (((inst >> 20) & 0b11111)) {1459case 0: op = rv_op_fcvt_h_s; break;1460case 1: op = rv_op_fcvt_h_d; break;1461case 3: op = rv_op_fcvt_h_q; break;1462}1463break;1464case 35:1465switch (((inst >> 20) & 0b11111)) {1466case 0: op = rv_op_fcvt_q_s; break;1467case 1: op = rv_op_fcvt_q_d; break;1468case 2: op = rv_op_fcvt_q_h; break;1469}1470break;1471case 44:1472switch (((inst >> 20) & 0b11111)) {1473case 0: op = rv_op_fsqrt_s; break;1474}1475break;1476case 45:1477switch (((inst >> 20) & 0b11111)) {1478case 0: op = rv_op_fsqrt_d; break;1479}1480break;1481case 46:1482switch (((inst >> 20) & 0b11111)) {1483case 0: op = rv_op_fsqrt_h; break;1484}1485break;1486case 47:1487switch (((inst >> 20) & 0b11111)) {1488case 0: op = rv_op_fsqrt_q; break;1489}1490break;1491case 80:1492switch (((inst >> 12) & 0b111)) {1493case 0: op = rv_op_fle_s; break;1494case 1: op = rv_op_flt_s; break;1495case 2: op = rv_op_feq_s; break;1496}1497break;1498case 81:1499switch (((inst >> 12) & 0b111)) {1500case 0: op = rv_op_fle_d; break;1501case 1: op = rv_op_flt_d; break;1502case 2: op = rv_op_feq_d; break;1503}1504break;1505case 82:1506switch (((inst >> 12) & 0b111)) {1507case 0: op = rv_op_fle_h; break;1508case 1: op = rv_op_flt_h; break;1509case 2: op = rv_op_feq_h; break;1510}1511break;1512case 83:1513switch (((inst >> 12) & 0b111)) {1514case 0: op = rv_op_fle_q; break;1515case 1: op = rv_op_flt_q; break;1516case 2: op = rv_op_feq_q; break;1517}1518break;1519case 96:1520switch (((inst >> 20) & 0b11111)) {1521case 0: op = rv_op_fcvt_w_s; break;1522case 1: op = rv_op_fcvt_wu_s; break;1523case 2: op = rv_op_fcvt_l_s; break;1524case 3: op = rv_op_fcvt_lu_s; break;1525}1526break;1527case 97:1528switch (((inst >> 20) & 0b11111)) {1529case 0: op = rv_op_fcvt_w_d; break;1530case 1: op = rv_op_fcvt_wu_d; break;1531case 2: op = rv_op_fcvt_l_d; break;1532case 3: op = rv_op_fcvt_lu_d; break;1533}1534break;1535case 98:1536switch (((inst >> 20) & 0b11111)) {1537case 0: op = rv_op_fcvt_w_h; break;1538case 1: op = rv_op_fcvt_wu_h; break;1539case 2: op = rv_op_fcvt_l_h; break;1540case 3: op = rv_op_fcvt_lu_h; break;1541}1542break;1543case 99:1544switch (((inst >> 20) & 0b11111)) {1545case 0: op = rv_op_fcvt_w_q; break;1546case 1: op = rv_op_fcvt_wu_q; break;1547case 2: op = rv_op_fcvt_l_q; break;1548case 3: op = rv_op_fcvt_lu_q; break;1549}1550break;1551case 104:1552switch (((inst >> 20) & 0b11111)) {1553case 0: op = rv_op_fcvt_s_w; break;1554case 1: op = rv_op_fcvt_s_wu; break;1555case 2: op = rv_op_fcvt_s_l; break;1556case 3: op = rv_op_fcvt_s_lu; break;1557}1558break;1559case 105:1560switch (((inst >> 20) & 0b11111)) {1561case 0: op = rv_op_fcvt_d_w; break;1562case 1: op = rv_op_fcvt_d_wu; break;1563case 2: op = rv_op_fcvt_d_l; break;1564case 3: op = rv_op_fcvt_d_lu; break;1565}1566break;1567case 106:1568switch (((inst >> 20) & 0b11111)) {1569case 0: op = rv_op_fcvt_h_w; break;1570case 1: op = rv_op_fcvt_h_wu; break;1571case 2: op = rv_op_fcvt_h_l; break;1572case 3: op = rv_op_fcvt_h_lu; break;1573}1574break;1575case 107:1576switch (((inst >> 20) & 0b11111)) {1577case 0: op = rv_op_fcvt_q_w; break;1578case 1: op = rv_op_fcvt_q_wu; break;1579case 2: op = rv_op_fcvt_q_l; break;1580case 3: op = rv_op_fcvt_q_lu; break;1581}1582break;1583case 112:1584switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1585case 0: op = rv_op_fmv_x_s; break;1586case 1: op = rv_op_fclass_s; break;1587}1588break;1589case 113:1590switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1591case 0: op = rv_op_fmv_x_d; break;1592case 1: op = rv_op_fclass_d; break;1593}1594break;1595case 114:1596switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1597case 0: op = rv_op_fmv_x_h; break;1598case 1: op = rv_op_fclass_h; break;1599}1600break;1601case 115:1602switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1603case 0: op = rv_op_fmv_x_q; break;1604case 1: op = rv_op_fclass_q; break;1605}1606break;1607case 120:1608switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1609case 0: op = rv_op_fmv_s_x; break;1610}1611break;1612case 121:1613switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1614case 0: op = rv_op_fmv_d_x; break;1615}1616break;1617case 122:1618switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1619case 0: op = rv_op_fmv_h_x; break;1620}1621break;1622case 123:1623switch (((inst >> 17) & 0b11111000) | ((inst >> 12) & 0b00000111)) {1624case 0: op = rv_op_fmv_q_x; break;1625}1626break;1627}1628break;1629case 22:1630switch (((inst >> 12) & 0b111)) {1631case 0: op = rv_op_addid; break;1632case 1:1633switch (((inst >> 26) & 0b111111)) {1634case 0: op = rv_op_sllid; break;1635}1636break;1637case 5:1638switch (((inst >> 26) & 0b111111)) {1639case 0: op = rv_op_srlid; break;1640case 16: op = rv_op_sraid; break;1641}1642break;1643}1644break;1645case 24:1646switch (((inst >> 12) & 0b111)) {1647case 0: op = rv_op_beq; break;1648case 1: op = rv_op_bne; break;1649case 4: op = rv_op_blt; break;1650case 5: op = rv_op_bge; break;1651case 6: op = rv_op_bltu; break;1652case 7: op = rv_op_bgeu; break;1653}1654break;1655case 25:1656switch (((inst >> 12) & 0b111)) {1657case 0: op = rv_op_jalr; break;1658}1659break;1660case 27: op = rv_op_jal; break;1661case 28:1662switch (((inst >> 12) & 0b111)) {1663case 0:1664switch (((inst >> 20) & 0b111111100000) | ((inst >> 7) & 0b000000011111)) {1665case 0:1666switch (((inst >> 15) & 0b1111111111)) {1667case 0: op = rv_op_ecall; break;1668case 32: op = rv_op_ebreak; break;1669case 64: op = rv_op_uret; break;1670}1671break;1672case 256:1673switch (((inst >> 20) & 0b11111)) {1674case 2:1675switch (((inst >> 15) & 0b11111)) {1676case 0: op = rv_op_sret; break;1677}1678break;1679case 4: op = rv_op_sfence_vm; break;1680case 5:1681switch (((inst >> 15) & 0b11111)) {1682case 0: op = rv_op_wfi; break;1683}1684break;1685}1686break;1687case 288: op = rv_op_sfence_vma; break;1688case 512:1689switch (((inst >> 15) & 0b1111111111)) {1690case 64: op = rv_op_hret; break;1691}1692break;1693case 768:1694switch (((inst >> 15) & 0b1111111111)) {1695case 64: op = rv_op_mret; break;1696}1697break;1698case 1952:1699switch (((inst >> 15) & 0b1111111111)) {1700case 576: op = rv_op_dret; break;1701}1702break;1703}1704break;1705case 1: op = rv_op_csrrw; break;1706case 2: op = rv_op_csrrs; break;1707case 3: op = rv_op_csrrc; break;1708case 5: op = rv_op_csrrwi; break;1709case 6: op = rv_op_csrrsi; break;1710case 7: op = rv_op_csrrci; break;1711}1712break;1713case 30:1714switch (((inst >> 22) & 0b1111111000) | ((inst >> 12) & 0b0000000111)) {1715case 0: op = rv_op_addd; break;1716case 1: op = rv_op_slld; break;1717case 5: op = rv_op_srld; break;1718case 8: op = rv_op_muld; break;1719case 12: op = rv_op_divd; break;1720case 13: op = rv_op_divud; break;1721case 14: op = rv_op_remd; break;1722case 15: op = rv_op_remud; break;1723case 256: op = rv_op_subd; break;1724case 261: op = rv_op_srad; break;1725}1726break;1727}1728break;1729}1730dec->op = op;1731}17321733/* operand extractors */17341735static uint32_t operand_rd(rv_inst inst) {1736return (inst << 52) >> 59;1737}17381739static uint32_t operand_rs1(rv_inst inst) {1740return (inst << 44) >> 59;1741}17421743static uint32_t operand_rs2(rv_inst inst) {1744return (inst << 39) >> 59;1745}17461747static uint32_t operand_rs3(rv_inst inst) {1748return (inst << 32) >> 59;1749}17501751static uint32_t operand_aq(rv_inst inst) {1752return (inst << 37) >> 63;1753}17541755static uint32_t operand_rl(rv_inst inst) {1756return (inst << 38) >> 63;1757}17581759static uint32_t operand_pred(rv_inst inst) {1760return (inst << 36) >> 60;1761}17621763static uint32_t operand_succ(rv_inst inst) {1764return (inst << 40) >> 60;1765}17661767static uint32_t operand_rm(rv_inst inst) {1768return (inst << 49) >> 61;1769}17701771static uint32_t operand_shamt5(rv_inst inst) {1772return (inst << 39) >> 59;1773}17741775static uint32_t operand_shamt6(rv_inst inst) {1776return (inst << 38) >> 58;1777}17781779static uint32_t operand_shamt7(rv_inst inst) {1780return (inst << 37) >> 57;1781}17821783static uint32_t operand_crdq(rv_inst inst) {1784return (inst << 59) >> 61;1785}17861787static uint32_t operand_crs1q(rv_inst inst) {1788return (inst << 54) >> 61;1789}17901791static uint32_t operand_crs1rdq(rv_inst inst) {1792return (inst << 54) >> 61;1793}17941795static uint32_t operand_crs2q(rv_inst inst) {1796return (inst << 59) >> 61;1797}17981799static uint32_t operand_crd(rv_inst inst) {1800return (inst << 52) >> 59;1801}18021803static uint32_t operand_crs1(rv_inst inst) {1804return (inst << 52) >> 59;1805}18061807static uint32_t operand_crs1rd(rv_inst inst) {1808return (inst << 52) >> 59;1809}18101811static uint32_t operand_crs2(rv_inst inst) {1812return (inst << 57) >> 59;1813}18141815static uint32_t operand_cimmsh5(rv_inst inst) {1816return (inst << 57) >> 59;1817}18181819static uint32_t operand_csr12(rv_inst inst) {1820return (inst << 32) >> 52;1821}18221823static int32_t operand_imm12(rv_inst inst) {1824return ((int64_t)inst << 32) >> 52;1825}18261827static int32_t operand_imm20(rv_inst inst) {1828return (((int64_t)inst << 32) >> 44) << 12;1829}18301831static int32_t operand_jimm20(rv_inst inst) {1832return (((int64_t)inst << 32) >> 63) << 20 |1833((inst << 33) >> 54) << 1 |1834((inst << 43) >> 63) << 11 |1835((inst << 44) >> 56) << 12;1836}18371838static int32_t operand_simm12(rv_inst inst) {1839return (((int64_t)inst << 32) >> 57) << 5 |1840(inst << 52) >> 59;1841}18421843static int32_t operand_sbimm12(rv_inst inst) {1844return (((int64_t)inst << 32) >> 63) << 12 |1845((inst << 33) >> 58) << 5 |1846((inst << 52) >> 60) << 1 |1847((inst << 56) >> 63) << 11;1848}18491850static uint32_t operand_cimmsh6(rv_inst inst) {1851return ((inst << 51) >> 63) << 5 |1852(inst << 57) >> 59;1853}18541855static int32_t operand_cimmi(rv_inst inst) {1856return (((int64_t)inst << 51) >> 63) << 5 |1857(inst << 57) >> 59;1858}18591860static int32_t operand_cimmui(rv_inst inst) {1861return (((int64_t)inst << 51) >> 63) << 17 |1862((inst << 57) >> 59) << 12;1863}18641865static uint32_t operand_cimmlwsp(rv_inst inst) {1866return ((inst << 51) >> 63) << 5 |1867((inst << 57) >> 61) << 2 |1868((inst << 60) >> 62) << 6;1869}18701871static uint32_t operand_cimmldsp(rv_inst inst) {1872return ((inst << 51) >> 63) << 5 |1873((inst << 57) >> 62) << 3 |1874((inst << 59) >> 61) << 6;1875}18761877static uint32_t operand_cimmlqsp(rv_inst inst) {1878return ((inst << 51) >> 63) << 5 |1879((inst << 57) >> 63) << 4 |1880((inst << 58) >> 60) << 6;1881}18821883static int32_t operand_cimm16sp(rv_inst inst) {1884return (((int64_t)inst << 51) >> 63) << 9 |1885((inst << 57) >> 63) << 4 |1886((inst << 58) >> 63) << 6 |1887((inst << 59) >> 62) << 7 |1888((inst << 61) >> 63) << 5;1889}18901891static int32_t operand_cimmj(rv_inst inst) {1892return (((int64_t)inst << 51) >> 63) << 11 |1893((inst << 52) >> 63) << 4 |1894((inst << 53) >> 62) << 8 |1895((inst << 55) >> 63) << 10 |1896((inst << 56) >> 63) << 6 |1897((inst << 57) >> 63) << 7 |1898((inst << 58) >> 61) << 1 |1899((inst << 61) >> 63) << 5;1900}19011902static int32_t operand_cimmb(rv_inst inst) {1903return (((int64_t)inst << 51) >> 63) << 8 |1904((inst << 52) >> 62) << 3 |1905((inst << 57) >> 62) << 6 |1906((inst << 59) >> 62) << 1 |1907((inst << 61) >> 63) << 5;1908}19091910static uint32_t operand_cimmswsp(rv_inst inst) {1911return ((inst << 51) >> 60) << 2 |1912((inst << 55) >> 62) << 6;1913}19141915static uint32_t operand_cimmsdsp(rv_inst inst) {1916return ((inst << 51) >> 61) << 3 |1917((inst << 54) >> 61) << 6;1918}19191920static uint32_t operand_cimmsqsp(rv_inst inst) {1921return ((inst << 51) >> 62) << 4 |1922((inst << 53) >> 60) << 6;1923}19241925static uint32_t operand_cimm4spn(rv_inst inst) {1926return ((inst << 51) >> 62) << 4 |1927((inst << 53) >> 60) << 6 |1928((inst << 57) >> 63) << 2 |1929((inst << 58) >> 63) << 3;1930}19311932static uint32_t operand_cimmw(rv_inst inst) {1933return ((inst << 51) >> 61) << 3 |1934((inst << 57) >> 63) << 2 |1935((inst << 58) >> 63) << 6;1936}19371938static uint32_t operand_cimmd(rv_inst inst) {1939return ((inst << 51) >> 61) << 3 |1940((inst << 57) >> 62) << 6;1941}19421943static uint32_t operand_cimmq(rv_inst inst) {1944return ((inst << 51) >> 62) << 4 |1945((inst << 53) >> 63) << 8 |1946((inst << 57) >> 62) << 6;1947}19481949/* decode operands */19501951static void decode_inst_operands(rv_decode *dec)1952{1953rv_inst inst = dec->inst;1954dec->codec = opcode_data[dec->op].codec;1955switch (dec->codec) {1956case rv_codec_none:1957dec->rd = dec->rs1 = dec->rs2 = rv_ireg_zero;1958dec->imm = 0;1959break;1960case rv_codec_u:1961dec->rd = operand_rd(inst);1962dec->rs1 = dec->rs2 = rv_ireg_zero;1963dec->imm = operand_imm20(inst);1964break;1965case rv_codec_uj:1966dec->rd = operand_rd(inst);1967dec->rs1 = dec->rs2 = rv_ireg_zero;1968dec->imm = operand_jimm20(inst);1969break;1970case rv_codec_i:1971dec->rd = operand_rd(inst);1972dec->rs1 = operand_rs1(inst);1973dec->rs2 = rv_ireg_zero;1974dec->imm = operand_imm12(inst);1975break;1976case rv_codec_i_sh5:1977dec->rd = operand_rd(inst);1978dec->rs1 = operand_rs1(inst);1979dec->rs2 = rv_ireg_zero;1980dec->imm = operand_shamt5(inst);1981break;1982case rv_codec_i_sh6:1983dec->rd = operand_rd(inst);1984dec->rs1 = operand_rs1(inst);1985dec->rs2 = rv_ireg_zero;1986dec->imm = operand_shamt6(inst);1987break;1988case rv_codec_i_sh7:1989dec->rd = operand_rd(inst);1990dec->rs1 = operand_rs1(inst);1991dec->rs2 = rv_ireg_zero;1992dec->imm = operand_shamt7(inst);1993break;1994case rv_codec_i_csr:1995dec->rd = operand_rd(inst);1996dec->rs1 = operand_rs1(inst);1997dec->rs2 = rv_ireg_zero;1998dec->imm = operand_csr12(inst);1999break;2000case rv_codec_s:2001dec->rd = rv_ireg_zero;2002dec->rs1 = operand_rs1(inst);2003dec->rs2 = operand_rs2(inst);2004dec->imm = operand_simm12(inst);2005break;2006case rv_codec_sb:2007dec->rd = rv_ireg_zero;2008dec->rs1 = operand_rs1(inst);2009dec->rs2 = operand_rs2(inst);2010dec->imm = operand_sbimm12(inst);2011break;2012case rv_codec_r:2013dec->rd = operand_rd(inst);2014dec->rs1 = operand_rs1(inst);2015dec->rs2 = operand_rs2(inst);2016dec->imm = 0;2017break;2018case rv_codec_r_m:2019dec->rd = operand_rd(inst);2020dec->rs1 = operand_rs1(inst);2021dec->rs2 = operand_rs2(inst);2022dec->imm = 0;2023dec->rm = operand_rm(inst);2024break;2025case rv_codec_r4_m:2026dec->rd = operand_rd(inst);2027dec->rs1 = operand_rs1(inst);2028dec->rs2 = operand_rs2(inst);2029dec->rs3 = operand_rs3(inst);2030dec->imm = 0;2031dec->rm = operand_rm(inst);2032break;2033case rv_codec_r_a:2034dec->rd = operand_rd(inst);2035dec->rs1 = operand_rs1(inst);2036dec->rs2 = operand_rs2(inst);2037dec->imm = 0;2038dec->aq = operand_aq(inst);2039dec->rl = operand_rl(inst);2040break;2041case rv_codec_r_l:2042dec->rd = operand_rd(inst);2043dec->rs1 = operand_rs1(inst);2044dec->rs2 = rv_ireg_zero;2045dec->imm = 0;2046dec->aq = operand_aq(inst);2047dec->rl = operand_rl(inst);2048break;2049case rv_codec_r_f:2050dec->rd = dec->rs1 = dec->rs2 = rv_ireg_zero;2051dec->pred = operand_pred(inst);2052dec->succ = operand_succ(inst);2053dec->imm = 0;2054break;2055case rv_codec_cb:2056dec->rd = rv_ireg_zero;2057dec->rs1 = operand_crs1q(inst) + 8;2058dec->rs2 = rv_ireg_zero;2059dec->imm = operand_cimmb(inst);2060break;2061case rv_codec_cb_imm:2062dec->rd = dec->rs1 = operand_crs1rdq(inst) + 8;2063dec->rs2 = rv_ireg_zero;2064dec->imm = operand_cimmi(inst);2065break;2066case rv_codec_cb_sh5:2067dec->rd = dec->rs1 = operand_crs1rdq(inst) + 8;2068dec->rs2 = rv_ireg_zero;2069dec->imm = operand_cimmsh5(inst);2070break;2071case rv_codec_cb_sh6:2072dec->rd = dec->rs1 = operand_crs1rdq(inst) + 8;2073dec->rs2 = rv_ireg_zero;2074dec->imm = operand_cimmsh6(inst);2075break;2076case rv_codec_ci:2077dec->rd = dec->rs1 = operand_crs1rd(inst);2078dec->rs2 = rv_ireg_zero;2079dec->imm = operand_cimmi(inst);2080break;2081case rv_codec_ci_sh5:2082dec->rd = dec->rs1 = operand_crs1rd(inst);2083dec->rs2 = rv_ireg_zero;2084dec->imm = operand_cimmsh5(inst);2085break;2086case rv_codec_ci_sh6:2087dec->rd = dec->rs1 = operand_crs1rd(inst);2088dec->rs2 = rv_ireg_zero;2089dec->imm = operand_cimmsh6(inst);2090break;2091case rv_codec_ci_16sp:2092dec->rd = rv_ireg_sp;2093dec->rs1 = rv_ireg_sp;2094dec->rs2 = rv_ireg_zero;2095dec->imm = operand_cimm16sp(inst);2096break;2097case rv_codec_ci_lwsp:2098dec->rd = operand_crd(inst);2099dec->rs1 = rv_ireg_sp;2100dec->rs2 = rv_ireg_zero;2101dec->imm = operand_cimmlwsp(inst);2102break;2103case rv_codec_ci_ldsp:2104dec->rd = operand_crd(inst);2105dec->rs1 = rv_ireg_sp;2106dec->rs2 = rv_ireg_zero;2107dec->imm = operand_cimmldsp(inst);2108break;2109case rv_codec_ci_lqsp:2110dec->rd = operand_crd(inst);2111dec->rs1 = rv_ireg_sp;2112dec->rs2 = rv_ireg_zero;2113dec->imm = operand_cimmlqsp(inst);2114break;2115case rv_codec_ci_li:2116dec->rd = operand_crd(inst);2117dec->rs1 = rv_ireg_zero;2118dec->rs2 = rv_ireg_zero;2119dec->imm = operand_cimmi(inst);2120break;2121case rv_codec_ci_lui:2122dec->rd = operand_crd(inst);2123dec->rs1 = rv_ireg_zero;2124dec->rs2 = rv_ireg_zero;2125dec->imm = operand_cimmui(inst);2126break;2127case rv_codec_ci_none:2128dec->rd = dec->rs1 = dec->rs2 = rv_ireg_zero;2129dec->imm = 0;2130break;2131case rv_codec_ciw_4spn:2132dec->rd = operand_crdq(inst) + 8;2133dec->rs1 = rv_ireg_sp;2134dec->rs2 = rv_ireg_zero;2135dec->imm = operand_cimm4spn(inst);2136break;2137case rv_codec_cj:2138dec->rd = dec->rs1 = dec->rs2 = rv_ireg_zero;2139dec->imm = operand_cimmj(inst);2140break;2141case rv_codec_cj_jal:2142dec->rd = rv_ireg_ra;2143dec->rs1 = dec->rs2 = rv_ireg_zero;2144dec->imm = operand_cimmj(inst);2145break;2146case rv_codec_cl_lw:2147dec->rd = operand_crdq(inst) + 8;2148dec->rs1 = operand_crs1q(inst) + 8;2149dec->rs2 = rv_ireg_zero;2150dec->imm = operand_cimmw(inst);2151break;2152case rv_codec_cl_ld:2153dec->rd = operand_crdq(inst) + 8;2154dec->rs1 = operand_crs1q(inst) + 8;2155dec->rs2 = rv_ireg_zero;2156dec->imm = operand_cimmd(inst);2157break;2158case rv_codec_cl_lq:2159dec->rd = operand_crdq(inst) + 8;2160dec->rs1 = operand_crs1q(inst) + 8;2161dec->rs2 = rv_ireg_zero;2162dec->imm = operand_cimmq(inst);2163break;2164case rv_codec_cr:2165dec->rd = dec->rs1 = operand_crs1rd(inst);2166dec->rs2 = operand_crs2(inst);2167dec->imm = 0;2168break;2169case rv_codec_cr_mv:2170dec->rd = operand_crd(inst);2171dec->rs1 = operand_crs2(inst);2172dec->rs2 = rv_ireg_zero;2173dec->imm = 0;2174break;2175case rv_codec_cr_jalr:2176dec->rd = rv_ireg_ra;2177dec->rs1 = operand_crs1(inst);2178dec->rs2 = rv_ireg_zero;2179dec->imm = 0;2180break;2181case rv_codec_cr_jr:2182dec->rd = rv_ireg_zero;2183dec->rs1 = operand_crs1(inst);2184dec->rs2 = rv_ireg_zero;2185dec->imm = 0;2186break;2187case rv_codec_cs:2188dec->rd = dec->rs1 = operand_crs1rdq(inst) + 8;2189dec->rs2 = operand_crs2q(inst) + 8;2190dec->imm = 0;2191break;2192case rv_codec_cs_sw:2193dec->rd = rv_ireg_zero;2194dec->rs1 = operand_crs1q(inst) + 8;2195dec->rs2 = operand_crs2q(inst) + 8;2196dec->imm = operand_cimmw(inst);2197break;2198case rv_codec_cs_sd:2199dec->rd = rv_ireg_zero;2200dec->rs1 = operand_crs1q(inst) + 8;2201dec->rs2 = operand_crs2q(inst) + 8;2202dec->imm = operand_cimmd(inst);2203break;2204case rv_codec_cs_sq:2205dec->rd = rv_ireg_zero;2206dec->rs1 = operand_crs1q(inst) + 8;2207dec->rs2 = operand_crs2q(inst) + 8;2208dec->imm = operand_cimmq(inst);2209break;2210case rv_codec_css_swsp:2211dec->rd = rv_ireg_zero;2212dec->rs1 = rv_ireg_sp;2213dec->rs2 = operand_crs2(inst);2214dec->imm = operand_cimmswsp(inst);2215break;2216case rv_codec_css_sdsp:2217dec->rd = rv_ireg_zero;2218dec->rs1 = rv_ireg_sp;2219dec->rs2 = operand_crs2(inst);2220dec->imm = operand_cimmsdsp(inst);2221break;2222case rv_codec_css_sqsp:2223dec->rd = rv_ireg_zero;2224dec->rs1 = rv_ireg_sp;2225dec->rs2 = operand_crs2(inst);2226dec->imm = operand_cimmsqsp(inst);2227break;2228};2229}22302231/* decompress instruction */22322233static void decode_inst_decompress(rv_decode *dec, rv_isa isa)2234{2235int decomp_op;2236switch (isa) {2237case rv32: decomp_op = opcode_data[dec->op].decomp_rv32; break;2238case rv64: decomp_op = opcode_data[dec->op].decomp_rv64; break;2239case rv128: decomp_op = opcode_data[dec->op].decomp_rv128; break;2240}2241if (decomp_op != rv_op_illegal) {2242if ((opcode_data[dec->op].decomp_data & rvcd_imm_nz) && dec->imm == 0) {2243dec->op = rv_op_illegal;2244} else {2245dec->op = decomp_op;2246dec->codec = opcode_data[decomp_op].codec;2247}2248}2249}22502251/* check constraint */22522253static bool check_constraints(rv_decode *dec, const rvc_constraint *c)2254{2255if (c == rvcc_last)2256return false;22572258int32_t imm = dec->imm;2259uint8_t rd = dec->rd, rs1 = dec->rs1, rs2 = dec->rs2;2260while (*c != rvc_end) {2261switch (*c) {2262case rvc_rd_eq_ra: if (!(rd == 1)) return false; break;2263case rvc_rd_eq_x0: if (!(rd == 0)) return false; break;2264case rvc_rs1_eq_x0: if (!(rs1 == 0)) return false; break;2265case rvc_rs2_eq_x0: if (!(rs2 == 0)) return false; break;2266case rvc_rs2_eq_rs1: if (!(rs2 == rs1)) return false; break;2267case rvc_rs1_eq_ra: if (!(rs1 == 1)) return false; break;2268case rvc_imm_eq_zero: if (!(imm == 0)) return false; break;2269case rvc_imm_eq_n1: if (!(imm == -1)) return false; break;2270case rvc_imm_eq_p1: if (!(imm == 1)) return false; break;2271case rvc_csr_eq_0x001: if (!(imm == 0x001)) return false; break;2272case rvc_csr_eq_0x002: if (!(imm == 0x002)) return false; break;2273case rvc_csr_eq_0x003: if (!(imm == 0x003)) return false; break;2274case rvc_csr_eq_0xc00: if (!(imm == 0xc00)) return false; break;2275case rvc_csr_eq_0xc01: if (!(imm == 0xc01)) return false; break;2276case rvc_csr_eq_0xc02: if (!(imm == 0xc02)) return false; break;2277case rvc_csr_eq_0xc80: if (!(imm == 0xc80)) return false; break;2278case rvc_csr_eq_0xc81: if (!(imm == 0xc81)) return false; break;2279case rvc_csr_eq_0xc82: if (!(imm == 0xc82)) return false; break;2280default: break;2281}2282c++;2283}2284return true;2285}22862287/* lift instruction to pseudo-instruction */22882289static void decode_inst_lift_pseudo(rv_decode *dec)2290{2291const rv_comp_data *comp_data = opcode_data[dec->op].pseudo;2292if (!comp_data) {2293return;2294}2295while (comp_data->constraints) {2296if (check_constraints(dec, comp_data->constraints)) {2297dec->op = comp_data->op;2298dec->codec = opcode_data[dec->op].codec;2299return;2300}2301comp_data++;2302}2303}23042305/* format instruction */23062307static char *append(char *buf, const char *src, const char *end)2308{2309while (buf < end && *src)2310*buf++ = *src++;2311return buf;2312}23132314#define INST_FMT_2 "%04" PRIx64 " "2315#define INST_FMT_4 "%08" PRIx64 " "2316#define INST_FMT_6 "%012" PRIx64 " "2317#define INST_FMT_8 "%016" PRIx64 " "23182319static void decode_inst_format(char *buf, size_t buflen, size_t tab, rv_decode *dec)2320{2321char tmp[64];2322const char *fmt;2323const char *start = buf;2324const char *end = &buf[buflen];23252326size_t len = riscv_inst_length(dec->inst);2327switch (len) {2328case 2:2329buf += snprintf(buf, buflen, INST_FMT_2, dec->inst);2330break;2331case 4:2332buf += snprintf(buf, buflen, INST_FMT_4, dec->inst);2333break;2334case 6:2335buf += snprintf(buf, buflen, INST_FMT_6, dec->inst);2336break;2337default:2338buf += snprintf(buf, buflen, INST_FMT_8, dec->inst);2339break;2340}23412342fmt = opcode_data[dec->op].format;2343while (*fmt) {2344switch (*fmt) {2345case 'O':2346buf = append(buf, opcode_data[dec->op].name, end);2347break;2348case '(':2349buf = append(buf, "(", end);2350break;2351case ',':2352buf = append(buf, ",", end);2353break;2354case ')':2355buf = append(buf, ")", end);2356break;2357case '0':2358buf = append(buf, rv_ireg_name_sym[dec->rd], end);2359break;2360case '1':2361buf = append(buf, rv_ireg_name_sym[dec->rs1], end);2362break;2363case '2':2364buf = append(buf, rv_ireg_name_sym[dec->rs2], end);2365break;2366case '3':2367buf = append(buf, rv_freg_name_sym[dec->rd], end);2368break;2369case '4':2370buf = append(buf, rv_freg_name_sym[dec->rs1], end);2371break;2372case '5':2373buf = append(buf, rv_freg_name_sym[dec->rs2], end);2374break;2375case '6':2376buf = append(buf, rv_freg_name_sym[dec->rs3], end);2377break;2378case '7':2379snprintf(tmp, sizeof(tmp), "%d", dec->rs1);2380buf = append(buf, tmp, end);2381break;2382case 'i':2383snprintf(tmp, sizeof(tmp), "%d", dec->imm);2384buf = append(buf, tmp, end);2385break;2386case 'o':2387snprintf(tmp, sizeof(tmp), "%d", dec->imm);2388buf = append(buf, tmp, end);2389while (buf < &start[2*tab]) {2390*buf++ = ' ';2391}2392snprintf(tmp, sizeof(tmp), "# 0x%" PRIx64,2393dec->pc + dec->imm);2394buf = append(buf, tmp, end);2395break;2396case 'c': {2397const char *name = csr_name(dec->imm & 0xfff);2398if (name) {2399buf = append(buf, name, end);2400} else {2401snprintf(tmp, sizeof(tmp), "0x%03x", dec->imm & 0xfff);2402buf = append(buf, tmp, end);2403}2404break;2405}2406case 'r':2407switch (dec->rm) {2408case rv_rm_rne:2409buf = append(buf, "rne", end);2410break;2411case rv_rm_rtz:2412buf = append(buf, "rtz", end);2413break;2414case rv_rm_rdn:2415buf = append(buf, "rdn", end);2416break;2417case rv_rm_rup:2418buf = append(buf, "rup", end);2419break;2420case rv_rm_rmm:2421buf = append(buf, "rmm", end);2422break;2423case rv_rm_dyn:2424buf = append(buf, "dyn", end);2425break;2426default:2427buf = append(buf, "inv", end);2428break;2429}2430break;2431case 'p':2432if (dec->pred & rv_fence_i) {2433buf = append(buf, "i", end);2434}2435if (dec->pred & rv_fence_o) {2436buf = append(buf, "o", end);2437}2438if (dec->pred & rv_fence_r) {2439buf = append(buf, "r", end);2440}2441if (dec->pred & rv_fence_w) {2442buf = append(buf, "w", end);2443}2444break;2445case 's':2446if (dec->succ & rv_fence_i) {2447buf = append(buf, "i", end);2448}2449if (dec->succ & rv_fence_o) {2450buf = append(buf, "o", end);2451}2452if (dec->succ & rv_fence_r) {2453buf = append(buf, "r", end);2454}2455if (dec->succ & rv_fence_w) {2456buf = append(buf, "w", end);2457}2458break;2459case '\t':2460while (buf < &start[tab]) {2461*buf++ = ' ';2462}2463break;2464case 'A':2465if (dec->aq) {2466buf = append(buf, ".aq", end);2467}2468break;2469case 'R':2470if (dec->rl) {2471buf = append(buf, ".rl", end);2472}2473break;2474default:2475break;2476}2477fmt++;2478}2479*buf = '\0';2480}24812482/* instruction length */24832484size_t riscv_inst_length(rv_inst inst)2485{2486/* NOTE: supports maximum instruction size of 64-bits */24872488/* instruction length coding2489*2490* aa - 16 bit aa != 112491* bbb11 - 32 bit bbb != 1112492* 011111 - 48 bit2493* 0111111 - 64 bit2494*/24952496return (inst & 0b11) != 0b11 ? 22497: (inst & 0b11100) != 0b11100 ? 42498: (inst & 0b111111) == 0b011111 ? 62499: (inst & 0b1111111) == 0b0111111 ? 82500: 0;2501}25022503/* instruction fetch */25042505void riscv_inst_fetch(const uint8_t *data, rv_inst *instp, size_t *length)2506{2507rv_inst inst = ((rv_inst)data[1] << 8) | ((rv_inst)data[0]);2508size_t len = *length = riscv_inst_length(inst);2509if (len >= 8) inst |= ((rv_inst)data[7] << 56) | ((rv_inst)data[6] << 48);2510if (len >= 6) inst |= ((rv_inst)data[5] << 40) | ((rv_inst)data[4] << 32);2511if (len >= 4) inst |= ((rv_inst)data[3] << 24) | ((rv_inst)data[2] << 16);2512*instp = inst;2513}25142515/* disassemble instruction */25162517void riscv_disasm_inst(char *buf, size_t buflen, rv_isa isa, uint64_t pc, rv_inst inst)2518{2519rv_decode dec{};2520dec.pc = pc;2521dec.inst = inst;2522decode_inst_opcode(&dec, isa);2523decode_inst_operands(&dec);2524decode_inst_decompress(&dec, isa);2525decode_inst_lift_pseudo(&dec);2526decode_inst_format(buf, buflen, 32, &dec);2527}252825292530