linux驱动(2)-led驱动

主要内容: Linux中led驱动

1, 应用空间调用到驱动的框架
2, 自动创建设备节点的方法
3, 内核对硬件初始化的方式
4, 应用空间和内核之间的数据交互
5, linux中ioctl的实现和gpio库函数的使用
---------------------------------------------------
2, 自动创建设备节点的方法
    #define MINORBITS    20
    #define MINORMASK    ((1U << MINORBITS) - 1)

    #define MAJOR(dev)    ((unsigned int) ((dev) >> MINORBITS))
    #define MINOR(dev)    ((unsigned int) ((dev) & MINORMASK))
    #define MKDEV(ma,mi)    (((ma) << MINORBITS) | (mi))


    // 参数1---当前模块--THIS_MODULE
    // 参数2---字符串,表示类的名字
    //返回值--struct class指针类型
    class_create(owner, name);


    //创建一个设备节点
    // 参数1---class_create返回的指针
    // 参数2---该设备非父类--一般都是填NULL
    //参数3--设备号--包含了主设备号major和次设备号minor
    //参数4---私有数据指针---一般都是填NULL
    //参数5---设备节点的名字
    //结果  /dev/led
    // 返回值--struct device指针
    struct device *device_create(struct class *cls, struct device *parent,
                    dev_t devt, void *drvdata,
                    const char *fmt, ...)
    
    


    // 3, 初始化硬件
    //参数1---物理地址
    //参数2--映射的长度
    //返回值--映射之后的虚拟地址
    gpc0_conf = ioremap(0xE0200060, 8);


    4, 应用空间和内核之间的数据交互
        //  从用户空间获取数据, 一般都用在驱动中写操作中-- xxx_write
        // 参数1---目标地址---内核中的空间的地址
        //参数2---原地址---用户空间的地址
        //参数3---拷贝数据个数
        //返回值--没有拷贝成功的数据个数, 0表示成功
        unsigned long copy_from_user(void * to,const void __user * from,unsigned long n)
        
        
        
        //给用户空间数据, 一般都用在驱动中读操作中-- xxx_read
        unsigned long copy_to_user(void __user * to,const void * from,unsigned long n)


5, linux中ioctl的实现和gpio库函数的使用

    如果需要给用户提供更多的使用api,可以添加一个ioctl接口:
    ioctl()用于给驱动发送指令: 某个灯亮, 某个等灭, 全亮,全灭


            应用空间:
                int ioctl(int fd, int cmd,  .../unsigned long args);
        --------------------------------------------------------
        驱动:
                xxx_ioctl(struct file *filp, unsigned int cmd, unsigned long args)
                {
                    //区分不同的命令
                    switch(cmd)
                    {
                        case 命令1:
                        case 命令2:
                        case 命令3:
                        case 命令4:
                    }
                
                }

扫描二维码关注公众号,回复: 3031092 查看本文章

        命令是如何定义: 由程序猿决定, 一定是要一个整数
            1,直接用一个整数--有可能与系统中的已经存在的命令发生冲突
                #define LED_ALL_ON    0x2222
                #define LED_ALL_OFF 0x3333
        

            2, 用内核提供接口来定义一个整数
                _IO(type,nr) //参数1--魔幻数--字符
                            // 参数2--唯一的数字
                _IOW(type,nr,size)
                _IOR(type,nr,size)
                _IOWR(type,nr,size)

                例子:
                    #define LED_NUM_ON  _IOW('L',0x3456, int)
                    #define LED_NUM_OFF  _IOW('L',0x3457, int)
                    
                    #define LED_ALL_ON  _IO('L',0x3458)
                    #define LED_ALL_OFF  _IO('L',0x3459)
    


    gpio操作方法:
        1, 直接操作gpio口对应的寄存器地址(看原理图---数据手册---地址--ioremap)
            *gpc0_conf &= ~(0xff<<12);
            *gpc0_conf |= (0x11<<12);
    
        2, gpio库函数的接口---只需要知道gpio口的号码即可
            gpio_request(unsigned gpio,const char * label)
            // 将某个gpio配置成输出功能,并且直接输出高低电平
            gpio_direction_output(unsigned gpio,int value)
            
            // 将某个gpio配置成输入功能
            gpio_direction_input(unsigned gpio)
            
            // 将某个gpio配置成特定功能
            s3c_gpio_cfgpin(unsigned int pin,unsigned int config)
            
            //将某个gpio口内部上拉或者下拉
            s3c_gpio_setpull(unsigned int pin,s3c_gpio_pull_t pull)
            
            //获取到gpio的值
            gpio_get_value
            //设置gpio的值
            gpio_set_value
            //通过gpio口获取到中断号码
            gpio_to_irq    
            gpio_free(unsigned gpio)
    

