第四章:调试技术

原创 2012年03月29日 12:29:50

注:当初偷懒,本章内容不完整,详见书。

通过使用printk函数调试

printk打印信息拥有不同日志级别,这些表示日志级别的宏展开为一个字符串,所以使用此函数时,日志级别与打印信息之间没有逗号。

printk采用的默认输出级别是定义在<kernel/printk.c>中的DEFAULT_MESSAGE_LOGLEVEL宏。

printk的优先级小于console_loglevel值时,消息才会被显示出来。

如果系统同时运行了klogdsyslogd,则内核消息都将追加到/var/log/messages中,如果klogd没有运行,则这些消息就不会传递到用户空间,只能查看/proc/kmsg文件(使用dmesg命令)。

/proc/sys/kernel/printk文件中4个整数值分别表示:当前的日志级别,默认的日志级别,最小允许的日志级别,引导时的默认日志级别。可以通过echo命令修改其中的值。

开启及关闭消息

程序开发初期需要打开消息输出,正式程序发布时,应该禁止不必要的消息输出。可以通过宏来控制消息的输出。此方法仅有的缺点是每次开启以及关闭消息时需要重新编译模块。

速度限制

我们可以通过下函数来防止一个printk函数重复被执行,以避免有大量重复的消息输出。此函数如下:

int printk_ratelimit(void);

典型调用如下:

if (printk_ratelimit())

printk(KERN_NOTIC “The printer is on fire\n”);

我们可以通过修改/proc/sys/kernel/printk_ratelimit(在重新打开消息之前应该等待的秒数)以及/proc/sys/kernel/printk_ratelimit_burst(在进行速度限制之前可以接受的消息数)。

打印设备编号

<linux/kdev_t.h>有两个可以打印设备编号的宏:

int print_dev_t(char *buffer, dev_t dev); //返回打印的字符数

char *format_dev_t(char *buff, dev_t dev); //返回的是缓冲区

因将来使用64位设备编号的可能性很大,所以buffer缓冲区的大小至少应该有20字节。

使用/proc文件系统

首先必须包含头文件<linux/proc_fs.h>

创建一个只读的/proc文件,驱动程序必须实现一个函数,用于在读取文件是生成数据。在某个进程读取我们的/proc文件时,内核会分配一个内存页,驱动程序可以将数据通过这个内存页返回到用户空间,该函数成为read_proc方法:

int (*read_proc)(char *page, char **start, off_t offset, int count, int *eof, void *data);

starteof是输出参数page指针指向用来写入数据的缓冲区;函数应该使用start返回实际的数据写到内存页的具体位置;eof指向一个整型数,当没有数据可返回时,驱动程序必须设置这个参数;data参数是提供给驱动程序的专用数据指针,用于内部记录。函数的返回值是存放到内存页的字节数。

创建自己的/proc文件

需要把定义好的read_proc函数与一个/proc入口项连接起来,通过下函数:

struct proc_dir_entry *create_proc_read_entry(const char *name

mode_t mode, struct proc_dir_entry *base,

read_proc_t *read_proc, void *data);

name是所创建的文件名称,mode是该文件的保护掩码,base指定该文件所在的目录(如果为NULL,则为/proc根目录),read_proc函数指针,内核会忽略data参数。

对应卸载模块时,删除入口项的函数为:

remove_proc_entry(const char *name, struct_dir_entry *base);

当前使用/proc文件系统更多的是使用seq_file接口。

通过监视调试

strace命令可以显示由用户空间程序所发出的所有系统调用。可以显示每个函数的参数以及函数的返回值。

其他:

在一些关键点上插入schedule调用可以防止死循环,schedule函数会调用调度器,并以此允许其他进程占用当前CPU。一定不要在驱动程序持有自旋锁的任何时候调用schedule

内核空间始于0xc0000000,大于0xc0000000的值肯定是内核空间的地址。

第四章调试技术

printk的默认级别是DEFAULT_MESSAGE_LOGLEVEL这个宏定义在kernel/printk.c中,等于4。这个文件在发行版本ubuntu910里面不存在,但是在标准的2.6.31和...
  • TsuiLei
  • TsuiLei
  • 2009年12月28日 10:39
  • 266

第四章--调试技术

本位作为第四章--调试技术,主要讲述: 1、打印调试。
  • apple_guet
  • apple_guet
  • 2014年03月11日 11:51
  • 694

调试技术(第四章 )

1、输出文件/proc/kmsg的内容时,回应为里面的内容为空而将进程暂时阻塞,等到有数据是再输出。 2、在打印一条可能被重复的信息之前,应调用函数      int printk_ratelim...
  • cnxxrj
  • cnxxrj
  • 2013年11月19日 13:34
  • 914

【Linux 驱动】第四章 调试技术

一,内核中的调试支持       在内核配置菜单中有“kernel hacking”菜单选项,这些选项帮助用户检查很多错误,这里我列了一个表方便大家参考:    kernel hac...
  • tianshuai11
  • tianshuai11
  • 2012年04月11日 23:16
  • 1439

【原创】PHP调试技术手册V1.0.0 (PDF)

《PHP调试技术手册》主要是针对目前流行的各种PHP调试技术的一个实验和总结,所有调试方法都是在容易部署实现的基础来进行讲解,也希望对PHP开发者有一个系统的调试方法的归纳和使用的帮助。 ...
  • heiyeshuwu
  • heiyeshuwu
  • 2010年06月21日 16:48
  • 13401

linux 内核调试技术

linux 调试 目录[-] 一  调试前的准备二  内核中的bug三  内核调试配置选项1  内核配置2  调试原子操作四  引发bug并打印信息1  BUG()和BUG...
  • linuxarmsummary
  • linuxarmsummary
  • 2014年03月29日 20:20
  • 2889

第四章课后习题

第四章处理器体系结构,看了蛮久的。内容倒是挺多的,首先这章将汇编指令转化为二进制代码,然后又研究了下Y86命令集下汇编指令的微实现。然后就开始了处理器的实现。处理器的实现,则是首先从SEQ的顺序实现出...
  • fsk119
  • fsk119
  • 2017年05月15日 21:08
  • 352

嵌入式软件调试技术 读书笔记

第一章 软件调试概述 第二章 边界扫描测试技术 (JTAG) 第三章 学习使用GDB调试器 第四章 GDB远程调试技术 第五章 网络应用程序调试 第六章 多进程与多线程调试 第七章 静态库...
  • jerryutscn
  • jerryutscn
  • 2010年04月27日 19:24
  • 3712

算法导论第四章最后思考题

最近买了《算法导论》第三版新书,所以就不按照第二版的习题来写答案了。 第二版的4.2找出所缺的整数,这题已经在第三版被删除,所以就不做解答。 4.2参数传递的代价 整个这本书中,我们都假...
  • z84616995z
  • z84616995z
  • 2013年12月19日 21:17
  • 2480

机器学习实战第四章——朴素贝叶斯分类(源码解析)

机器学习实战第四章——朴素贝叶斯分类(源码解析)
  • Quincuntial
  • Quincuntial
  • 2016年01月12日 20:05
  • 1250
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:第四章:调试技术
举报原因:
原因补充:

(最多只允许输入30个字)