- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我写了下面的代码,你能解释一下程序集在这里告诉我什么吗?
typedef struct
{
int abcd[5];
} hh;
void main()
{
printf("%d", ((hh*)0)+1);
}
程序集:
.file "aa.c"
.section ".rodata"
.align 8
.LLC0:
.asciz "%d\n"
.section ".text"
.align 4
.global main
.type main, #function
.proc 020
main:
save %sp, -112, %sp
sethi %hi(.LLC0), %g1
or %g1, %lo(.LLC0), %o0
mov 20, %o1
call printf, 0
nop
return %i7+8
nop
.size main, .-main
.ident "GCC: (GNU) 4.2.1"
最佳答案
哇哦,SPARC 汇编语言,我已经年没见过了。
我想我们逐行进行?我将跳过一些无趣的样板文件。
.section ".rodata"
.align 8
.LLC0:
.asciz "%d\n"
这是您在 printf
中使用的字符串常量(很明显,我知道!)需要注意的重要事项是它位于 .rodata
部分(部分是最终可执行镜像的分区;这个分区用于“只读数据”,实际上在运行时是不可变的)并且它被赋予了标签 .LLC0
。以点开头的标签是目标文件私有(private)的。稍后,编译器在要加载字符串常量的地址时将引用该标签。
.section ".text"
.align 4
.global main
.type main, #function
.proc 020
main:
.text
是实际机器代码的部分。这是用于定义名为 main
的全局函数的样板文件头,它在汇编级别与任何其他函数没有区别(在 C 中——在 C++ 中不一定如此)。我不记得 .proc 020
做了什么。
save %sp, -112, %sp
保存之前的寄存器窗口,向下调整堆栈指针。如果你不知道什么是寄存器窗口,你需要阅读架构手册:http://sparc.org/wp-content/uploads/2014/01/v8.pdf.gz . (V8 是 SPARC 的最后一个 32 位迭代,V9 是第一个 64 位迭代。这看起来是 32 位代码。)
sethi %hi(.LLC0), %g1
or %g1, %lo(.LLC0), %o0
这两条指令序列的实际效果是将地址 .LLC0
(这是您的字符串常量)加载到寄存器 %o0
中,这是第一个 传出参数寄存器。 (此函数的参数 在传入参数寄存器中。)
mov 20, %o1
将立即数 100 加载到 %o1
,即第二个传出参数寄存器。这是由 ((foo *)0)+1
计算的值。它是 20,因为您的 struct foo
有 20 个字节长(五个 4 字节的 int
)并且您要求从地址零开始的数组中的第二个。
顺便说一下,只有在基指针的地址处实际上有一个足够大的数组时,计算指针的偏移量才在 C 中得到明确定义; ((foo *)0)
是一个空指针,所以那里没有数组,所以表达式 ((foo *)0)+1
在技术上没有定义行为。针对托管 SPARC 的 GCC 4.2.1 碰巧将其解释为“假装在地址零处有一个任意大的 foo
数组并计算数组成员 1 的预期偏移量”,但其他 (尤其是较新的)编译器可能会做一些完全不同的事情。
call printf, 0
nop
调用 printf
。我不记得零是做什么用的。 call
指令有一个延迟槽(再次阅读体系结构手册),其中填充了一条不执行指令,nop
。
return %i7+8
nop
跳转到寄存器%i7
加八的地址。这具有从当前函数返回的效果。
return
还有一个延迟槽,用另一个nop
填充。在这个延迟槽中应该有一个restore
指令,匹配函数顶部的save
,这样main
的调用者取回其注册窗口。我不知道为什么它不在那里。评论中的讨论讨论了 main
可能不需要弹出注册窗口,和/或您已将 main
声明为 void main()
(它不能保证与任何 C 实现一起工作,除非它的文档特别说明,并且 总是 糟糕的风格)...但是按下而不是弹出寄存器窗口是一件很麻烦的事情一个 SPARC,我认为这两个解释都不令人信服。我什至可以称之为编译器错误。
关于c - 以下程序集对以下 .c 文件做了什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9287986/
C#内部访问修饰符的定义是内部:只能在包含程序集或友元程序集内访问。所以我的问题是什么是 C# 程序集?在包含程序集或 friend 程序集之内是什么意思?它是否意味着在同一个命名空间或项目中? 最佳
任何人都可以回答以下问题。我正在使用 c# 语言。 我可以将程序集调用为 .ddl 或 .exe 文件吗? 我可以将 Assembly Manifest 称为程序集吗? 程序集、元数据和程序集 lis
我正在从 Python 运行一个 .NET COM 程序集,只有当我将程序集 dll 和依赖项复制到我的 Python 根路径 c:\Python27 时才能使它正常工作。 这是不整洁的,所以我想将
作为世界上任何一名程序员,他/她一生中至少有一次,我正在尝试创建我的“革命性”,新的且唯一的操作系统。 :D 好吧,我正在使用虚拟模拟器(Oracle VM Virtual Box),为此我创建了一个
我创建了以下程序来读取 5 个数字,然后 dumpreg查看输入的数字... INCLUDE Irvine32.inc .data count = 5 scor
如何在保护模式下执行 IN 和 OUT 等受限指令? 我发现它需要足够高的特权级别(CPL)才能执行 IO 指令。我怎样才能在内核模式下运行,拥有 IO 权限或任何其他可能对我有帮助的东西? - 我希
目录 C# 程序集、模块和类型概念及关系 概述 程序集 模块 类型 程序集、模块和类型的关系 总结 引用
构建 maven assembly ,我留下了这样的东西: ${project.basedir} / LICENS
我的应用程序由几个核心程序集和几个扩展/插件程序集组成。为了让 MEF 知道插件必须提供的所有部件,即使我永远不会使用它们的任何部件,我也必须加载这些程序集。这使得应用程序需要更多时间来启动(如果我要
我对我们的构建基础结构有一个非常具体的要求,即将另一个 JAR 依赖项的一些内容复制到我的 Web 应用程序的特定子文件夹中。我们正在使用 maven-assembly-plugin,一个自然的方法是
为什么下面的指令会设置符号标志? mov al,0FEh sub al,2 据我了解,AL寄存器可以保存2^8 - 1或255。0FEh = 254(十进制)。减去 2 叶 252。这似乎是正数。 (
我以前使用过 NUnit,但已经有一段时间了,而且从来没有在这台机器上使用过。我在 Program Files 下解压了 2.4.8 版本,并且在尝试加载测试时不断收到此错误。 Could not l
我说的是一个使用 C# 进行游戏编程的小型游戏引擎。所以,我有一个嵌入单声道运行时的 C++ 应用程序(我称之为“启动器”)。我有一个用 C# 编写的程序集,它是我的游戏引擎类库。启动器按照 Embe
我对汇编相当陌生,并尝试从标准输入读取值(从 C 调用 scanf 函数)并将其打印回标准输出(使用 printf)。 .text readstr: .asciz "%d" #strin
谢谢帮助,我的问题是关于从下面的代码中收到的 ax 值? mov al,22h mov cl,0fdh imul cl 真机结果:ff9a 我的预期:00:9a(通过二进制相乘) 第一个数字是 22h
我正在开发一个汇编程序,该程序将整个文本文件读入缓冲区,然后将其显示在控制台中。它立即显示 24 行(每行的最大长度为 80,因为我使用 80 宽 * 25 高的 dossbox )然后等待用户输入,
我正在使用一个简单的程序集片段,使用 BIOS 作为引导加载程序的一部分将字符打印到屏幕上。这是引导加载程序代码。 [org 0x7c00] [bits 16] %include "a20_check
我只是想知道这段代码是什么意思: XOR EAX,EBX XOR EBX,EAX XOR EAX,EBX 最佳答案 那是 xor swapping . 在寄存器上执行它不会遇到常见的难看的失败案例。
我在新电脑上的 XNA 项目开始出现奇怪的错误。我有两个关于解决方案的项目和一个由它们使用的库。其中一个项目,一个 XNA 游戏项目,运行完美。另一个项目是 WindowsForm 和 XNA 的混合
是的,我正在努力实现类似的目标 __asm__(jmp label;); 其中 label 应替换为内存中保存的字符串值(结构体的字段)。 有没有办法做到这一点(或类似的方法可以让我跳转到运行时确定的
我是一名优秀的程序员,十分优秀!