代码示例:

利用ioctl函数实现读写:

/**************led_app.c*************************/

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/ioctl.h>


#define LED_NUM_ON  _IOW('L',0x3456, int)
#define LED_NUM_OFF  _IOW('L',0x3457, int)
#define LED_ALL_ON  _IO('L',0x3458)
#define LED_ALL_OFF  _IO('L',0x3459)


int main(int argc, char *argv[])
{

	int fd = open("/dev/led", O_RDWR);
	if(fd < 0)
	{
		perror("open");
		exit(1);
	}



	while(1)
	{
		ioctl(fd, LED_NUM_ON, 3);
		sleep(1);
		ioctl(fd, LED_NUM_OFF, 3);
		sleep(1);

		ioctl(fd, LED_NUM_ON, 4);
		sleep(1);
		ioctl(fd, LED_NUM_OFF, 4);
		sleep(1);

		ioctl(fd, LED_ALL_ON);
		sleep(1);
		ioctl(fd, LED_ALL_OFF);
		sleep(1);
	
	}
	

	close(fd);

	return 0;
}
/************************led_drv.c*******************************/

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/device.h>
#include <linux/slab.h>
#include <linux/gpio.h>

#include <asm/io.h>
#include <asm/uaccess.h>

#include <mach/gpio.h>


#define LED_NUM_ON  _IOW('L',0x3456, int)
#define LED_NUM_OFF  _IOW('L',0x3457, int)
#define LED_ALL_ON  _IO('L',0x3458)
#define LED_ALL_OFF  _IO('L',0x3459)



//设计一个全局的设备对象类
struct s5pv210_led{
	int dev_major ;
	struct class *cls;
	struct device *dev;
	int value; // 用于存放用户的数据
};

//声明一个对象

struct s5pv210_led *led_dev;



volatile unsigned long *gpc0_conf;
volatile unsigned long *gpc0_data;



int led_drv_open(struct inode *inode, struct file *filp)
{

	printk("-------^_^ %s-------\n", __FUNCTION__);

	static int a = 38;
	filp->private_data = &a;
	
	//做初始化动作
 	*gpc0_conf &= ~(0xff<<12);
	*gpc0_conf |= (0x11<<12);

	
		 
	return 0;

}

// ssize_t write(int fd, const void *buf, size_t count);
ssize_t led_drv_write(struct file *filp, const char __user *buf, size_t count, loff_t *fpos)
{
	int ret;
	
	printk("-------^_^ %s-------\n", __FUNCTION__);


	
	// 区分应用的需求

	//如果要从用户空间获取数据,需要用特定的函数
	//  从用户空间获取数据, 一般都用在驱动中写操作中-- xxx_write
	// 参数1---目标地址---内核中的空间的地址
	//参数2---原地址---用户空间的地址
	//参数3---拷贝数据个数
	//返回值--没有拷贝成功的数据个数, 0表示成功
	ret  = copy_from_user(&led_dev->value, buf, count);
	if(ret > 0)
	{
		printk(KERN_ERR "copy_from_user error\n");
		return -EFAULT;
	}

	if(led_dev->value)
	{
		//点灯
		*gpc0_data |= (0x3<<3); 
	
	}else{
		//灭灯
		*gpc0_data &= ~(0x3<<3); 
	
	}


	//返回传递的数据个数
	return count;

}

int led_drv_close(struct inode *inode, struct file *filp)
{

	
	printk("-------^_^ %s-------\n", __FUNCTION__);

	//如果要灭掉
	*gpc0_data &= ~(0x3<<3); 
	
	return 0;

}

