Linux内核模块编程--替代printk系列
代替 printk在本书开始的时候,我说过 X 和内核模块编程不能混合。在开发内核模块时那是对的,但在实际使用中你想可以向任何向模块发送命令的终端(Teletype, 最初是用于和Unix系统沟通的键盘和打印机的组合,而今天它是用于Unix程序的文本流的抽象,无论它是一个物理的终端、X显示器上的 xterm 还是用telnet 的网络连接,等等。)发送消息。这对在内核模块被释放后识别错误是很重要的,因为它将被所有模块使用。
实现这个的一个办法是使用指向当前运行作业的指针 current得到当前作业的终端结构。然后我们在那个终端结构里面找指向字符串写函数的指针,我们可以用它向终端写字符串。
范例 printk.c
/* printk.c - 向你正在运行的终端输出文本,无论它是否通过 X11, telnet, 等等。 *//* Copyright (C) 1998 by Ori Pomerantz *//* 必要头文件 *//* 标准头文件 */#include /* 内核工作 */#include /* 明确指定是模块 *//* 处理 CONFIG_MODVERSIONS */#if CONFIG_MODVERSIONS==1#define MODVERSIONS#include#endif/* 必要的 */#include /* 为了 current */#include /* 为了终端声明 *//* 向当前作业使用的终端打印字符串 */void print_string(char *str){struct tty_struct *my_tty;/* 当前作业终端 */my_tty = current->tty;/* 如果 my_tty 为 NULL则意味着当前作用没有你可以打印的终端。* (这是可能的,例如它是一个守护进程)* 在这中情况下我们不能做任何事。 */if (my_tty != NULL) {/* my_tty->driver 是包含终端函数的结构,它们中的一个(写)用于向终端写字符串。* 它可以用于从用户内存段或内核内存段取字符串。** 函数的第一个参数是要写向的终端,因为同一函数通常用于所有的属某种类型的终端。* 第二个参数控制是从内核内存段(false,0)还是从用户内存段(true,非零)接收字符串。* 第三个参数是指向字符串的指针,第四个是字符串的长度。*/(*(my_tty->driver).write)(my_tty, /* 终端自己 */0, /* 我们不从用户空间取字符串 */str, /* 字符串 */strlen(str)); /* 长度 *//* 终端是最初的硬件设备,它(通常)严格的坚持 ASCII 标准。* 根据 ASCII,换行需要两个字符,回车和走行。另一方面,在Unix 中,* ASCII 走行符用于这两个目的-因此我们不能仅仅用
,因为它没有回车,* 下一行将在上面那行后的走行符的右边的那列开始而非行首。** 顺便说一下,这就是为什么 Unix 和 Windows的文本文件不同的原因。* 在 CP/M 和它的派生产品,例如 MS-DOS 和 Windows, ASCII 被严格的坚持,因此* 新行需要走行和回车。*/(*(my_tty->driver).write)(my_tty,0,"