mini2440的LEDS驱动程序和测试程序详解
扫描二维码
随时随地手机看文章
一 leds的驱动程序
位置:linux 2.6.29/drivers/char/mini2440_leds.c
#include
#include
#include
#include
#include
#include
#include //具体的头文件位置为/opt/FriendlyARM/mini2440/linux-2.6.29/include/linux/*.h
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define DEVICE_NAME "leds" //定义驱动程序的名字为leds
static unsigned long led_table [] = {
S3C2410_GPB5,
S3C2410_GPB6,
S3C2410_GPB7,
S3C2410_GPB8,
}; //定义引脚的寄存器数组(无符号长整形,对应于引脚的地址)
static unsigned int led_cfg_table [] = {
S3C2410_GPB5_OUTP,
S3C2410_GPB6_OUTP,
S3C2410_GPB7_OUTP,
S3C2410_GPB8_OUTP,
}; //定义引脚功能,为输出(无符号整形)
static int sbc2440_leds_ioctl(
struct inode *inode,
struct file *file,
unsigned int cmd,
unsigned long arg)
{
switch(cmd) {
case 0:
case 1:
if (arg > 4) { //设备节点,文件描述符,LED灯编号,LED灯状态四个命令参数
return -EINVAL;
}
s3c2410_gpio_setpin(led_table[arg], !cmd);
return 0;
default:
return -EINVAL; //EINVAL:表示向函数传递了无效的参数(errno符号变量)
}
}
//初始化字符设备驱动的file_operations 的结构体
static struct file_operations dev_fops = {
.owner = THIS_MODULE,
.ioctl = sbc2440_leds_ioctl,
};
static struct miscdevice misc = {
.minor = MISC_DYNAMIC_MINOR, /* 动态设备号 */
.name = DEVICE_NAME, /* 将在/dev目录生成led设备 */
.fops = &dev_fops, /* 驱动接口 */
};
static int __init dev_init(void)
{
int ret;
int i;
for (i = 0; i < 4; i++) {
/*设置GPIO对应的配置寄存器GPIOCON为输出状态*/
s3c2410_gpio_cfgpin(led_table[i], led_cfg_table[i]);
/*设置GPIO对应的数据寄存器GPIODAT为低电平,在模块加载结束后,四个LED应该是全部都是发光状态*/
s3c2410_gpio_setpin(led_table[i], 0);
}
//注册设备
ret = misc_register(&misc);
printk (DEVICE_NAME"tinitializedn");
return ret;
}
//注销设备驱动
static void __exit dev_exit(void)
{
misc_deregister(&misc);
}
module_init(dev_init); /*声明加载模块初始化函数*/
module_exit(dev_exit); /*声明卸载模块清除函数*/
MOUDLE_LICENSE("GPL"); /*许可证声明*/
MODULE_AUTHOR("FriendlyARM Inc."); /*作者信息*/
1 static 关键字的重要性
全局变量和函数全部用static 进行修饰,则其作用的范围仅仅限于当前的文件,而不是整个系统。防止编译器在连接时,会报告命名错误的“名字空间污染”的问题。
2 ioctl()函数
static int sbc2440_leds_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg)
ioctl函数是文件结构中的一个属性分量。ioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。
struct inode *inode,是设备节点号。fd就是用户程序打开设备时使用open函数返回的文件标示符,cmd就是用户程序对设备的控制命令,unsigned long arg是控制命令的个数。
驱动程序提供了对ioctl的支持,用户就可以在用户程序中使用ioctl函数控制设备的I/O通道。如果函数返回一个非负值,那么该值会被返回给调用程序,表示成功。韩式一般通过switch{case}对设备的一些特性进行控制。switch{case}结构,每一个case对应一个命令码,做出一些相应的操作。在本例中的cmd有两个可选项0和1.0表示灯灭,1表示灯亮。所以case 0,1都要进行操作。由于实际的硬件连接中,是低电平灯亮。所以在对引脚赋值时要取反。 s3c2410_gpio_setpin(led_table[arg], !cmd)
3 static int __init dev_init(void)
_init 宏,定义在include/linux/init.h中。对于非模块加载的驱动程序,通过_init 宏,会把函数中的代码放到.text.init段。这个段在系统启动后会被释放。这样函数代码只有在启动时执行一次,所以可以释放它们以节省内存空间,
3初始化字符设备驱动的file_operations 的结构体
结构体file_operations在头文件 linux/fs.h中定义,用来存储驱动内核模块提供的对 设备进行各种操作的函数的指针。该结构体的每个域都对应着驱动内核模块用来处理某个被请求的 事务的函数的地址。
4ret = misc_register(&misc);
misc_register()用主编号10调用 register_chrdev(),设备名称和函数表指针通过miscdevice数据结构获得。同样,miscdevice 数据结构还保存设备驱动程序所使用的次要号码。完成设备的注册。
5 printk()
利用 printk可以实现内核到Linux 控制台的格式化输出。其用法与标准C的printf类似。在调用驱动程序时,依靠printk输出信息跟踪程序,是很有效的方法。与标准C的printf 不同的是,printk支持分级输出。默认为第四级的输出KERN_ERR。
二 LED测试程序
/opt/FriendlyARM/mini2440/examples/leds
#include
#include
#include
#include
int main(int argc, char **argv) /*运行时参数传递,开或关哪个LED*/
{
int on; /*定义led状态变量,1表示灯亮,2表示灯灭*/
int led_no; /*定义led变量--哪个led*/
int fd; /*定义led设备文件描述符的变量*/
if ( argc != 3 || /*判断命令输入参数个数*/
sscanf(argv[1], "%d", &led_no) != 1 || /* 第一个字符串参数表示要操作led*/