long led_drv_ioctl(struct file *filp, unsigned int cmd, unsigned long args)
{

	// 通过filp找到inode
	struct inode *node = filp->f_path.dentry->d_inode;
	//通过inode获取到注册设备号
	int minor = iminor(node);
	int major = imajor(node);
	//获取到数据
	int *p = (int *)filp->private_data;

	printk("-------major = %d, minor = %d, data = %d\n", major, minor, *p);
	

	//区分不同的命令
	int num = args;

		switch(cmd)
		{
		
			case LED_NUM_ON :
				gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 1);
				gpio_free(S5PV210_GPC0(3));
				break;
		
			case LED_NUM_OFF:
			 	gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 0);
				gpio_free(S5PV210_GPC0(3));
				break;
					
			case LED_ALL_ON:

				gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 1);
				gpio_free(S5PV210_GPC0(3));

				
				gpio_request(S5PV210_GPC0(4), "gpc0_4_led");
				gpio_direction_output(S5PV210_GPC0(4), 1);
				gpio_free(S5PV210_GPC0(4));
				break;
		
					
			case LED_ALL_OFF:
				 gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 0);
				gpio_free(S5PV210_GPC0(3));
				
				gpio_request(S5PV210_GPC0(4), "gpc0_4_led");
				gpio_direction_output(S5PV210_GPC0(4), 0);
				gpio_free(S5PV210_GPC0(4));
				break;
		
			default : 
				printk("unkown cmd\n");
				return -EINVAL;
			
		}
	
	return 0;
}


const struct file_operations led_fops = {
	.open = led_drv_open,
	.write = led_drv_write,
	.release = led_drv_close,
	.unlocked_ioctl = led_drv_ioctl,
};


static int __init led_drv_init(void)
{

	/*
		编写驱动的套路
		0, 实例化全局的设备对象-- kzalloc
		1,  申请主设备号---register_chrdev
		2, 自动创建设备节点---class_create, device_create
		3, 初始化硬件--ioremap
		4,实现 file_operation

	*/ 
	
	// 模块加载函数中主要完成系统资源的申请
	printk("-------^_^ %s-------\n", __FUNCTION__);
	int ret;

	// 0, 实例化全局的设备对象
	//参数1---分配大小
	//参数2--分配的标志, GFP_KERNEL--如果当前暂时没有内存,会尝试等待
	led_dev = kzalloc(sizeof(struct s5pv210_led), GFP_KERNEL);
	if(led_dev == NULL)
	{
		printk(KERN_ERR"kzalloc error\n");
		return -ENOMEM;
	}
	
	
	// 1,  申请主设备号
	led_dev->dev_major = 250;
	ret = register_chrdev(led_dev->dev_major, "led_drv",  &led_fops);
	if(ret < 0)
	{
		printk("register_chrdev error\n");
		ret = -EINVAL;
		goto err_free;
	}

	// 2 ---自动创建设备节点

	//创建一个类
	// 参数1---当前模块--THIS_MODULE
	// 参数2---字符串,表示类的名字
	//返回值--struct class指针类型
	led_dev->cls = class_create(THIS_MODULE,"led_cls");
	if(IS_ERR(led_dev->cls))
	{
		printk("class_create error\n");
		ret = PTR_ERR(led_dev->cls);
		goto err_unregister;
		
	}

	//创建一个设备节点
	// 参数1---class_create返回的指针
	// 参数2---该设备非父类--一般都是填NULL
	//参数3--设备号--包含了主设备号major和次设备号minor 
	//参数4---私有数据指针---一般都是填NULL
	//参数5---设备节点的名字
	//结果  /dev/led
	// 返回值--struct device指针
	led_dev->dev = device_create(led_dev->cls, NULL,MKDEV(led_dev->dev_major, 0), NULL, "led");
	if(IS_ERR(led_dev->dev))
	{
		printk("device_create error\n");
		ret = PTR_ERR(led_dev->dev);
		goto err_class_destroy;
		
	}

	// 3, 初始化硬件
	//参数1---物理地址
	//参数2--映射的长度
	//返回值--映射之后的虚拟地址
	gpc0_conf = ioremap(0xE0200060, 8);
	gpc0_data = gpc0_conf + 1;


	
	return 0;


err_class_destroy:
	class_destroy(led_dev->cls);

err_unregister:
	unregister_chrdev(led_dev->dev_major, "led_drv");

err_free:
	kfree(led_dev);
	return ret;


}


static void __exit led_drv_exit(void)
{

	printk("-------^_^ %s-------\n", __FUNCTION__);
	
	iounmap(gpc0_conf);
	// 模块卸载函数中主要完成系统资源的释放
	device_destroy(led_dev->cls, MKDEV(led_dev->dev_major, 0));
	
	class_destroy(led_dev->cls);
	
	//参数1---已经申请到的设备号
	//参数2--字符串--描述设备驱动信息--自定义
	unregister_chrdev(led_dev->dev_major, "led_drv");

	kfree(led_dev);

}


