最近在学着写bootloader,由于汇编太繁杂,希望可以使用C语言完成一部分,然后转成NASM汇编代码,经过摸索,最终找到了一个解决方案,记录于此,留作参考。

核心步骤

  • 使用gcc编译得到.o文件

    这一步需要加上一些参数对编译行为进行控制。具体自行参考gcc文档。

    我最终使用的命令参数为

    shell gcc -fno-pic -fno-asynchronous-unwind-tables -m32 -O2 -s -c -o $@ $<

    具体含义我也没有深入了解,-m32是生成32位代码,-O2是一种优化级别,-fno-pic似乎是必要,要不然每个文件最终都会生成一个额外的函数。

  • 使用objconv程序将.o文件转换成汇编代码

    objconv -fnasm $<

    objconv可以将.o文件转成多种形式的汇编,具体自行查阅,上面的参数是要转成NASM代码。

  • 问题

    还有一个问题是,这样直接转换成的汇编代码包含多个段,以及各种对齐指令,以及enbdrXX指令,需要自己根据需要进行删减。

    我所做的是只保留.text段,并删减对齐指令、enbdrXX指令。

我的需求及实现

最终我需要的是将使用C语言完成的函数转换成汇编形式的函数,并去掉编译器相关、对齐相关的一些代码。我的Makefile如下。

all : $(subst .c,.func,$(wildcard *.c))

%.func : %.asm
./extractFunc.py $< %.asm : %.o
./objconv -fnasm $<
%.o : %.c
gcc -fno-pic -fno-asynchronous-unwind-tables -m32 -O2 -s -c -o $@ $< clean :
rm -rf *.asm *.o *.func

其功能是将当前目录中的所有C语言文件转换成NASM汇编语言文件。

C函数文件示例:

//filename : stripe.c
#define DISPLAY_MEMORY_BASE 0xa0000
#define WIDTH 320
#define HEIGHT 200 void stripe()
{
for (int i = DISPLAY_MEMORY_BASE; i <= DISPLAY_MEMORY_BASE+WIDTH*HEIGHT/2; i++)
{
*((int *)i) = (0b11111100 & i);
}
}

使用make命令得到的结果如下:

; filename : stripe.func
stripe: ; Function begin
mov eax, 655360 ; 0004 _ B8, 000A0000
; Filling space: 7H
; Filler type: lea with same source and destination
; db 8DH, 0B4H, 26H, 00H, 00H, 00H, 00H ?_001: mov edx, eax ; 0010 _ 89. C2
add eax, 1 ; 0012 _ 83. C0, 01
and edx, 0FCH ; 0015 _ 81. E2, 000000FC
mov dword [eax-1H], edx ; 001B _ 89. 50, FF
cmp eax, 687361 ; 001E _ 3D, 000A7D01
jnz ?_001 ; 0023 _ 75, EB
ret ; 0025 _ C3
; stripe End of function

gcc和objconv已经在前面提到过了,那个python脚本的作用删掉一些我所不需要的代码,只保留.text部分,并删除其中的对齐指令和enbdrXX指令。内容如下:

#!/usr/bin/python3

#filename : extractFunc.py

import os
import sys def extractFunc(filename):
fin = open(filename)
lines = fin.readlines()
start = 0
end = 0
lineNum = -1
for line in lines:
lineNum += 1
if line.startswith(f'SECTION .text'):
start = lineNum + 1
continue
if line.startswith(f'SECTION .data'):
end = lineNum
break
funcLines = lines[start:end]
cleanFuncLines = []
for line in funcLines:
if line.startswith(' endbr32') or line.startswith('ALIGN'):
continue
cleanFuncLines.append(line) fout = open(f'{filename[:-4]}.func', 'w')
fout.writelines(cleanFuncLines)
print(filename) if __name__ == '__main__':
extractFunc(sys.argv[1])

参考链接

  1. https://stackoverflow.com/questions/20737947/how-to-generate-a-nasm-compilable-assembly-code-from-c-source-code-on-linux
  2. https://stackoverflow.com/questions/1647359/is-there-a-way-to-get-gcc-to-output-raw-binary

最新文章

  1. Asp.Net WebApi开发注意
  2. UML常用图
  3. find命令的拾遗
  4. php课程---建立一个简单的下拉列表框
  5. 使用PushSharp进行IOS发布应用的消息推送
  6. Android开发探秘之二:导入存在的项目及其注意事项
  7. angular_form
  8. Unix commands in Mac OS X
  9. XCODE 代码行统计
  10. Impala 源码分析-FE
  11. 【JAVA学习】单例模式的七种写法
  12. Let&#39;s Encrypt泛域名SSL证书申请
  13. python学习记录
  14. Ubuntu16.04安装后开发环境配置和常用软件安装
  15. Apache Kafka学习 (二) - 多代理(broker)集群
  16. 无线LoRa智能远传水表
  17. chrome中Timeline的使用(译)
  18. 设置了winform的背景图片随窗体的变化而变化
  19. 重载tostring()方法重要性
  20. bootstrap做的导航

热门文章

  1. redis 数据类型之列表
  2. &lt;c:forEach /c:forEach&gt;中varStatus的作用
  3. Mysql 8+ 版本完全踩坑记录
  4. Windows 7原版映像中添加usb3.0驱动
  5. js 获取元素坐标 和鼠标点击坐标
  6. 生成URL(而不是链接) Generating URLs (and Not Links) | 在视图中生成输出URL |高级路由特性 | 精通ASP-NET-MVC-5-弗瑞曼
  7. Creating Form Elements --Using BeginForm and EndForm 使用内建的Form辅助器方法 精通ASP.NET MVC 5
  8. Web容器、Servlet容器、Spring容器、SpringMVC容器之间的关系
  9. Linux 系统常用管理命令(精简笔记)
  10. pc端的弹性布局适配方案