中断发生时,操作系统会为当前的任务建立一个快照,陷入内核,把CPU的控制权交给内核。内核趁这个机会做一些工作,比如调度执行其他任务。这只是中断的作用之一。

使用中断有一套固定的流程,掌握它即可。流程大概如下:

初始化工作是对主从8259A的两类端口赋值。这两类端口是:ICWOCW

这是初始化命令字端口,一共有四个。

  1. ICW1,设置是否级联、是否向ICW4写入数据。
  2. ICW2,设置主从8259A的初始中断向量号。注意,有一批中断向量号已经在实模式下使用,设置初始中断向量号应该避免覆盖那批中断向量号。
  3. ICW3,主片使用位图标识几号引脚挂载从片。从片使用低3位识别主片发送的数据的接收方是否是自己。
  4. ICW4,设置是否主动发送EOI。不清楚这点。

这是控制命令字端口。目前只需要操作一个。

设置屏蔽哪些端口、放行哪些端口。1表示屏蔽,0表示放行。

IDT是中断向量表。类似GDT,也是内存中的一块区域。但它内部包含的全是”门描述符”。

IDT中的每个门描述符的名称是中断向量号,选择子是中断向量号对应的处理中断的代码。

  1. ; Gate : offsetselectorattrparamCount
  2. %macro Gate 4
  3. dw %1 & 0FFFFh
  4. dw %2 & 0FFFFh
  5. dw ((%3 & 0FFh) << 8) | (%4 & 01Fh)
  6. db %1 >> 16
  7. %endmacro
  8. [SECTION .idt]
  9. LABEL_IDT:
  10. %rep 128
  11. Gate selector32, SpuriousHandler, attr, paramCount
  12. %endrep
  13. .080h: Gate selector32, UserIntHandler, attr, paramCount
  14. IDTLen equ $ - LABEL_IDT
  15. IDTPtr dw IDTLen - 1
  16. dd 0

IDT中,第一个元素是可以用的,这与GDT不同。

  1. %rep 128
  2. Gate selector32, offset, attr, paramCount
  3. %endrep

表示IDT中有128个门描述符都指向相同的目标代码段。从0算起,128个门描述符,紧接着的中断向量号应该是128,正好是80h

  1. _SpuriousHandler:
  2. SpuriousHandler equ _SpuriousHandler - $$
  3. mov al, 'A'
  4. mov ah, 0Fh
  5. mov [gs:(80*20+20)*2], ax
  6. _UserIntHandler:
  7. UserIntHandler equ _UserIntHandler - $$
  8. mov al, 'A'
  9. mov ah, 0Fh
  10. mov [gs:(80*20+20)*2], ax

上面的代码和IDT在同一代码段code32。SpuriousHandlerUserIntHandler是code32内的两个偏移量。

不能完全理解这种写法。不过,可暂且当作一种语法规则去记住就行了。以后有类似功能需要实现,就用这种写法。

当中断向量号是在0~127(包括0和127)时,会执行中断处理代码SpuriousHandler

当中断向量号是080h时,会执行中断处理代码UserIntHandler

调用中断的语句很简单,如下:

  1. int 00h
  2. int 01h
  3. int 10h
  4. int 80h

增加一个中断的流程:

  1. IDT中增加一个门描述符,提供目标代码(中断处理程序)的选择子、偏移量等。
  2. 编写中断处理代码。
  3. 调用中断。

时钟中断很特殊,不需要直接调用就能被调用。

编写时钟中断的完整代码如下:

  1. ; Gate : offsetselectorattrparamCount
  2. %macro Gate 4
  3. dw %1 & 0FFFFh
  4. dw %2 & 0FFFFh
  5. dw ((%3 & 0FFh) << 8) | (%4 & 01Fh)
  6. db %1 >> 16
  7. %endmacro
  8. [SECTION .idt]
  9. LABEL_IDT:
  10. %rep 128
  11. Gate selector32, SpuriousHandler, attr, paramCount
  12. %endrep
  13. .080h: Gate selector32, UserIntHandler, attr, paramCount
  14. .81h: Gate selector32, ClockIntHandler, attr, paramCount
  15. IDTLen equ $ - LABEL_IDT
  16. IDTPtr dw IDTLen - 1
  17. dd 0
  18. _SpuriousHandler:
  19. SpuriousHandler equ _SpuriousHandler - $$
  20. mov al, 'A'
  21. mov ah, 0Fh
  22. mov [gs:(80*20+20)*2], ax
  23. _UserIntHandler:
  24. UserIntHandler equ _UserIntHandler - $$
  25. mov al, 'A'
  26. mov ah, 0Fh
  27. mov [gs:(80*20+20)*2], ax
  28. _ClockIntHandler:
  29. ClockIntHandler equ _ClockIntHandler - $$
  30. inc [gs:(80*20+20)*2]
  31. int 80h
  32. ;在死循环中,时钟中断会以一定的时间间隔发生。
  33. jmp $

[gs:(80*20+20)*2]这块内存已经有数据了,inc [gs:(80*20+20)*2]会增加这块内存中的数据的值,效果是在屏幕上出现不断变换的字母。

版权声明:本文为chuganghong原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/chuganghong/p/14471985.html