首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C语言 >

有懂汇编的大神么?该怎么处理

2012-06-02 
有懂汇编的大神么?最近在学习c与汇编的混合编程,遇到了一个问题。简单说来,就是我希望用汇编在内存里预留一

有懂汇编的大神么?
最近在学习c与汇编的混合编程,遇到了一个问题。简单说来,就是我希望用汇编在内存里预留一块内存,然后把这块内存的地址作为C中的一个结构体指针,用对结构体的成员赋值的方式来直接改变内存单元的值。代码如下:

/*文件 asm.c */
//首先定义了一个简单结构体,只有两个int成员
struct tmp{
  int a;
  int b;
};
extern struct tmp *t; //将 src.s 文件中预留的内存块导入,并强制转换为struct tmp类型的指针
int main()
{
  t->a=1; //用对结构体成员赋值的方式直接操作相关内存单元
  t->b=2;
  return 0;
}

/*超简单的汇编文件 src.s ,用的是AT&T 风格的代码*/
.global t //导出变量 t
t:
  .word 0,0 //占用两个双字的内存空间并填0,
  .word 0,0 //即整好两个int变量的长度 4×2

编译并链接
 gcc asm.c -c
 gcc src.s -c
 gcc src.o asm.o
生成了可执行的 a.out文件。到现在为止还没有任何问题,但是当执行a.out文件时,会直接报段错误。想来如此简单的代码,逻辑上出错的可能性微乎其微,内存操作上也似乎并无太大不妥啊,为什么会错呢?求大神指点迷津啊~~

[解决办法]
你没有考虑内存对齐,由于内存对齐的需要,结构体成员之间通常存在间隙,就是说tmp的二进制宽度通常超过4x2,而你只给.global分配了4x2的空间,这样访问t->b的时候就会超出.global的段段描述符所定义的段界限了。给.global中的t预留足够放下sizeof(tmp)的空间就行了。
[解决办法]
汇编的代码里面需要加public么? 还有标号的名字是否合法,C语言的有些编译器对不会改变这个呢 最好Dump一下你那个编译器编译过的文件查看一下 一般不会变
[解决办法]
.global t //导出变量 t
.data
t:
.word 0,0 //占用两个双字的内存空间并填0,
.word 0,0

extern struct tmp t;
t.a=1; 
t.b=2;
[解决办法]
VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习汇编以及C和汇编的对应关系。
从汇编的角度理解和学习C语言的指针,原本看似复杂的东西就会变得非常简单!
指针即地址。“地址又是啥?”“只能从汇编语言和计算机组成原理的角度去解释了。”

提醒:
“学习用汇编语言写程序”

“VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习C和汇编的对应关系。”
不是一回事!

不要迷信书、考题、老师、回帖;
要迷信CPU、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!

不要写连自己也预测不了结果的代码!

热点排行