记得以前学<<Linux Device Driver>>中断的章节,找了一块PCI转并口的卡,然后焊了一块小板,手动去触发中断。
最近看<<linux Kernel development>>中断的章节,已经是两年后了。因为找到了在虚拟机下调试内核的方法,所以也希望调试中断的实现可以在虚拟机下完成,而不是再去焊一块板子。
virtualbox里面的串口设置提供了这种功能,其中一个串口如前所述用于调试信息输出,可以另外再激活一个串口调试中断。
==============================================================
Port Number: COM2 IRQ: 3 I/O Port: 0x2F8
Port Mode: Host Pipe
Creat Pipe
Port/File_Path: /home/qianjiang/temp/test/myserial2
==============================================================
这样,启动虚拟机后,
# cat /proc/interrupts
CPU0
0: 133 XT-PIC-XT-PIC timer
1: 16 XT-PIC-XT-PIC i8042
2: 0 XT-PIC-XT-PIC cascade
3: 1 XT-PIC-XT-PIC
4: 926 XT-PIC-XT-PIC serial
其中第3个IRQ就是我们要用于调试的中断脚了。
写了一个内核模块来验证中断的生效,如下:
Makefile:
obj-m := uarttest.o
uarttest.c
其中8250控制器的寄存器描述参考:
http://www.doc88.com/p-95426735937.html for register description
==========================================================================
#include <linux/init.h>
#include <linux/sched.h>
#include <linux/interrupt.h>
#include <linux/module.h>
#include <asm/io.h>
MODULE_LICENSE("Dual BSD/GPL");
//typedef irqreturn_t (*irq_handler_t)(int, void *);
static irqreturn_t rtc_interrupt(int irq, void * dev_id)
{
u8 IIR;
u8 c;
do
{
IIR = inb(0x2fa); //中断状态寄存器
if((IIR & 0x01) == 0x01) break; //no interrupt
if((IIR & 0x06) == 0x06) //receive error
{
inb(0x2fd); //to clear error
printk(KERN_ALERT "receive error, clear it\n");
}
else if((IIR & 0x04) == 0x04) //receive data
{
c = inb(0x2f8);
printk(KERN_ALERT "got 0x%02x: %c\n", c, c);
}
else if((IIR & 0x02) == 0x02) //can send data
{
printk(KERN_ALERT "should not happen as we did not enable send data interrupt\n");
}
else
{
printk(KERN_ALERT "unkown interrupt 0x%02x\n", IIR);
}
}while(1);
return IRQ_HANDLED;
}
static int hello_init(void)
{
int ret;
printk(KERN_ALERT "Hello kernel\n");
ret = request_irq(3, rtc_interrupt, 0, "uart", NULL);
if(ret != 0)
{
printk("reqiest irq 3 failed\n");
}
//2fb 线路控制寄存器
outb(0xe3, 0x2fb);//DLAB=1 to set baudraut
outb(0x1, 0x2f8); outb(0x0, 0x2f9); //to set baudraut as 115200
outb(0x63, 0x2fb);//set DLAB=0 and set 8N1
//clear interrupt
inb(0x2fd); //read LSR(线路状态寄存器) to clear interrupt status if any error during receive
inb(0x2f8); //read RBR to clear receive interrupt
outb(0x5, 0x2f9); //enable interrupt - 包括接收线错误中断和接收到数据中断
printk(KERN_ALERT "DLAB=0 2f8: %02x %02x %02x\n", inb(0x2f8), inb(0x2f9), inb(0x2fa));
return 0;
}
static void hello_exit(void)
{
printk(KERN_ALERT "Goodbye cruel world\n");
printk(KERN_ALERT "DLAB=0 2f8: %02x %02x %02x\n", inb(0x2f8), inb(0x2f9), inb(0x2fa));
outb(0x0, 0x2f9); //disable interrupt
free_irq(3, NULL);
}
module_init(hello_init);
module_exit(hello_exit);
==========================================================================
make -C ~/prj/pc-kernel/linux-2.6.39.y/ M=$PWD modules
所以再开一个minicom, 连接到虚拟机的com2,在这个窗口输入字符,然后可以在虚拟机的com1得到显示。
在虚拟机下玩中断
原创jiangjqian 博主文章分类:old (2011及之前) ©著作权
©著作权归作者所有:来自51CTO博客作者jiangjqian的原创作品,请联系作者获取转载授权,否则将追究法律责任
下一篇:抓bug的乐趣
提问和评论都可以,用心的回复会被更多人看到
评论
发布评论
相关文章
-
浅玩websocket
浅玩websocket
nodejs -
VMware - - - VMware 虚拟机安装统信服务器3c d3 f5
-
174-地下城游戏
其他房间要么是空的(房间里的值为 0),要么包含增加骑士健康点数的魔法球(若房间里的值为正整数,则表
leetcode 动态规划 算法 矩阵 线性代数 -
LeetCode题解(0174):地下城游戏(Python)
题目:原题链接(困难)标签:动态规划
leetcode python 动态规划 初始化 空间复杂度 -
174. Dungeon Game 地下城游戏
Title一些恶魔抓住了公主(P)并将她关在了地下城的右下角。地
赋值 动态规划 空间复杂度