在上一篇文章中,实现了对大于号(>)的处理,那么对if表达式的编译也就是信手拈来的事了,不解释太多。在本篇中,将会讲述一下如何产生可以调用来自于C语言标准库的exit(3)函数的汇编代码。

在Common Lisp中并没有一个叫做EXIT的内置函数,所以如同之前实现的_exit一样,我会新增一种需要识别的(first expr),即符号exit。为了可以调用C语言标准库中的exit函数,需要遵循调用约定。对于exit这种只有一个参数的函数而言,情形比较简单,只需要跟对_exit一样处理即可。刚开始,我写下的代码是这样的

(defun jjcc2 (expr globals)

;; 省略不必要的内容

(cond ;; 省略不必要的内容

((member (first expr) '(_exit exit))

;; 暂时以硬编码的方式识别一个函数是否来自于C语言的标准库

`((movl ,(get-operand expr 0) %edi)

(call :|_exit|)))))

对(exit 1)进行编译,会得到如下的代码

.data

.section __TEXT,__text,regular,pure_instructions

.globl _main

_main:

MOVL $1, %EDI

CALL _exit

不过这样的代码经过编译链接之后,一运行就会遇到段错误(segmentation fault)。经过一番放狗搜索后,才知道原来在macOS上调用C函数的时候,需要先将栈对齐到16字节——我将其理解为将指向栈顶的指针对齐到16字节。于是乎,我将jjcc2修改为如下的形式

(defun jjcc2 (expr globals)

;; 省略不必要的内容

(cond ;; 省略不必要的内容

((member (first expr) '(_exit exit))

;; 暂时以硬编码的方式识别一个函数是否来自于C语言的标准库

`((movl ,(get-operand expr 0) %edi)

;; 据这篇回答(https://stackoverflow.com/questions/12678230/how-to-print-argv0-in-nasm)所说,在macOS上调用C语言函数,需要将栈对齐到16位

;; 假装要对齐的是栈顶地址。因为栈顶地址是往低地址增长的,所以只需要将地址的低16位抹掉就可以了

(and ,(format nil "$0x~X" #XFFFFFFF0) %esp)

(call :|_exit|)))))

结果发现还是不行。最后,实在没辙了,只好先写一段简单的C代码,然后用gcc -S生成汇编代码,来看看究竟应当如何处理这个栈的对齐要求。一番瞎折腾之后,发现原来是要处理RSP寄存器而不是ESP寄存器——我也不晓得这是为什么,ESP不就是RSP的低32位而已么。

最后,把jjcc2写成下面这样后,终于可以成功编译(exit 1)了

(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)))

((eq (first expr) 'if)

;; 假定if语句的测试表达式的结果也是放在%eax寄存器中的,所以只需要拿%eax寄存器中的值跟0做比较即可(类似于C语言)

(let ((label-else (intern (symbol-name (gensym)) :keyword))

(label-end (intern (symbol-name (gensym)) :keyword)))

(append (jjcc2 (second expr) globals)

`((cmpl $0 %eax)

(je ,label-else))

(jjcc2 (third expr) globals)

`((jmp ,label-end)

,label-else)

(jjcc2 (fourth expr) globals)

`(,label-end))))

((member (first expr) '(_exit exit))

;; 暂时以硬编码的方式识别一个函数是否来自于C语言的标准库

`((movl ,(get-operand expr 0) %edi)

;; 据这篇回答(https://stackoverflow.com/questions/12678230/how-to-print-argv0-in-nasm)所说,在macOS上调用C语言函数,需要将栈对齐到16位

;; 假装要对齐的是栈顶地址。因为栈顶地址是往低地址增长的,所以只需要将地址的低16位抹掉就可以了

(and ,(format nil "$0x~X" #XFFFFFFFFFFFFFFF0) %rsp)

(call :|_exit|)))))

生成的汇编代码如下

.data

.section __TEXT,__text,regular,pure_instructions

.globl _main

_main:

MOVL $1, %EDI

AND $0xFFFFFFFFFFFFFFF0, %RSP

CALL _exit

好了,这个时候我就在想,如果想要支持其它来自C语言标准库的函数的话,只要依葫芦画瓢就好了,好像还挺简单的——天真的我如此天真地想着。

全文完

c 汇编语言用标准函数代替,调用C标准库的exit函数相关推荐

  1. c语言程序如何调用标准库函数,如何调用C标准库的exit函数详解

    编译大于运算符 原定的计划中这一篇应当是要讲如何编译if表达式的,但是我发现没什么东西可以作为if的test-form的部分的表达式,所以觉得,要不还是先实现一下比较两个数字这样子的功能吧.说干就干, ...

  2. [转载] [python标准库]math——数学函数

    参考链接: Python数学库| expm1()方法 [python标准库]math--数学函数         作用:提供函数完成特殊的数学运算.         Python 版本:1.4 及以后 ...

  3. 咸鱼Micropy标准库—utime 时间函数库

    咸鱼Micropy标准库-utime 时间函数库 查看micropython标准库 utime库提供获取时间和日期.测量时间间隔.延时等函数. 初始时刻:Unix使用了POSIX的系统标准,从1970 ...

  4. C语言程序设计之标准库快速排序qsort函数用法示例

    C语言程序设计之标准库快速排序qsort函数,排序效率高,使用方便,太棒了. qsort函数定义如下: #include <stdlib.h>void qsort(void *base, ...

  5. python语音识别的第三方库_python标准库+内置函数+第三方库: 7.音频处理

    python标准库+内置函数+第三方库 欲善其事,必先利其器 这其器必是python的标准库+内置函数,话说许多第三方库, 也是对标准库的使用,进行封装,使得使用起来更方便. 这些库以使用场景来分类: ...

  6. python定义fmax_Python标准库:内置函数max(iterable, *[, key, default])说明

    max(arg1, arg2, *args[, key]) 本函数是迭代对象iterable进行比较,找出最大值返回.当key参数不为空时,就以key的函数对象为判断的标准. 例子: #max() a ...

  7. C++中标准库 输出 puts()函数

    puts()函数在标准库<iostream>中,输入为字符串. puts("I am a student!");

  8. python标准库random中函数的作用_Python随机函数库random的使用方法详解

    Python随机函数库random的使用方法详解 前言 众所周知,python拥有丰富的内置库,还支持众多的第三方库,被称为胶水语言,随机函数库random,就是python自带的标准库,他的用法极为 ...

  9. C++---STL标准库之set函数全解析,示例讲解,清晰易懂!

    STL---set set定义 set常用函数 set定义 set翻译为集合,是一个内部自动有序且不含重复元素的容器.在考试中,有可能出现需
要去掉重复元素的情况,而且有可能因这些元素比较大或者类型不 ...

最新文章

  1. Scrum之 站立例会
  2. boost::gregorian模块实现测试时钟的测试程序
  3. 2019数据安装勾选_万能的XY数据标签插件,柱形图也可以呈现变化率
  4. softmax函数上溢出和下溢出(转载+自己理解)
  5. linux改目录权限和宿主。
  6. 海康威视面试-java应用开发
  7. macOS U盘烧录Linux iso镜像
  8. php无法创建cookie,php cookie无法正常工作
  9. 设置背景色为渐变色 css
  10. python随机产生手机号码
  11. php ayyay,PHP: curl_setopt - Manual
  12. 2015-2016 Petrozavodsk Winter Training Camp, Moscow SU Trinity Contest
  13. 在使用git bash输入命令时,已输入命令按Backspace键无法删除
  14. fixed在ios失效解决方案
  15. (.Net常识)(int),Int32.Parse,Conver.ToInt32三者在什么情况下使用以及其区别。
  16. 业务系统如何评估服务器,系统容量预估
  17. 记事本app TOP5(个人观点)
  18. 链路追踪google dapper论文 中文
  19. php 正则筛选靓号如AABBCC(连对),abcdef(顺子)等QQ靓号保留
  20. Mask Transfiner for High-Quality Instance Segmentation

热门文章

  1. 深度图像和彩色图像对齐
  2. css定位属性详解(position属性)
  3. 与或非的总结和几种好用的赋值
  4. UDP协议的详细解析
  5. centos7下解压tar.xz文件
  6. 家庭收支记录软件 java
  7. python所有方向的学习路线,千万别做无用功了,正确掌握学习方法
  8. 美团大众点评网:大数据勾勒“大钱景”
  9. 2020年电工(高级)考试题库及电工(高级)试题及答案
  10. 2021-08-05 Java练习题