- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
org 0x7c00
是在平面二进制文件中获取正确绝对地址的正常方法,但我对我希望使用的另一种方法感到好奇。
我尝试使用section boot vstart=0x7c00 align=1
告诉YASM正确的内存地址,而在另一个使用symbol
的部分中使用start=300
。
mov [symbol+$$], register
yasm -fbin boot.asm
在该行上给出
error: effective address too complex
。
symbol+$$
应该能够处理成一个数字(而不是一个段+偏移量),对吗?如果我错了,请告诉我,但是如果我错了,为什么YASM告诉我地址太复杂了?
start=
和/或
vstart=
代替
org
并仍然获得正确的绝对寻址吗?
[symbol]
无效。汇编为
[0000]
的绝对地址
; example.asm
; yasm -fbin example.asm
%define virtual(_name, _offset) section _name vstart=_offset align=1
%define absolute(_name, _offset) section _name start=_offset align=1
virtual(boot, 0x7c00) ; Virtual Offset of 0x7c00 (in-file offset of 0)
start:
; This is just an example
; There isn't going to be much here.
mov [boot_drive+$$], dl
cli
hlt
absolute(vars, 300) ; Virtual AND in-file offset of 300
boot_drive db 0
最佳答案
您的基本问题是您实际上并未将两个数字相加,而是添加了两个符号,并且汇编程序通常不允许这样做。这是因为目标文件格式无法将两个符号的添加表示为重定位,这是因为添加两个符号实际上没有多大意义。虽然在这种情况下,您正在生成不支持重定位的二进制文件,所以汇编程序可以发明自己的虚拟重定位来处理此问题,但是显然,YASM并没有将其实现为一般规则的例外。
为什么汇编器不允许添加符号
在通常情况下(可能会生成目标文件时),添加两个符号没有意义,原因是符号不仅仅是数字。它们还引用一个部分,并且这些部分最终可能驻留在内存中的任何位置。您的[bootdrive + $$]
表达式表示要获取加载到内存中的bootdrive
的实际地址,并将其添加到当前节开始的实际地址中。当生成目标文件时,汇编器将不知道这些实际地址是什么,符号所属的节可以放在任何地方。即使链接器可能也不知道,如果它正在生成可重定位的可执行文件,则将取决于操作系统在何处加载可执行文件。
(这忽略了以下事实:您已经告诉汇编程序,应将bootdrive
的实际地址与汇编程序认为的实际地址不同。这也是汇编程序在通常的输出情况下不支持的东西。对象文件。)
二进制文件可能是一个例外,但不是
现在,在生成二进制文件的情况下,不涉及链接器,因此YASM可以知道bootdrive
的“实际”地址为300,而$$
的实际地址为0x7c00。但这要求汇编器在评估有效地址时将其作为例外,将其传播到生成二进制文件的后端。该异常尚未在汇编程序中实现,因此您可能很难说服YASM(或NASM)开发人员这样做。
您很难说服它们的原因是,即使使用二进制文件,即使可以,添加两个符号也没有任何意义。您的示例代码仅会起作用,因为bootdrive
的地址不是其实际地址。确实,向其中添加$$
的原因是为了计算其实际地址。由于您的示例用例是人为设计的,因此是不必要的,因此有更好的方法来编写可自行重定位的引导加载程序,因此对于添加两个符号为何有意义,它并没有很好的论据。
可能没有直接的解决方法
至于解决方法,我真的想不出任何直接解决方案,仍然需要使用bootloader
和$$
。当有人尝试添加两个符号时,通常有一种方法可以将其重写为有效形式,通常是减去两个符号。减去在同一节中的两个符号受汇编程序的支持,因为它从公式中删除了公共节。因此,例如,[foo + bar_begin - bar_end]
可以写为[foo + (bar_begin - bar_end)]
。但是,我不确定您可以从bootloader
和$$
减去来从方程式中删除它们的任何一部分。
虽然我确定还有其他解决问题的方法,但仍然可以使您使用要使用的section指令完成所需的工作,但我不会费心尝试找出可能是什么。相反,我将建议您一种您不想要的解决方法,如果不是为了您自己的利益,那么是为了其他可能会遇到类似情况的人的利益。
我的解决方案,即使不是您想要的
我的解决方案是不使用段指令来解决引导扇区位于两个不同地址的问题。相反,您可以使用ORG来反映大多数代码在复制后所处的位置。可以很容易地使需要在原始位置执行的少量代码独立于位置,因此不必关心使用什么ORG。
以下是自重定位MBR引导块的框架。为简洁起见,省略了实现MBR所需的大多数代码。
BITS 16
RELOC_OFFSET EQU 0x600
ORG RELOC_OFFSET
start:
xor ax, ax
mov ds, ax
mov es, ax
mov ss, ax
mov sp, 0x7c00
mov di, RELOC_OFFSET
mov si, 0x7c00
mov cx, 512 / 2
cld
rep movsw
jmp 0:relocated_entry
relocated_entry:
mov [boot_drive], dl
; ...
mov dl, [boot_drive]
jmp 0:0x7c00
boot_drive DB 0
TIMES 446 - ($ - $$) DB 0
partition_table:
DB 0x80, 0x01, 0x00, 0x05, 0x17, 0x01, 0x03, 0x01, 0x04, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00
; ...
TIMES 510 - ($ - $$) DB 0
DB 0x55, 0xaa
boot_drive
仅在代码移动后才能访问。
objdump -D -b binary -m i8086 -M intel --adjust-vma=0x600
的输出:
600: 31 c0 xor ax,ax
602: 8e d8 mov ds,ax
604: 8e c0 mov es,ax
606: 8e d0 mov ss,ax
608: bc 00 7c mov sp,0x7c00
60b: bf 00 06 mov di,0x600
60e: be 00 7c mov si,0x7c00
611: b9 00 01 mov cx,0x100
614: fc cld
615: f3 a5 rep movs WORD PTR es:[di],WORD PTR ds:[si]
617: ea 1c 06 00 00 jmp 0x0:0x61c
61c: 88 16 29 06 mov BYTE PTR ds:0x629,dl
620: 8a 16 29 06 mov dl,BYTE PTR ds:0x629
624: ea 00 7c 00 00 jmp 0x0:0x7c00
...
7bd: 00 80 01 00 add BYTE PTR [bx+si+0x1],al
7c1: 05 17 01 add ax,0x117
7c4: 03 01 add ax,WORD PTR [bx+di]
7c6: 04 00 add al,0x0
7c8: 00 00 add BYTE PTR [bx+si],al
7ca: 04 00 add al,0x0
...
7fc: 00 00 add BYTE PTR [bx+si],al
7fe: 55 push bp
7ff: aa stos BYTE PTR es:[di],al
关于assembly - YASM [symbol + $$]有效地址在平面二进制格式中太复杂,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59010047/
在 Android 中集成 FFMPEG 库时出现上述错误。以下是屏幕截图。有人可以帮我整合这个库吗?我正在使用 Ubuntu OS 来实现这一点。 最佳答案 只需安装yasm sudo apt-ge
构建将近半小时后,出现以下错误。到目前为止,谷歌搜索没有任何帮助...... In file included from external/chromium_org/third_party/yasm/
我正在尝试通过 brew 安装 ffmpeg。我用 yasm 得到的错误是 “错误:在安装 ffmpeg 之前,您必须‘brew link pkg-config yasm’” 这样做时,我被告知我的权
org 0x7c00是在平面二进制文件中获取正确绝对地址的正常方法,但我对我希望使用的另一种方法感到好奇。 我尝试使用section boot vstart=0x7c00 align=1告诉YASM正
我有一个 x86_64 汇编程序,我试图在 Linux 上调试,但是当我尝试使用 gdb 时,它会随机跳过并循环执行相同的指令对或重复指令。它还似乎根据我设置断点的位置循环执行不同的指令。 我试着在线
我遇到了一个奇怪的问题。我试图安装 x264。当运行 sudo ./configure --enable-shared 时,它给出了: 找不到汇编器最低版本是 yasm-0.7.0如果您真的想在没有
我正在尝试为使用 的英特尔语法编写的 x86_64 汇编源代码64 位寄存器 .我使用以下命令行标志: yasm foo.asm -a x86 -m amd64 我不断收到以下错误: warning:
假设我声明了以下内容: section .bss buffer resb 1 这些说明位于.text部分: mov al, 5 ; mov-i
我想在c代码中调用ASM函数,如何将参数传递给ASM代码? #include extern int * asm_mod_array(int *ptr,int size); int main()
我需要 Microsoft Visual 2015 中的 Windows 10 中的 YASM 软件,谁能告诉安装 yasm 的步骤。 我已经使用了下面的链接。 https://github.com/
我有一个以原始模式读取字符的程序。也就是说,任何输入的字符都会被立即读取,而不是被缓冲。 我想知道如何执行退格。也就是说,当我按下退格键时,它应该删除左边的字符并将光标向左移动一位。 我尝试输出一个退
我有一个以原始模式读取字符的程序。也就是说,任何输入的字符都会被立即读取,而不是被缓冲。 我想知道如何执行退格。也就是说,当我按下退格键时,它应该删除左边的字符并将光标向左移动一位。 我尝试输出一个退
我正在尝试让 libtool 和 yasm 一起工作。 yasm 从我的 .asm 源创建正确的 .o 文件,但我不知道如何获取 libtool 构建相关的 .lo 和 .dep 文件。它想要构建共享
这个问题在这里已经有了答案: Can't run executable linked with libc (1 个回答) 关闭 2 年前。 当尝试在 linux (x86-64) 上从汇编代码 (y
我已经使用 yasm 汇编了代码,并链接到我的 C++ 程序,但我无法在 gdb 中对来自汇编语言文件的符号设置断点。 命令行可能不是很清楚,但我们开始吧: "g++" -ftemplate-dep
我正在尝试在 64 位 Linux 上运行的 NASM 中实现一个数组的选择排序。 数组声明为: section .bss strbuf resb 10 small resb 1 ; curre
我正在尝试使用 yasm 汇编下面的代码。我在 yasm 报告错误“错误:操作数 2 的大小无效”的地方添加了“此处”注释。为什么会发生这个错误? segment .data a db 25
我有一个关于 nasm 及其与 C++ 的链接的问题。我将一个小测试函数声明为 extern "C"void __cdecl myTest( byte i1, byte i2, int stride,
我正在尝试编写一个即时编译器,但我有一段代码不想工作。我的平台是 x86-64 ubuntu。 我在 yasm 中编写了以下代码: bits 64 mov rdx, 1 mov rcx, 'A' mo
我想在 yasm 程序中使用 POSIX 线程(或简称 pthread)库实现并行处理。 代码 这是我程序中最重要的部分。 section .data pThreadID1 dq
我是一名优秀的程序员,十分优秀!