module_init(led_drv_init);
module_exit(led_drv_exit);
MODULE_LICENSE("GPL");

read/write函数实现读写:

/****************************led_app.c********************************/

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/ioctl.h>


#define LED_NUM_ON  _IOW('L',0x3456, int)
#define LED_NUM_OFF  _IOW('L',0x3457, int)
#define LED_ALL_ON  _IO('L',0x3458)
#define LED_ALL_OFF  _IO('L',0x3459)


int main(int argc, char *argv[])
{

	int fd = open("/dev/led", O_RDWR);
	if(fd < 0)
	{
		perror("open");
		exit(1);
	}



	while(1)
	{
		ioctl(fd, LED_NUM_ON, 3);
		sleep(1);
		ioctl(fd, LED_NUM_OFF, 3);
		sleep(1);

		ioctl(fd, LED_NUM_ON, 4);
		sleep(1);
		ioctl(fd, LED_NUM_OFF, 4);
		sleep(1);

		ioctl(fd, LED_ALL_ON);
		sleep(1);
		ioctl(fd, LED_ALL_OFF);
		sleep(1);
	
	}
	

	close(fd);

	return 0;
}
/**********************led_drv.c***********************************/

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/device.h>
#include <linux/slab.h>
#include <linux/gpio.h>

#include <asm/io.h>
#include <asm/uaccess.h>

#include <mach/gpio.h>


#define LED_NUM_ON  _IOW('L',0x3456, int)
#define LED_NUM_OFF  _IOW('L',0x3457, int)
#define LED_ALL_ON  _IO('L',0x3458)
#define LED_ALL_OFF  _IO('L',0x3459)



//设计一个全局的设备对象类
struct s5pv210_led{
	int dev_major ;
	struct class *cls;
	struct device *dev;
	int value; // 用于存放用户的数据
};

//声明一个对象

struct s5pv210_led *led_dev;



volatile unsigned long *gpc0_conf;
volatile unsigned long *gpc0_data;



int led_drv_open(struct inode *inode, struct file *filp)
{

	printk("-------^_^ %s-------\n", __FUNCTION__);
	//做初始化动作
 	*gpc0_conf &= ~(0xff<<12);
		*gpc0_conf |= (0x11<<12);
	
		 
	return 0;

}

// ssize_t write(int fd, const void *buf, size_t count);
ssize_t led_drv_write(struct file *filp, const char __user *buf, size_t count, loff_t *fpos)
{
	int ret;
	
	printk("-------^_^ %s-------\n", __FUNCTION__);
	// 区分应用的需求

	//如果要从用户空间获取数据,需要用特定的函数
	//  从用户空间获取数据, 一般都用在驱动中写操作中-- xxx_write
	// 参数1---目标地址---内核中的空间的地址
	//参数2---原地址---用户空间的地址
	//参数3---拷贝数据个数
	//返回值--没有拷贝成功的数据个数, 0表示成功
	ret  = copy_from_user(&led_dev->value, buf, count);
	if(ret > 0)
	{
		printk(KERN_ERR "copy_from_user error\n");
		return -EFAULT;
	}

	if(led_dev->value)
	{
		//点灯
		*gpc0_data |= (0x3<<3); 
	
	}else{
		//灭灯
		*gpc0_data &= ~(0x3<<3); 
	
	}


	//返回传递的数据个数
	return count;

}

int led_drv_close(struct inode *inode, struct file *filp)
{

	
	printk("-------^_^ %s-------\n", __FUNCTION__);

	//如果要灭掉
	*gpc0_data &= ~(0x3<<3); 
	
	return 0;

}

long led_drv_ioctl(struct file *filp, unsigned int cmd, unsigned long args)
{
	//区分不同的命令
	int num = args;

		switch(cmd)
		{
		
			case LED_NUM_ON :
				gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 1);
				gpio_free(S5PV210_GPC0(3));
				break;
		
			case LED_NUM_OFF:
			 	gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 0);
				gpio_free(S5PV210_GPC0(3));
				break;
					
			case LED_ALL_ON:

				gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 1);
				gpio_free(S5PV210_GPC0(3));

				
				gpio_request(S5PV210_GPC0(4), "gpc0_4_led");
				gpio_direction_output(S5PV210_GPC0(4), 1);
				gpio_free(S5PV210_GPC0(4));
				break;
		
					
			case LED_ALL_OFF:
				 gpio_request(S5PV210_GPC0(3), "gpc0_3_led");
				gpio_direction_output(S5PV210_GPC0(3), 0);
				gpio_free(S5PV210_GPC0(3));
				
				gpio_request(S5PV210_GPC0(4), "gpc0_4_led");
				gpio_direction_output(S5PV210_GPC0(4), 0);
				gpio_free(S5PV210_GPC0(4));
				break;
		
			default : 
				printk("unkown cmd\n");
				return -EINVAL;
			
		}
	
	return 0;
}


