From b07e09d645092a8d047968049338664ae8cb50ce Mon Sep 17 00:00:00 2001 From: WANG Xuerui Date: Fri, 6 May 2022 17:18:54 +0800 Subject: [PATCH] Add the LoongArch instruction format conventions doc. Add clarification that manual assembly syntax is not affected by this doc. Explain why FCSR operands is treated as immediates in this doc. Many wording tweaks. --- ...rch-instruction-format-conventions-CN.adoc | 452 ++++++++++++++++++ docs/README-CN.adoc | 3 + 2 files changed, 455 insertions(+) create mode 100644 docs/LoongArch-instruction-format-conventions-CN.adoc diff --git a/docs/LoongArch-instruction-format-conventions-CN.adoc b/docs/LoongArch-instruction-format-conventions-CN.adoc new file mode 100644 index 0000000..b5d22b1 --- /dev/null +++ b/docs/LoongArch-instruction-format-conventions-CN.adoc @@ -0,0 +1,452 @@ += LoongArch 指令格式约定 +LoongArch 社区开发者一同 +v1.00 +:docinfodir: ../themes +:docinfo: shared +:doctype: book +:toc: left +:toc-title: 目录 +:scripts: cjk + +== 背景 + +作为指令集生态的重要组成部分,许多编译器、虚拟机运行时等相关项目都有处理 +LoongArch 机器语言的需求,进而有必要感知 LoongArch 的具体指令编码。 +但是,目前《龙芯架构参考手册》v1.00 所规定的“9 种基本指令格式”(下称“**手册名**”)无法完全满足需求: + +- 一些指令不属于 9 种格式中的任何一种,如 `lu12i.w`、`asrt{le,gt}.d`、`bstrpick.[wd]`。 + 其中甚至包括一些最常用的指令,如 `lu12i.w`。相关项目必须自行发明一些新名字用来称呼这些非标准格式, + 进而造成命名上的不统一、跨项目无法共享知识。 +- **手册名**都以数字打头(如 `1RI21`),导致这些名字无法被用作绝大多数编程语言的标识符。 + 相关项目仍然需要发明一些新名字,或以 `FMT` 之类字样作为前缀,有失冗长。 +- **手册名**较为写意,不反映具体编码细节。这对需要处理这些细节的软件、开发者而言不够直白。 + 例如同为单寄存器操作数、单立即数的格式,`1RI21` 和所谓“`1RI20`”(`lu12i.w` 指令的真正格式)在编码细节上完全不同。 + 这些信息不能直白地从命名中看出,因此即使有经验的开发者也要付出一层额外理解成本。 + +我们认为,尽管作为编译器、虚拟机开发者,熟读参考手册自然属于必备素养, +但在合适的场合尽量降低理解与沟通成本,提升开发者体验,仍然是 LoongArch 作为一个新时代、对开发者友好的指令集架构的题中应有之义。 + +本文将对 LoongArch 的指令格式作出补充、细化约定,意在实现 LoongArch +指令编码信息的标准化描述、机器可读性、可处理性, +降低维护人员心智负担,以及提倡跨项目的知识共享。 + + +== 约定 + +=== 设计思想 + +一个好的指令格式命名方案,应当满足: + +* 指令格式名称应当在多数编程语言都是合法标识符。 ++ +这是为了方便下游项目原样使用。 +如果指令格式名称不能直接作标识符使用,下游项目则不得不各自发明一些新名字。 +这会导致不同项目使用的名称不同,不利于基础软件开发者学习掌握。 + +* 指令格式名称最好不需要严格区分大小写。 ++ +这是为了适应不同项目、编程语言的编码风格要求。 +例如一些项目、语言要求常量、枚举值名称必须全大写;或如 Go 语言,标识符的首字母大小写会改变语义。 +如果存在仅能通过大小写不同区分的指令格式名称,在此种情形下,下游项目则也需要发明一些新名字,带来的坏处同前一种情形。 + +* 指令格式名称应当尽量精确表意。 ++ +不同项目对指令格式划分精确与否的需求不同,但从精确表意的名字中去除信息,比向粗略表意的名字添加信息要方便得多。 +具体而言: ++ +** 相同操作数个数,但种类不同的指令,应当归属不同的指令格式。 +** 相同操作数个数、种类,但操作数大小和/或布局不同的指令,应当归属不同的指令格式。 + +* 指令格式名称最好具有前向兼容性。 ++ +对于有意迅速迭代发展的指令集架构而言,添加新的指令乃至指令格式应被视作一件经常会发生的事。 +因而,作为下游项目的维护者,会希望各个下游的代码不需太多调整即可支持新指令; +作为规范制定者,会希望不需要每次发布指令集的新版本,都要对相关规范作大范围修改; +作为指令集规范的读者与用户,也会希望每次新增指令之后,记忆负担能尽量小,尽量可用已有知识辅助新内容的记忆。 + +下文将“按照本约定书写的指令格式名称”称作“**规范名**”。 + +[NOTE] +==== +本文档为“**手册名**”、“**规范名**”两个词组赋予的特定含义,仅仅用来方便本文档之内的描述。 +这种用法不应“泄漏”到本文档之外。 + +在本文档成为 LoongArch 的正式规范文档之后,我们预期“**手册名**”将渐渐淡出历史舞台。 +对规范读者群体而言,直接称呼“指令格式”、“指令格式名”就相当于本文中所说的“**规范名**”了。 +==== + +[NOTE] +==== +本约定只覆盖 LoongArch 机器语言(指令字)的格式描述,而不覆盖 LoongArch 汇编语言。 + +目前 LoongArch 汇编语法中,也存在书写顺序与本约定不一致(如下文有讨论的 `bstrins/bstrpick` 等指令),以及一些需要额外记忆的情形(如有些指令带的立即数,按照手册语法,汇编写法与指令位域的内容不同,存在 `+1`、`<<2` 等特殊处理),本约定不予以感知、处理。 +这些情形将在其他文档进行规范。 +==== + +=== 形式化描述 + +**规范名**的语法遵循以下 ABNF 描述: + +``` +insn-format = "EMPTY" +insn-format =/ reg-slots +insn-format =/ imm-slots +insn-format =/ reg-slots imm-slots + +reg-slots = 1*reg +reg = int-reg / fp-reg / fcc-reg +int-reg = "D" / "J" / "K" / "A" +fp-reg = "F" index +fcc-reg = "C" index + +index-length = index length +index = "d" / "j" / "k" / "a" / "m" +length = 1*DIGIT + +imm-slots = 1*imm +imm = signedness 1*index-length +signedness = "S" / "U" +``` + +=== 自然语言描述 + +每条 LoongArch 指令定长 32 位,我们通过**规范名**描述的是每条指令中为操作数挖出的槽。 +那些没有对应到槽位的位(比特)都属于指令字中固定的、为硬件译码单元所识别的操作码部分。 +规定位序号从 0 开始,序号为 0 的位为最低位(LSB)。 + +有些指令没有操作数槽,其所有操作对象都由指令语义隐含。 +人为规定不带操作数槽的**规范名**为 `EMPTY` 以避免空字符串带来的不便。 + +操作数分为寄存器操作数、立即数操作数两类。 +根据具体指令格式不同,寄存器操作数标记的是某个寄存器类别(register bank)中的一个寄存器编号; +立即数操作数也存在有符号、无符号之分。 + +为求表示形式统一、易处理,规定**规范名**中的操作数槽书写顺序为: + +* 先寄存器槽后立即数槽, +* 每组内从低位到高位。 + +对于寄存器操作数,为缩减常用指令的**规范名**长度,通用寄存器都有单字母的名字,与汇编中常用的 `rd rj rk ra` 占位符保持一致。 +其他类别的寄存器操作数都以形如 `类别 位置标记` 的方式指定,例如 `Fj` 意为占据 `j` 位置(具体含义见下文)的浮点寄存器。 + +.寄存器操作数槽位 +[%header,cols="^1,2,^1,2"] +|=== +|槽位名称 +|起始位序号 +|位域宽度 +|寄存器类别 + +|`D` +|0 +|5 +|通用寄存器 + +|`J` +|5 +|5 +|通用寄存器 + +|`K` +|10 +|5 +|通用寄存器 + +|`A` +|15 +|5 +|通用寄存器 + +|`C` +|由位置标记指定 +|3 +|浮点条件码寄存器 + +|`F` +|由位置标记指定 +|5 +|浮点寄存器 +|=== + +为节省大写字母以便未来使用,本约定不将浮点控制状态寄存器(FCSR)操作数视作一类寄存器, +而是将其视作用来寻址 FCSR 地址空间的**立即数**;一如控制状态寄存器(CSR)操作指令的立即数被看作对 CSR 地址空间的寻址。 + +[NOTE] +==== +手册将 FCSR 视为单独一类寄存器。 +但正如其名,FCSR 无论在理解上还是使用上都与 CSR 更为相似:并不会因为存在 FCSR0 ~ FCSR3 四个名字, +就意味着硬件上存在四个真实的寄存器。 +实际上 FCSR1 ~ FCSR3 也确实只是 FCSR0 的个别位域的“视图”。 +其次,FCSR 也不像 GPR、FPR、FCC 一样可以参与寄存器分配,因此将其强行与其他寄存器类别相提并论,并无太大意义。 + +此外,由于历史原因,binutils、LLVM 等一些项目将 FCSR 操作数实现成了 GPR(仅汇编语法意义上;并不是在 FCSR 位置写了 `$r0` 就是访问真正的 `$zero`,实际访问的还是 FCSR0),这更是错误的。 +比起建立一个新的不能当寄存器用的“寄存器类别”,还不如将其视作立即数来得正确、方便。 +==== + +对于非通用寄存器类型的寄存器操作数,以及立即数操作数,需要指定其位域的起始位编号。 +为方便记忆,对于那些与各个通用寄存器起始位重合的位置,以表示相应通用寄存器操作数的字母的小写形式为相应的位置标记。 + +[NOTE] +==== +由于**规范名**语法的前缀构造,复用这些字母不会造成歧义,且即使指令格式名称中的所有字母都变为大写或小写,也不会有歧义。 + +**规范名**仍然同时使用大小写字母,是为了可读性(除 EMPTY 外,名称含有多少大写字母,相应的格式就有多少操作数)。 +==== + +在 LoongArch 基础指令中,存在从第 16 位开始的立即数操作数,没有寄存器操作数从该位开始。 +为方便记忆,人为规定字母 `m` (middle)为对应的位置标记。 + +.位置标记 +[%header,cols="^1,1"] +|=== +|位置标记名 +|代表的位序号 + +|`d` +|0 + +|`j` +|5 + +|`k` +|10 + +|`a` +|15 + +|`m` +|16 +|=== + +对于立即数操作数,有些指令将立即数域的内容视作有符号数,有些则视作无符号数。 +立即数操作数的表示格式为 `符号 1或多个立即数槽`,以 `S` 代表有符号(signed)立即数、 `U` 代表无符号(unsigned)立即数。 +每个立即数槽的表示格式为 `位置标记 宽度`,宽度即为十进制阿拉伯数字。 + +由多个槽组成的立即数操作数所表示的数值,是将组成它的每个槽按顺序从高到低排列之后,将其内容连接而得到的值。 +例如 `Sd5k16` 这个立即数有两个槽 `d5` 与 `k16`,其表示的数值为 `(d5 << 16) | k16`。 + +[NOTE] +==== +来自 RISC-V 背景的读者需要注意:LoongArch 保留了逻辑操作指令立即数无符号的传统,这一点与 RISC-V 不同。 +(否则只需单个字母如 `I` 即可表示所有立即数了。) +==== + +=== 示例与讨论 + +在实际工程实践中,编译器、汇编器、反汇编器等组件经常需要区分不同的寄存器类别。 +**手册名**不作该区分,而使用**规范名**则可方便区分。 + +.“相同格式”,但寄存器类别不同的指令示例 +[%header,cols="^1,^1,^1"] +|=== +|指令名 +|**手册名** +|**规范名** + +|`rdtime` +|2R +|DJ + +|`movgr2fr.w` +|2R +|FdJ + +|`movfr2gr.s` +|2R +|DFj + +|`add.d` +|3R +|DJK + +|`fadd.d` +|3R +|FdFjFk + +|`ld.w` +|2RI12 +|DJSk12 + +|`fld.s` +|2RI12 +|FdJSk12 +|=== + +在处理机器语言时,操作数位置、形状不同,指令格式显然不同;并非汇编语法相似,指令编码就一定相似。 +**手册名**不考虑具体编码方式,而使用**规范名**则能精确区分。 + +.“相同格式”,但具体编码不同的指令示例 +[%header,cols="^1,^1,^1"] +|=== +|指令名 +|**手册名** +|**规范名** + +|`clo.w` +|2R +|DJ + +|`asrtgt.d` +|无(2R?3R 变体?) +|JK + +|`movgr2cf` +|2R +|CdJ + +|`movcf2gr` +|2R +|DCj + +|`lu12i.w` +|无(“1RI20”?1RI21 变体?) +|DSj20 + +|`beqz` +|1RI21 +|JSd5k16 +|=== + +可见 `asrtle/asrtgt`、`lu12i.w` 等指令的格式其实相当特殊,和其他一些指令实际一点关系也没有,尽管那些指令的格式的**手册名**也是“2R”、“1RI21”。 + +在实现汇编器、JIT 等组件时,经常需要在生成机器语言前,确保指令的立即数都不溢出相应位域。此时需要感知立即数是否有符号。 +**手册名**不作该区分,用**规范名**则可实现。 + +.“相同格式”,但立即数符号不同的指令示例 +[%header,cols="^1,^1,^1"] +|=== +|指令名 +|**手册名** +|**规范名** + +|`addi.w` +|2RI12 +|DJSk12 + +|`ori` +|2RI12 +|DJUk12 +|=== + +最后,由于严格遵循“从低位到高位”的描述原则,有时透过**规范名**,可以窥见 LoongArch 个别一些被手册语法的不对称、不一致所掩盖的深层逻辑。 + +.被手册语法掩盖的设计思路:整数大小比较 +[%header,cols="^1,^1,^2,^1"] +|=== +|指令名 +|**规范名** +|手册语法 +|“手册规范名” + +|`blt` +|DJSk16 +|`blt rj, rd, offs16` +|*JDSk16 + +|`bge` +|DJSk16 +|`bge rj, rd, offs16` +|*JDSk16 + +|`asrtgt.d` +|JK +|`asrtgt.d rj, rk` +|JK + +|`asrtle.d` +|JK +|`asrtle.d rj, rk` +|JK + +|`ldgt.d` +|DJK +|`ldgt.d rd, rj, rk` +|DJK + +|`ldle.d` +|DJK +|`ldle.d rd, rj, rk` +|DJK +|=== + +“手册规范名”是假设直接将手册语法转写为**规范名**,而得到的不一定合法的“规范名”。 +前缀的星号代表相应的“规范名”不合法。 + +我们于是可以发现,LoongArch 的整数大小比较操作其实是统一按“大于、小于等于”划分的。 +条件分支指令 `b{lt/ge}[u]` 实际是交换了被比较操作数顺序的 `b{gt/le}[u]`。 +这样交换之后,所有涉及整数大小比较的指令,其比较方式就都一致了。 + +.被手册语法掩盖的设计思路:位域操作指令的立即数含义 +[%header,cols="^1,^1,^2,^1"] +|=== +|指令名 +|**规范名** +|手册语法 +|“手册规范名” + +|`bstrins.w` +|DJUk5Um5 +|`bstrins.w rd, rj, msbw, lsbw` +|*DJUm5Uk5 + +|`bstrpick.w` +|DJUk5Um5 +|`bstrpick.w rd, rj, msbw, lsbw` +|*DJUm5Uk5 +|=== + +`bstrins/bstrpick` 系列指令有两个立即数,但与其他指令不同,写在前面的立即数实际是较高的那个。 +两个立即数分别代表被操作的位域起止位序号(闭区间),顺序不能反,否则行为不确定。 +那么哪个是高位哪个是低位呢? + +答案其实很简单:较高位置的立即数代表位域最高位(含)序号,较低位置的则代表最低位(含)序号。 +手册汇编语法刻意将两个立即数位置颠倒,可能是为了使汇编形状更像自然语言,如 `rd = rj[msbw:lsbw]`。 +但这样做相当于引入了一组需要记忆的特例,其实不见得有必要。 + +== 附:**手册名**与**规范名**的对应关系 + +为方便查阅、学习,此处将**手册名**与**规范名**的对应关系整理为表格。 + +.**手册名**与**规范名**的对应关系 +[%header,cols="^1,^2,4"] +|=== +|**手册名** +|**规范名** +|备注 + +|2R +|DJ、FdFj 等等 +|**手册名**不区分寄存器类别 + +|3R +|DJK、FdFjFk 等等 +|同上 + +|4R +|FdFjFkFa、FdFjFkCa +|同上 + +|2RI8 +|DJUk8 +|仅 `lddir` ( `ldpte` 是少一个寄存器操作数的变体 JUk8) + +|2RI12 +|DJSk12、DJUk12、FdJSk12 等等 +|多数带单个立即数的指令 + +|2RI14 +|DJSk14、DJUk14 +|仅 `ldptr` `stptr` 系列及 `csrxchg` + +|2RI16 +|DJSk16 +|条件分支指令、 `jirl` + +|1RI21 +|JSd5k16、CjSd5k16 +|`beqz/bnez` 、 `bceqz/bcnez` ,比其他条件分支指令省了一个寄存器操作数 + +|I26 +|Sd10k16 +|`b` `bl` ,无条件跳转不需要(或隐含寻址了)寄存器操作数 +|=== diff --git a/docs/README-CN.adoc b/docs/README-CN.adoc index 9f62768..3678fd2 100644 --- a/docs/README-CN.adoc +++ b/docs/README-CN.adoc @@ -46,6 +46,9 @@ * 龙芯架构 SMBIOS 规范:该文档定义了龙芯架构处理器附加信息,是 SMBIOS 结构 type 44 的补充。本文档仅提供 *英文版*。 ** link:LoongArch-Processor-SMBIOS-Spec-EN.html[HTML 版本]。 +* 龙芯架构指令格式约定:该文档面向需要处理 LoongArch 机器语言的项目、开发者,介绍了龙芯架构指令格式、操作数槽的统一命名约定。 +** link:LoongArch-instruction-format-conventions-CN.html[HTML 版本]。 +** link:LoongArch-instruction-format-conventions-CN.pdf[PDF 版本]。 [[getting-start]] == 开始