原定的计划中这一篇应当是要讲如何编译if表达式的,可是我发现没什么东西能够做为if的test-form的部分的表达式,因此以为,要不仍是先实现一下比较两个数字这样子的功能吧。说干就干,我决定用大于运算符来做为例子——大于运算符就是指>
啦。因此,个人目标是要编译下面这样的代码git
(> 1 2)
而且比较以后的结果要放在EAX
寄存器中。鉴于如今这门语言还很是地简陋,没有布尔类型这样子的东西,因此在此仿照C语言的处置方式,以数值0表示逻辑假,其它的值表示逻辑真。因此上面的表达式在编译成汇编代码并最终运行后,应当能够看到EAX
寄存器中的值为0。github
为了编译大于运算符,而且将结果放入到EAX
寄存器中,须要用到新的指令CMP
、JG
,以及JMP
了。个人想法是,先将第一个操做数放入到EAX
寄存器,将第二个操做数放入到EBX
寄存器。而后,使用CMP
指令比较这两个寄存器。若是EAX
中的数值大于EBX
,那么就使用JG
指令跳到一个MOV
指令上,这道MOV
会将寄存器EAX
的值修改成1;不然,JG
不被执行,执行后续的一道MOV
指令,将数值0写入到EAX
寄存器,而后使用JMP
跳走,避免又执行到了刚才的第一道MOV
指令。思路仍是挺简单的。app
在修改jjcc2
以前,还须要在inside-out/aux
中对>
予以支持,但没什么特别的,就是往member
的参数中加入>
这个符号而已。以后,将jjcc2
改成以下的形式ide
(defun jjcc2 (expr globals) "支持两个数的四则运算的编译器" (check-type globals hash-table) (cond ((eq (first expr) '+) `((movl ,(get-operand expr 0) %eax) (movl ,(get-operand expr 1) %ebx) (addl %ebx %eax))) ((eq (first expr) '-) `((movl ,(get-operand expr 0) %eax) (movl ,(get-operand expr 1) %ebx) (subl %ebx %eax))) ((eq (first expr) '*) ;; 将两个数字相乘的结果放到第二个操做数所在的寄存器中 ;; 由于约定了用EAX寄存器做为存放最终结果给continuation用的寄存器,因此第二个操做数应当为EAX `((movl ,(get-operand expr 0) %eax) (movl ,(get-operand expr 1) %ebx) (imull %ebx %eax))) ((eq (first expr) '/) `((movl ,(get-operand expr 0) %eax) (cltd) (movl ,(get-operand expr 1) %ebx) (idivl %ebx))) ((eq (first expr) 'progn) (let ((result '())) (dolist (expr (rest expr)) (setf result (append result (jjcc2 expr globals)))) result)) ((eq (first expr) 'setq) ;; 编译赋值语句的方式比较简单,就是将被赋值的符号视为一个全局变量,而后将eax寄存器中的内容移动到这里面去 ;; TODO: 这里expr的second的结果必须是一个符号才行 ;; FIXME: 不知道应该赋值什么比较好,先随便写个0吧 (setf (gethash (second expr) globals) 0) (values (append (jjcc2 (third expr) globals) ;; 为了方便stringify函数的实现,这里直接构造出RIP-relative形式的字符串 `((movl %eax ,(get-operand expr 0)))) globals)) ((eq (first expr) '_exit) ;; 由于知道_exit只须要一个参数,因此将它的第一个操做数塞到EDI寄存器里面就能够了 ;; TODO: 更好的写法,应该是有一个单独的函数来处理这种参数传递的事情(以符合calling convention的方式) `((movl ,(get-operand expr 0) %edi) (movl #x2000001 %eax) (syscall))) ((eq (first expr) '>) ;; 为了能够把比较以后的结果放入到EAX寄存器中,以我目前不完整的汇编语言知识,能够想到的方法以下 (let ((label-greater-than (intern (symbol-name (gensym)) :keyword)) (label-end (intern (symbol-name (gensym)) :keyword))) ;; 根据这篇文章(https://en.wikibooks.org/wiki/X86_Assembly/Control_Flow#Comparison_Instructions)中的说法,大于号左边的数字应该放在CMP指令的第二个操做数中,右边的放在第一个操做数中 `((movl ,(get-operand expr 0) %eax) (movl ,(get-operand expr 1) %ebx) (cmpl %ebx %eax) (jg ,label-greater-than) (movl $0 %eax) (jmp ,label-end) ,label-greater-than (movl $1 %eax) ,label-end)))))
而后即可以在REPL中运行下列代码了函数
(let* ((ht (make-hash-table)) (asm (jjcc2 (inside-out '(_exit (> 1 2))) ht))) (stringify asm ht))
输出的汇编代码为rest
.data G809: .long 0 .section __TEXT,__text,regular,pure_instructions .globl _main _main: MOVL $1, %EAX MOVL $2, %EBX CMPL %EBX, %EAX JG G810 MOVL $0, %EAX JMP G811 G810: MOVL $1, %EAX G811: MOVL %EAX, G809(%RIP) MOVL G809(%RIP), %EDI MOVL $33554433, %EAX SYSCALL
编译连接运行后,就能够获得预期的结果了。code
全文完。orm
阅读原文htm