const struct file_operations led_fops = {
	.open = led_drv_open,
	.write = led_drv_write,
	.release = led_drv_close,
	.unlocked_ioctl = led_drv_ioctl,
};


static int __init led_drv_init(void)
{

	/*
		编写驱动的套路
		0, 实例化全局的设备对象-- kzalloc
		1,  申请主设备号---register_chrdev
		2, 自动创建设备节点---class_create, device_create
		3, 初始化硬件--ioremap
		4,实现 file_operation

	*/ 
	
	// 模块加载函数中主要完成系统资源的申请
	printk("-------^_^ %s-------\n", __FUNCTION__);
	int ret;

	// 0, 实例化全局的设备对象
	//参数1---分配大小
	//参数2--分配的标志, GFP_KERNEL--如果当前暂时没有内存,会尝试等待
	led_dev = kzalloc(sizeof(struct s5pv210_led), GFP_KERNEL);
	if(led_dev == NULL)
	{
		printk(KERN_ERR"kzalloc error\n");
		return -ENOMEM;
	}
	
	
	// 1,  申请主设备号
	led_dev->dev_major = 250;
	ret = register_chrdev(led_dev->dev_major, "led_drv",  &led_fops);
	if(ret < 0)
	{
		printk("register_chrdev error\n");
		ret = -EINVAL;
		goto err_free;
	}

	// 2 ---自动创建设备节点

	//创建一个类
	// 参数1---当前模块--THIS_MODULE
	// 参数2---字符串,表示类的名字
	//返回值--struct class指针类型
	led_dev->cls = class_create(THIS_MODULE,"led_cls");
	if(IS_ERR(led_dev->cls))
	{
		printk("class_create error\n");
		ret = PTR_ERR(led_dev->cls);
		goto err_unregister;
		
	}

	//创建一个设备节点
	// 参数1---class_create返回的指针
	// 参数2---该设备非父类--一般都是填NULL
	//参数3--设备号--包含了主设备号major和次设备号minor 
	//参数4---私有数据指针---一般都是填NULL
	//参数5---设备节点的名字
	//结果  /dev/led
	// 返回值--struct device指针
	led_dev->dev = device_create(led_dev->cls, NULL,MKDEV(led_dev->dev_major, 0), NULL, "led");
	if(IS_ERR(led_dev->dev))
	{
		printk("device_create error\n");
		ret = PTR_ERR(led_dev->dev);
		goto err_class_destroy;
		
	}

	// 3, 初始化硬件
	//参数1---物理地址
	//参数2--映射的长度
	//返回值--映射之后的虚拟地址
	gpc0_conf = ioremap(0xE0200060, 8);
	gpc0_data = gpc0_conf + 1;


	
	return 0;


err_class_destroy:
	class_destroy(led_dev->cls);

err_unregister:
	unregister_chrdev(led_dev->dev_major, "led_drv");

err_free:
	kfree(led_dev);
	return ret;


}

Makefile:

CROSS_COMPILE = arm-none-linux-gnueabi-
CC = $(CROSS_COMPILE)gcc
APP_NAME = led_app
MODULE_NAME = led_drv


#内核源码路径
KERNEL_DIR = /home/farsight/linux_system/kernel/linux-3.0.8 

CUR_DIR = $(shell pwd)

all :
	make -C $(KERNEL_DIR) M=$(CUR_DIR) modules 
	$(CC) $(APP_NAME).c -o  $(APP_NAME)

clean : 
	make -C $(KERNEL_DIR) M=$(CUR_DIR) clean
	rm -rf $(APP_NAME)	


install:
	cp -raf  *.ko $(APP_NAME)  /opt/rootfs/drv_module/


#指定编译哪个源文件
obj-m = $(MODULE_NAME).o

猜你喜欢

转载自blog.csdn.net/linken_yue/article/details/82318743