驱动程序 没在控制台输出原因 printk及控制台的日志级别

 
开始自己的第一个驱动程序时,大家应该都是从hello,world开始的吧,是否会遇到虽然我们加载模块成功了,却没有输出hello,world的情况呢,我就遇到了,找了下资料,原来是printk与控制台在作怪。

2010-05-31 16:13:09  :刚才又去查了下资料,下面的方法不一定解决问题,原因是:

转自:http://forum.ubuntu.org.cn/viewtopic.php?f=97&t=127667&start=0

      printk 是内核的调用接口,它在系统init之前,把消息写往控制台,但是一旦系统init之后,便改写到系统的日志中。
这样做的好处不言而喻的,如果内核在任何的时候都可以写信息到控制台,那控制台一定会被这样的信息淹没,而无法工作。printk的实现很简单,就是在有了日志消息后,首先申请控制台的信号量,如果申请到,则调用控制台写方法,写控制台。当linux初始化完成的时候,会在init之前打开控制台,此后,printk便无法申请到信号量,而把消息组织为skb发往特定的netlink缓冲区,由klogd读出,写向/var/log/messages 文件中。

 

    根据上面描述的,我用的是ubuntu,我们在图形界面gnome下的控制台不能显示我们的printk输出,原因就是上面所说的,printk函数是系统内核调用,在init进程启动后就申请不到控制台,所以调用它就没能在图形界面下的控制台输出。那么我们编写驱动程序,有要用printk来调试的时候该怎么办呢?相信还有另外的方法来重定向输出吧,相信linux 的强大,继续学习中……。不过就算不能输出也没关系,我们可以用命令:   cat /var/log/messages来查看printk的输出,或者命令:dmesg  ,或者命令:cat  /var/log/syslog ,都可以,看哪种好用就用吧。


(2010-05-3117:52:07   在这里插一句,输出问题,我终于解决拉!!看这里:http://blog.csdn.net/fantasylf/archive/2010/05/31/5637414.aspx

函数printk的使用方法和printf相似,用于内核打印消息。printk根据日志级别(loglevel)对消息进行分类。

日志级别用宏定义,日志级别宏展开为一个字符串,在编译时由预处理器将它和消息文本拼接成一个字符串,因此printk 函数中日志级别宏和格式字符串间不能有逗号。

下面是两个printk的例子,一个用于打印调试信息,另一个用于打印临界条件信息。

printk(KERN_DEBUG "Here I am: %s:%i/n", _ _FILE_ _, _ _LINE_ _); 
 
 
 
printk(KERN_CRIT "I'm trashed; giving up on %p/n", ptr);
 
 

printk的日志级别定义如下(在linux26/includelinux/kernel.h中):

#defineKERN_EMERG"<0>"/*紧急事件消息,系统崩溃之前提示,表示系统不可用*/
 
 
 
#defineKERN_ALERT"<1>"/*报告消息,表示必须立即采取措施*/
 
 
 
#defineKERN_CRIT"<2>"/*临界条件,通常涉及严重的硬件或软件操作失败*/
 
 
 
#defineKERN_ERR"<3>"/*错误条件,驱动程序常用KERN_ERR来报告硬件的错误*/
 
 
 
#defineKERN_WARNING"<4>"/*警告条件,对可能出现问题的情况进行警告*/
 
 
 
#defineKERN_NOTICE"<5>"/*正常但又重要的条件,用于提醒。常用于与安全相关的消息*/
 
 
 
#defineKERN_INFO"<6>"/*提示信息,如驱动程序启动时,打印硬件信息*/
 
#defineKERN_DEBUG"<7>"/*调试级别的消息*/
 
 
 
extern int console_printk[];
 
#define console_loglevel  (console_printk[0])
 
#define default_message_loglevel  (console_printk[1])
 
#define minimum_console_loglevel  (console_printk[2])
 
#define default_console_loglevel  (console_printk[3]

日志级别的范围是0~7,没有指定日志级别的printk语句默认采用的级别是 DEFAULT_ MESSAGE_LOGLEVEL,其定义列出如下(在linux26/kernel/printk.c中):

/*没有定义日志级别的printk使用下面的默认级别*/
 
 
 
#define DEFAULT_MESSAGE_LOGLEVEL 4 /* KERN_WARNING 警告条件*/
 
 

内核可把消息打印到当前控制台上,可以指定控制台为字符模式的终端或打印机等。默认情况下,“控制台”就是当前的虚拟终端。

为了更好地控制不同级别的信息显示在控制台上,内核设置了控制台的日志级别console_loglevel。printk日志级别的作用是打印一定级别的消息,与之类似,控制台只显示一定级别的消息。

当日志级别小于console_loglevel时,消息才能显示出来。控制台相应的日志级别定义如下:

/* 显示比这个级别更重发的消息*/
 
 
 
#define MINIMUM_CONSOLE_LOGLEVEL  1   /*可以使用的最小日志级别*/
 
#define DEFAULT_CONSOLE_LOGLEVEL  7 /*比KERN_DEBUG 更重要的消息都被打印*/
 
 
 
int console_printk[4] = {
 
DEFAULT_CONSOLE_LOGLEVEL,/*控制台日志级别,优先级高于该值的消息将在控制台显示*/
 
/*默认消息日志级别,printk没定义优先级时,打印这个优先级以上的消息*/
 
DEFAULT_MESSAGE_LOGLEVEL,
 
/*最小控制台日志级别,控制台日志级别可被设置的最小值(最高优先级)*/
 
MINIMUM_CONSOLE_LOGLEVEL,
 
DEFAULT_CONSOLE_LOGLEVEL,/* 默认的控制台日志级别*/
 
 
 
};
 
 

如果系统运行了klogd和syslogd,则无论console_loglevel为何值,内核消息都将追加到/var/log/messages中。如果klogd没有运行,消息不会传递到用户空间,只能查看/proc/kmsg。

变量console_loglevel的初始值是DEFAULT_CONSOLE_LOGLEVEL,可以通过sys_syslog系统调用进行修改。调用 klogd时可以指定-c开关选项来修改这个变量。如果要修改它的当前值,必须先杀掉klogd,再加-c选项重新启动它。
通过读写/proc/sys/kernel/printk文件可读取和修改控制台的日志级别。查看这个文件的方法如下:
#cat /proc/sys/kernel/printk
 
6 4 1 7
 
 

上面显示的4个数据分别对应控制台日志级别、默认的消息日志级别、最低的控制台日志级别和默认的控制台日志级别。

可用下面的命令设置当前日志级别:

# echo 8 > /proc/sys/kernel/printk
 
 
 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值