- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
尝试在测试引导加载程序上实现硬件中断。异常(exception)正在工作(因此发现它是 GPF)。当尝试sti
时,会发生GPF。这是我的主要代码:
cli
lgdt [gdt_desc]
lidt [idt_desc]
mov eax, cr0
or eax, 1
mov cr0, eax
jmp 0x8:bit_32
bit_32:
[bits 32]
mov ax, 0x10
mov ds, ax
mov es, ax
mov fs, ax
mov gs, ax
mov ss, ax
mov eax, 0x8000
mov esp, eax
mov ebp, esp
sti ; exception raised
这就是我的 GDT 的样子:
start_gdt:
null:
dd 0x0
dd 0x0
code:
dw 0xffff
dw 0x0
db 0x0
db 10011010b
db 01000000b
db 0x0
data:
dw 0xffff
dw 0
db 0x0
db 10010010b
db 01001011b
db 0x0
gdt_desc:
dw gdt_desc-start_gdt-1
dd start_gdt
这就是我的 IDT 的样子:
start_idt:
i0:
dw genroutine
dw 0x8
db 0
db 10001110b
dw 0
i1: dw genroutine
dw 0x8
db 0
db 10001111b
dw 0
i2: dw genroutine
dw 0x8
db 0
db 10001110b
dw 0
i3: dw genroutine
dw 0x8
db 0
db 10001111b
dw 0
i5: dw genroutine
dw 0x8
db 0
db 10001111b
dw 0
.
.
;around 50 times, with some modification like for keyboard, GPF etc.
我的 PIC 设置代码:
mov al, 0x11
out 0x20, al
jmp $+2
jmp $+2
out 0xA0, al
jmp $+2
jmp $+2
mov al, 0x20
out 0x21, al
jmp $+2
jmp $+2
mov al, 0x28
out 0xA1, al
jmp $+2
jmp $+2
mov al, 4
out 0x21, al
mov al, 2
jmp $+2
jmp $+2
out 0xA1, al
jmp $+2
jmp $+2
mov al, 11111101b
out 0x20, al
mov al , 11111101b
jmp $+2
jmp $+2
out 0x21, al
ret
在修改IDT中的该条目后尝试启用sti来检查键盘中断,但随后发现sti
导致GPF异常。qemu 日志:
check_exception old: 0xffffffff new 0xd
1: v=0d e=07c2 i=0 cpl=0 IP=0008:0000000000007c74 pc=0000000000007c74 SP=0010:0000000000008000 env->regs[R_EAX]=0000000000008000
EAX=00008000 EBX=00007e15 ECX=00000022 EDX=00002080
ESI=00007e00 EDI=00000800 EBP=00008000 ESP=00008000
EIP=00007c74 EFL=00000246 [---Z-P-] CPL=0 II=0 A20=1 SMM=0 HLT=0
ES =0010 00000000 000bffff 004b9300 DPL=0 DS [-WA]
CS =0008 00000000 0000ffff 00409a00 DPL=0 CS32 [-R-]
SS =0010 00000000 000bffff 004b9300 DPL=0 DS [-WA]
DS =0010 00000000 000bffff 004b9300 DPL=0 DS [-WA]
FS =0010 00000000 000bffff 004b9300 DPL=0 DS [-WA]
GS =0010 00000000 000bffff 004b9300 DPL=0 DS [-WA]
LDT=0000 00000000 0000ffff 00008200 DPL=0 LDT
TR =0000 00000000 0000ffff 00008b00 DPL=0 TSS32-busy
GDT= 00007d4c 00000017
IDT= 00007e15 0000038f
CR0=00000011 CR2=00000000 CR3=00000000 CR4=00000000
DR0=0000000000000000 DR1=0000000000000000 DR2=0000000000000000 DR3=0000000000000000
DR6=00000000ffff0ff0 DR7=0000000000000400
CCS=00000044 CCD=00008000 CCO=EFLAGS
EFER=0000000000000000
我不知道为什么会发生这种情况,我也没有太多的知识来找出答案。请帮忙。
最佳答案
我观察到在您的错误输出中出现了此异常:
check_exception old: 0xffffffff new 0xd
1: v=0d e=07c2 i=0 cpl=0 IP=0008:0000000000007c74 ...
重要的是,这是一个 #GP(一般保护故障)异常,错误代码为 0x7c2
。 OSdev Wiki 有异常概要以及如何解释 #GP 异常的错误代码:
错误代码 0x7c2 是二进制 11111000 01 0
。位 0 清除意味着这不是由外部原因引起的异常。位 1 和位 2 为01
,表示访问 IDT 时发生异常。 11111000
是中断向量的索引,即0xF8
。这是一个危险信号。您的 PIC 重新映射代码似乎将主 pic 重新映射到 0x20-0x27,从 PIC 重新映射到 0x28-0x2f。除非 PIC 重映射代码错误,否则中断 0xF8
没有任何意义。
在查看 PIC 重新映射代码后,我注意到一个问题:
mov al, 0x11
out 0x20, al
out 0xA0, al
mov al, 0x20
out 0x21, al
mov al, 0x28
out 0xA1, al
mov al, 4
out 0x21, al
mov al, 2
out 0xA1, al
mov al, 11111101b
out 0x20, al
mov al , 11111101b
out 0x21, al
ret
为了清楚起见,我删除了 jmp $+2
,因为它们不是必需的。如果您交替更新主 PIC 端口和从 PIC 端口,则 out
指令将充当所需的延迟。 OSDev Wiki 有一个关于执行 PIC remapping and initialization 的部分。您的代码在这里有所不同:
mov al, 4
out 0x21, al ; This is Correct
mov al, 2
out 0xA1, al ; This is Correct
mov al, 11111101b
out 0x20, al ; This is Wrong
mov al , 11111101b
out 0x21, al ; This is Wrong
ret
将 4 写入端口 0x21 并将 2 写入端口 0xA1 后,您需要将 1 写入端口 0xA1,将 1 写入端口 0xA2。然后您可以将中断屏蔽写入端口 0x21 和端口 0xA1 以启用和禁用所需的中断。正确的代码可能类似于:
mov al, 4
out 0x21, al ; This is Correct
mov al, 2
out 0xA1, al ; This is Correct
mov al, 1
out 0xA1, al ; This is Correct
out 0x21, al ; This is Correct
; Now set the PIC masks. Each bit in the mask is 0=enabled interrupt, 1=disabled.
mov al, 0
out 0x21, al ; Enable all interrupts on Slave
out 0xA1, al ; Enable all interrupts on Master
; Now set the PIC masks. Each bit in the mask is 0=enabled interrupt, 1=disabled.
; mov al, 0xfc
; out 0x21, al ; Disable all interrupts on Master except timer and keyboard
; 0xfc = 0b11111100
; mov al, 0xff
; out 0xA1, al ; Disable all interrupts on Slave
ret
我能够通过使用不正确的初始化代码来重现您的 QEMU 异常和中断。我会在 0xF8 处收到中断:
0: v=f8 e=0000 i=0 cpl=0 IP=0008:00007c51 pc=00007c51 ...
随后是 #GP 异常,因为它未处理且位于我的 IDT 之外:
1: v=0d e=07c2 i=0 cpl=0 IP=0008:00007c51 pc=00007c51 ...
修复后,我开始收到中断,例如带有正确条目的计时器,类似于:
0: v=20 e=0000 i=0 cpl=0 IP=0008:00007c4f pc=00007c4f ...
关于x86 - 尝试时出现一般保护错误 `sti`,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62828313/
问题故障解决记录 -- Java RMI Connection refused to host: x.x.x.x .... 在学习JavaRMI时,我遇到了以下情况 问题原因:可
我正在玩 Rank-N-type 并尝试输入 x x .但我发现这两个函数可以以相同的方式输入,这很不直观。 f :: (forall a b. a -> b) -> c f x = x x g ::
这个问题已经有答案了: How do you compare two version Strings in Java? (31 个回答) 已关闭 8 年前。 有谁知道如何在Java中比较两个版本字符串
这个问题已经有答案了: How do the post increment (i++) and pre increment (++i) operators work in Java? (14 个回答)
下面是带有 -n 和 -r 选项的 netstat 命令的输出,其中目标字段显示压缩地址 (127.1/16)。我想知道 netstat 命令是否有任何方法或选项可以显示整个目标 IP (127.1.
我知道要证明 : (¬ ∀ x, p x) → (∃ x, ¬ p x) 证明是: theorem : (¬ ∀ x, p x) → (∃ x, ¬ p x) := begin intro n
x * x 如何通过将其存储在“auto 变量”中来更改?我认为它应该仍然是相同的,并且我的测试表明类型、大小和值显然都是相同的。 但即使 x * x == (xx = x * x) 也是错误的。什么
假设,我们这样表达: someIQueryable.Where(x => x.SomeBoolProperty) someIQueryable.Where(x => !x.SomeBoolProper
我有一个字符串 1234X5678 我使用这个正则表达式来匹配模式 .X|..X|X. 我得到了 34X 问题是为什么我没有得到 4X 或 X5? 为什么正则表达式选择执行第二种模式? 最佳答案 这里
我的一个 friend 在面试时遇到了这个问题 找到使该函数返回真值的 x 值 function f(x) { return (x++ !== x) && (x++ === x); } 面试官
这个问题在这里已经有了答案: 10年前关闭。 Possible Duplicate: Isn't it easier to work with foo when it is represented b
我是 android 的新手,我一直在练习开发一个针对 2.2 版本的应用程序,我需要帮助了解如何将我的应用程序扩展到其他版本,即 1.x、2.3.x、3 .x 和 4.x.x,以及一些针对屏幕分辨率
为什么案例 1 给我们 :error: TypeError: x is undefined on line... //case 1 var x; x.push(x); console.log(x);
代码优先: # CASE 01 def test1(x): x += x print x l = [100] test1(l) print l CASE01 输出: [100, 100
我正在努力温习我的大计算。如果我有将所有项目移至 'i' 2 个空格右侧的函数,我有一个如下所示的公式: (n -1) + (n - 2) + (n - 3) ... (n - n) 第一次迭代我必须
给定 IP 字符串(如 x.x.x.x/x),我如何或将如何计算 IP 的范围最常见的情况可能是 198.162.1.1/24但可以是任何东西,因为法律允许的任何东西。 我要带198.162.1.1/
在我作为初学者努力编写干净的 Javascript 代码时,我最近阅读了 this article当我偶然发现这一段时,关于 JavaScript 中的命名空间: The code at the ve
我正在编写一个脚本,我希望避免污染 DOM 的其余部分,它将是一个用于收集一些基本访问者分析数据的第 3 方脚本。 我通常使用以下内容创建一个伪“命名空间”: var x = x || {}; 我正在
我尝试运行我的test_container_services.py套件,但遇到了以下问题: docker.errors.APIError:500服务器错误:内部服务器错误(“ b'{” message
是否存在这两个 if 语句会产生不同结果的情况? if(x as X != null) { // Do something } if(x is X) { // Do something } 编
我是一名优秀的程序员,十分优秀!