Linux嵌入式学习( Makefile 模板和 Linux 字符驱动)

Makefile 模板

ifneq ($(KERNELRELEASE),)

obj-m := xxx.o

else

PWD := $(shell pwd)

KVER := $(shell uname -r)

KDIR := /lib/modules/$(KVER)/build

all:

	$(MAKE) -C $(KDIR) M=$(PWD) modules

clean:

	rm -fr .*.cmd *.o *.mod.c *.ko .tmp_versions modules.* Module.*

endif

注意事项

  • xxx.o 处是你需要编译的驱动文件名
  • $(MAKE) -C $(KDIR) M = $(PWD) modulesrm -fr .* .cmd * .o * .mod.c * .ko .tmp_versions modules.* Module. *这两行应该按一次tab键进行缩进,而不是使用空格

字符驱动程序模板

#include <linux/miscdevice.h>  
#include <linux/delay.h>
#include <asm/irq.h>
#include <mach/regs-gpio.h>
#include <mach/hardware.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/types.h>
#include <linux/moduleparam.h>
#include <linux/slab.h>
#include <linux/errno.h>
#include <linux/ioctl.h>
#include <linux/cdev.h>
#include <linux/string.h>
#include <linux/list.h>
#include <linux/pci.h>
#include <linux/gpio.h>
#include <linux/uaccess.h>
#include <asm/atomic.h>
#include <asm/unistd.h>
#include <linux/device.h>


//文件名
#define DEVICE_NAME "devicename" 


//预定义的主设备号,预设是0的话就会动态创建设备号
static int xxx_major = 0; 

 
//字符设备结构体
//这一段和static struct cdev cdev;表达了一种意思
struct xxx_dev_t
{    
	struct cdev cdev;
}xxx_dev;


//在驱动初始化的代码里调用class_create为该设备创建一个class,再为每个设备调用device_create创建对应的设备
static struct class *xxx_class; 


//文件节点,仅有一个;file结构体,追踪文件运行时的状态信息,可为NULL
static int xxx_open(struct inode *inode, struct file *filp)  
{

}


//清理未结束的输入输出操作,释放资源,可为null
static int xxx_realease(struct inode *inode, struct file *filp) 
{

}


//目标文件结构体指针;对应放置信息的缓冲区(用户空间);要读取的信息长度;读的位置相对于文件开头的偏移
static ssize_t xxx_read(struct file *filp, char __user *buf, size_t len, loff_t *offset)
{

}


//目标文件结构体指针;要写入放置信息的缓冲区;要写入的信息长度;写的位置相对于文件开头的偏移
static ssize_t xxx_write(struct file *filp, const char __user *buf, size_t len, loff_t *offset)
{

}


//文件操作结构体
static struct file_operations xxx_fops= 
{
	.owner = THIS_MODULE,
	.read = xxx_read,
	.write = xxx_writ,
	.open = xxx_open,
	.release = xxx_release,
	.ioctl = xxx_ioctl,
};


//设备驱动模块加载函数
static int __init xxx_init(void) 
{
	int result;
   	int ret;
   	
   	dev_t xxx_dev_no = MKDEV(xxx_major,0); //由预定义的主设备号生成dev_t设备号
   	
   	if(xxx_major)
   	{
   		result = register_chrdev_region(xxx_dev_no,1,DEVICE_NAME);  //静态申请设备号
   	}
   	
   	else
   	{
   		result = alloc_chrdev_region(&xxx_dev_no,0,1,DEVICE_NAME); //动态分配设备号
   		xxx_major = MAJOR(xxx_dev_no);
  	}
	
  	if(result<0)
  		return result;
	
  	cdev_init(&xxx_dev.dev,&xxx_fops); //初始化cdev结构
  	xxx_dev.cdev.owner = THIS_MODULE; 
  	ret = cdev_add(&xxx_dev.cdev,MKDEV(xxx_major,0),1); //注册字符设备
  	
  	if (ret)
    {
    	printk("cdev add failed\n"); 
        goto fail1;   
    }
    
  	xxx_class = class_create(THIS_MODULE,"xxx_class");
  	 
  	if(IS_ERR(xxx_class)) 
  	{
         printk("Err: failed in creating class./n");
         goto fail2;  
  	} 
  
  	device_create(xxx_class, NULL, MKDEV(xxx_major,0), NULL, DEVICE_NAME);
    return 0;

  	fail2:
     	cdev_del(&xxx_dev.cdev);
  	fail1:
     	unregister_chrdev_region(xxx_dev_no, 1);
     	return result;
}


//设备驱动模块卸载函数
static void __exit xxx_exit(void) 
{
  	device_destory(xxx_class,MKDEV(xxx_major,0));
  	class_destroy(xxx_class);
  	cdev_del(&xxx_dev.cdev);
  	unregister_chrdev_region(MKDEV(xxx_major,0), 1);	
}

module_init(xxx_init);
module_exit(xxx_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("your_name"); //标明作者信息

参考文章

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值