详解Cairo指令

avatar
Sin7y
1年前
本文约1476字,阅读全文需要约2分钟
本文从多个角度对Cairo指令进行了详细解读,涉及状态转换、转换逻辑、指令校验、指令示例等。

1. 指令结构

Cairo CPU原生支持的word是一个域元素,而这个域是特征值大于P >2^63。每个指令会占用1到2个word,如果指令后面跟着立即值([ap]=12345678)则该指令占用2个word,并且值存在第二个word里。每个指令的第一个word由以下元素组成:

详解Cairo指令

详解Cairo指令

详解Cairo指令

2. 状态转换

状态转换函数代表了一个通用的状态转换单元(由于它包含了所有指令类型的处理逻辑),而一个计算通常会分解成多个连续执行的指令,因此我们需要:

a. 确保指令的内容,以及指令执行前后的状态的有效性(比如满足一定的范围校验,和状态一致性校验)

b. 确保执行的指令是一个有效的指令

2.1 转换逻辑

如果指令执行前后的状态是一致的,那么其状态的更新一定是按照以下逻辑执行:

详解Cairo指令

详解Cairo指令

详解Cairo指令

详解Cairo指令

2.2 指令校验

如图1所示,⼀个指令由以下元素组成:

详解Cairo指令

详解Cairo指令

3. 指令示例

3.1断言相等

断言相等指令可以用下述语法表示:

< left_handle_op > = < right_handle_op >

它确保了公式两边是相等的,否则程序的执行将会被返回。

详解Cairo指令

Note2:除法和减法可以分别表示为具有不同操作数顺序的乘法和加法。 

assert 指令可以被认为是一条赋值指令,其中⼀边是已知的,另一边是未知的。例如 [ap]=4 可以被认为是断言的 [ap] 值为4,或者根据上下文将[ap]赋值为4。

图4给出了断言相等指令的一些示例,以及每个指令对应的标志值:

详解Cairo指令

解释 指令 [fp + 1] = 5:

◦ 为assert指令 => opcode = 4 

◦ next_ap = ap => ap_update = 00 = 0 

◦ next_pc = pc + instruction_size => pc_update = 000 = 0 

◦ op0和op1没有add or mul => res_logic(res) = 00 = 0 

◦ 存在立即数 => op1_src(op1) = 001 = 1 

◦ 立即数地址指令地址相邻 => off_op1 = 1 

◦ 等式左边[fp + 1] => dst_reg(dst) = 1 

◦ 等式左边[fp + 1] => off_dst = 1 

◦ op0_reg/ off_op0 => inital value(1/-1) //因为这个指令用不到这些flags,所以填充默认值

3.2 条件和非条件跳转 

jmp 指令允许更改程序计数器 pc 的值。 

Cairo支持相对跳转(其操作数代表相对当前pc的偏移)和绝对跳转 - 分别用关键字rel和abs表示;jmp指令或许是有条件的,比如当某个内存单元的值不为0时,触发jmp指令。

指令的语法如下所示: 

# Unconditional jumps. 

jmp abs <adress>

jmp rel  <offset>

# Conditional jumps. 

jmp rel <offset> if <op>! 

图5给出了 jmp 指令的一些示例,以及每个指令对应的标志值:

详解Cairo指令

解释 指令 jmp rel [ap +1] + [fp - 7]: 

◦ 为jmp指令 => opcode = 0

◦ next_ap = ap => ap_update = b00 = 0 

◦ next_pc = pc + res=> pc_update = b010 = 2 

◦ res = op0 + op1 => res_logic(res) = b01 = 1

◦ op1: [fp - 7] => op1_src(op1) = b010 = 2 

◦ op1: [fp - 7] => off_op1 = -7 

◦ op0: [ap + 1] => op0_src(op0) = 0 

◦ op0: [ap + 1] => off_op0 = 1 

◦ dst_reg/ off_dst => inital value(1/-1) ///因为这个指令用不到这些flags,所以填充默认值

3.3 call 和 ret 

call和ret指令允许实现函数堆栈。 call指令更新程序计数器(pc)和帧指针(fp)寄存器。程序计数器的更新类似于 jmp 指令。之前 fp 的值被写入[ap] ,以允许 ret 指令将 fp 的值重置为调用之前的值;类似地,返回的pc (调用指令后面指令的地址)被写到 [ap+1] ,以允许 ret 指令跳回并继续执行调用指令后面的代码的执行。由于写入了两个存储单元,ap 向前进了2,fp 被设置为新的 ap。

指令的语法如下:

call ret <adress>

call rel <offset> 

ret

图6给出了 call 和 ret 指令的一些示例,以及每个指令对应的标志值:

详解Cairo指令

解释 指令 call abs [fp + 4]:

◦ 为call指令 => opcode = 0 

◦ next_ap = ap => ap_update = b00 = 0 

◦ next_pc = res => pc_update = b001 = 1 

◦ res = op1 => res_logic(res) = b00 = 0 

◦ op1: [fp + 4] => op1_src(op1) = b010 = 2 

◦ op1: [fp + 4] => off_op1 = 4 

◦ op0_reg/ off_op0 => inital value(0/1) ///因为这个指令用不到这些flags,所以填充默认值 

◦ dst_reg/ off_dst => inital value(0/0) ///因为这个指令用不到这些flags,所以填充默认值

3.4 高级 ap 

指令 ap + = < op > 通过给定的操作数增加 ap 的值。 

图7给出了高级 ap 指令的一些示,以及每个指令对应的标志:

详解Cairo指令

解释 指令 ap += 123: 

◦ 为advancing ap指令 => opcode = 0 

◦ next_ap = ap + res => ap_update = b01 = 1

◦ next_pc = pc + instruction_size => pc_update = b000 = 0 

◦ res = op1 => res_logic(res) = b00 = 0 

◦ op1 = 123 => op1_src(op1) = b001 = 1 

◦ op1 = 123 => off_op1 = 1 

◦ op0_reg/ off_op0 => inital value(1/-1) ///因为这个指令用不到这些flags,所以填充默认值 

◦ dst_reg/ off_dst => inital value(1/-1) ///因为这个指令用不到这些flags,所以填充默认

关于我们

Sin7y成立于2021年,由顶尖的区块链开发者组成。我们既是项目孵化器也是区块链技术研究团队,探索EVM、Layer2跨链隐私计算、自主支付解决方案等最重要和最前沿的技术。

微信公众号:Sin7Y

GitHub | Twitter | Telegram | MediumMirror | HackMD | HackerNoon

本文来自投稿,不代表Odaily立场。如若转载请注明出处。

ODAILY提醒,请广大读者树立正确的货币观念和投资理念,理性看待区块链,切实提高风险意识;对发现的违法犯罪线索,可积极向有关部门举报反映。

推荐阅读
星球精选