8259A.c 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. #include "8259A.h"
  2. #include "../../../common/printk.h"
  3. #include "../../../common/kprint.h"
  4. #include "../../../exception/gate.h"
  5. // 导出定义在irq.c中的中段门表
  6. extern void (*interrupt_table[24])(void);
  7. void init_8259A()
  8. {
  9. // 初始化中断门, 中断使用第二个ist
  10. for(int i=32;i<=55;++i)
  11. set_intr_gate(i, 2, interrupt_table[i-32]);
  12. kinfo("Initializing 8259A...");
  13. // 初始化主芯片
  14. io_out8(0x20, 0x11); // 初始化主芯片的icw1
  15. io_out8(0x21, 0x20); // 设置主芯片的中断向量号为0x20(0x20-0x27)
  16. io_out8(0x21, 0x04); // 设置int2端口级联从芯片
  17. io_out8(0x21, 0x01); // 设置为AEOI模式、FNM、无缓冲
  18. // 初始化从芯片
  19. io_out8(0xa0, 0x11);
  20. io_out8(0xa1, 0x28); // 设置从芯片的中断向量号为0x28(0x28-0x2f)
  21. io_out8(0xa1, 0x02); // 设置从芯片连接到主芯片的int2
  22. io_out8(0xa1, 0x01);
  23. // 设置ocw1, 允许所有中断请求
  24. io_out8(0x21, 0x00);
  25. io_out8(0xa1, 0x00);
  26. sti();
  27. kinfo("IRQ circuit 8259A initialized.");
  28. }
  29. /**
  30. * @brief 中断服务程序
  31. *
  32. * @param rsp 中断栈指针
  33. * @param number 中断号
  34. */
  35. void do_IRQ(struct pt_regs *regs, ul number)
  36. {
  37. unsigned char x;
  38. switch (number)
  39. {
  40. case 0x20: // 时钟中断信号
  41. break;
  42. case 0x21: // 键盘中断
  43. x = io_in8(0x60);
  44. printk_color(ORANGE, BLACK, "Received key irq, key code:%#018lx\n", x);
  45. break;
  46. default:
  47. break;
  48. }
  49. if(number!=0x20)
  50. printk_color(ORANGE, BLACK, "Received irq:%#018x\n", number);
  51. // 向主芯片发送中断结束信号
  52. io_out8(PIC_master, PIC_EOI);
  53. }