1

我试图以 200 毫秒的周期读取我通过内核模块创建的环回设备,但是当我尝试插入它时,它会使内核崩溃。

我认为我的读取模块有问题,但是没有计时器它可以正常工作。

我是内核编程的新手,请帮助。提前谢谢你:D

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/timer.h>
#include<linux/fs.h>
#include <linux/init.h>
#include <asm/segment.h>
#include <asm/uaccess.h>
#include <linux/buffer_head.h>

static struct timer_list my_timer;

static void read_file(char *filename)
{
  struct file *fd;
  char buf[1];
  unsigned long long offset=0;
  mm_segment_t old_fs = get_fs();
  set_fs(KERNEL_DS);

  fd = filp_open(filename, O_RDONLY, 0);
  if (fd >= 0) {
    printk(KERN_DEBUG);
    while (vfs_read(fd, buf, 1,&offset) == 1)
    {
      if((0 <= buf[0]) && (buf[0] <=255))
        printk("%c", buf[0]);
    } 
    printk(KERN_ALERT "Loop Ran\n");
    filp_close(fd,NULL);
  }
  set_fs(old_fs);
}

void my_timer_callback( unsigned long data )
{
  int ret;
  printk( "my_timer_callback called (%ld).\n", jiffies );  
  printk( "Starting timer to fire in 200ms (%ld)\n", jiffies );  
  read_file("/dev/loop0");  
  ret = mod_timer( &my_timer, jiffies + msecs_to_jiffies(3000) );
  if(ret)  
    printk("Error in mod_timer\n");  
}

int init_module( void )
{
  int ret;
  printk("Timer module installing\n");

  setup_timer( &my_timer, my_timer_callback, 0 );

  printk( "Starting timer to fire in 200ms (%ld)\n", jiffies );
  ret = mod_timer( &my_timer, jiffies + msecs_to_jiffies(200) );
  if(ret)
    printk("Error in mod_timer\n");

  return 0;
}

void cleanup_module( void )
{
  int ret;

  ret = del_timer( &my_timer );
  if(ret)
    printk("The timer is still in use...\n");

  printk("Timer module uninstalling\n");

  return;
}`enter code here`

MODULE_LICENSE("GPL"); 

我的制作文件:

obj-m := timer2.o
all:
    make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules

clean:
    make -C /lib/modules/$(shell uname -r)/build M=$(PWD)   clean 
4

2 回答 2

0

内核定时器函数应该是原子的。文件操作需要一个进程上下文。您的崩溃是由于您的读取操作中存在文件操作。

Linux 设备驱动程序 - 第 7 章应该让您了解内核计时器。

于 2013-09-16T14:33:22.557 回答
0

读取文件相当复杂,因为需要处理许多极端情况。(如果需要扩展 VM 映射怎么办?如果在等待磁盘时必须暂停线程怎么办?等等)

这篇文章讨论了你应该做什么:http ://www.linuxjournal.com/article/8110

不幸的是,这篇文章提供了一些示例代码来解决这个问题,这给人们带来了希望。但是示例代码仅适用于“用户进程调用内核”的上下文。在这种情况下,内核可以重新使用当前的用户进程上下文,但这是一个 hack。

在一般情况下(中断、计时器、蚀刻),您不能只是“获取随机用户上下文”,因为这会导致大量问题。

相反,您应该创建一个用户空间进程,将它需要的数据交给内核。

于 2013-07-06T19:31:48.273 回答