Linux实现字符设备驱动的基础步骤

Linux应用层想要操作kernel层的API,比方想操作相关GPIO或寄存器,能够通过写一个字符设备驱动来实现。

1、先在rootfs中的 /dev/ 下生成一个字符设备。注意主设备号 和 从设备号。可用例如以下shell脚本生成:

if [ ! -e audioIN ];then
     sudo mknod audioIN c 240 0
fi

生成的设备为 /dev/audioIN ,主设备号240,从设备号0。

2、写audioINdriver.ko ,audioINdriver.c 基本代码框架例如以下:代码中定义了设备名audioIN,设备号240, 0 ,与之前创建的设备一致。

/************************************************************************** * audioINdriver.c
 *
 * kang_liu <[email protected]>
 * 2014-07-15
\**************************************************************************/

#include <asm/uaccess.h>
#include <asm/errno.h>
#include <linux/types.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/cdev.h>
#include <mach/gpio.h>
//#include <mach/at91_rstc.h> /* debug */
//#include <mach/at91_pmc.h>
//#include <mach/at91_rstc.h>
//#include <mach/at91_shdwc.h>
#include <mach/irqs.h>
//#include "generic.h"
//#include "clock.h"
#include <mach/w55fa92_reg.h>
#include <asm/io.h>

#define DEV_MAJOR 240
#define DEV_MINOR 0
#define NUM_MINORS 1
#define DEVICE_NAME "audioIN"

#define ERR(fmt, args...) printk(KERN_ALERT __FILE__ ": " fmt, ##args)
#define MSG(fmt, args...) printk(KERN_INFO __FILE__ ": " fmt, ##args)
#define DBG(fmt, args...) printk(KERN_DEBUG __FILE__ ": " fmt, ##args)

static ssize_t user_gpio_read(struct file *fp, char __user *buff,
                       size_t count, loff_t *offp)
{
    char str[32] = {0};
    char out[32] = {0};
    int n, err;
//    printk("lk~~~~~~~read buff = %s\n",buff);
    err = copy_from_user(str, buff, count);
//    printk("lk~~~~~~~read str = %s\n",str);
    if (err)
        return -EFAULT;

	sprintf(out,"return values");
	memset(buff, 0, count);
	err = copy_to_user(buff, out, sizeof(out));
	if (err)
	    return -EFAULT;	

    return n;
}

static ssize_t user_gpio_write(struct file *fp, const char __user *buff,
                        size_t count, loff_t *offp)
{
    int err;
    char tmp[32];

//    printk("lk~~~~~~~write buff = %s\n",buff);
    err = copy_from_user(tmp, buff, count);
//    printk("lk~~~~~~~write tmp = %s\n",tmp);

    if (err)
        return -EFAULT;
	if('1' == tmp[0])
	{
		//LINE IN
		printk("line in\n");
	}
	else if('0' == tmp[0])
	{
		//MIC IN
		printk("mic in\n");
	}

    return count;
}

static ssize_t user_gpio_open(struct inode *inode,struct file *fp)
{
//  printk("open gpio devices\n"); 

  return 0;
} 

static struct file_operations user_gpio_file_ops =
{
  .owner = THIS_MODULE,
  .write = user_gpio_write,
  .read = user_gpio_read,
  .open = user_gpio_open,
};

static struct cdev *dev;

static void __exit user_audioIN_exit(void)
{
	printk("exit audioIN\n");
    dev_t devno = MKDEV(DEV_MAJOR, DEV_MINOR);

    unregister_chrdev_region(devno, NUM_MINORS);

    cdev_del(dev);

    return;
}

static int __init user_audioIN_init(void)
{
	printk("init audioIN\n");
    int err = 0;
    int i;
    dev_t devno = MKDEV(DEV_MAJOR, DEV_MINOR);

    err = register_chrdev_region(devno, NUM_MINORS, DEVICE_NAME);

    if (err)
        goto fail_devno;

    dev = cdev_alloc();
    dev->ops = &user_gpio_file_ops;
    dev->owner = THIS_MODULE;

    err = cdev_add(dev, devno, NUM_MINORS);

    if (err)
        goto fail_cdev;

    return err;
fail_cdev:
fail_devno:
    unregister_chrdev_region(devno, NUM_MINORS);
fail_gpio:
    return err;
}

module_init(user_audioIN_init);
module_exit(user_audioIN_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("kang_liu <[email protected]>");
MODULE_DESCRIPTION("Access GSEIO from userspace."); 

这里就能够调用kernel层的一些API进行底层的操作。

Makefile:生成audioINdriver.ko

# Comment/uncomment the following line to disable/enable debugging
#DEBUG = y
BUILD_TOOLS_PRE = arm-linux-

CC=$(BUILD_TOOLS_PRE)gcc
LD=$(BUILD_TOOLS_PRE)ld
# Add your debugging flag (or not) to CFLAGS
ifeq ($(DEBUG),y)
  DEBFLAGS = -O -g -DSCULL_DEBUG # "-O" is needed to expand inlines
else
  DEBFLAGS = -O2
endif

KERNEL_DIR = ../../../linux-2.6.35.4

EXTRA_CFLAGS += $(DEBFLAGS)
EXTRA_CFLAGS += -I$(LDDINC)
EXTRA_CFLAGS +=-I$(KERNEL_DIR)/arch/arm/mach-w55fa92/include
EXTRA_CFLAGS +=-I$(KERNEL_DIR)/arch/arm
EXTRA_CFLAGS +=-I$(KERNEL_DIR)/arch/arm/include
EXTRA_CFLAGS +=-I$(KERNEL_DIR)/arch/arm/include/linux

ifneq ($(KERNELRELEASE),)
# call from kernel build system

audioIN-objs := audioINdriver.o

obj-m	:= audioINdriver.o

else
KERNELDIR ?= $(KERNEL_DIR)
#KERNELDIR ?= /lib/modules/$(shell uname -r)/build
PWD       := $(shell pwd)

modules:
	$(MAKE) ARCH=arm CROSS_COMPILE=$(BUILD_TOOLS_PRE) -C $(KERNELDIR) M=$(PWD) LDDINC=$(PWD)/../include modules

endif

clean:
	rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions modules.order Module.symvers

depend .depend dep:
	$(CC) $(CFLAGS) -M *.c > .depend

ifeq (.depend,$(wildcard .depend))
include .depend
endif

3. 生成好 .ko 以后,就能够在ARM板上,载入驱动。

insmod audioINdriver.ko

4、载入驱动成功后,就能够在应用层直接操作设备 /dev/audioIN,来实现相关功能,将一些參数传到驱动层,运行相关kernel层的代码。

应用层測试程序例如以下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/syscall.h>
#define BUF_LEN 32
int s_audioInFd = 0;

int InitAudioInDevice()
{
	s_audioInFd = open("/dev/audioIN",O_RDWR);

	if (s_audioInFd > 0)
	{
		return 1;
	}
	else
	{
		printf("Can't open the GSE IO device\n");
		return 0;
	}
}

void UninitAudioInDevice()
{
	if (s_audioInFd > 0)
		close(s_audioInFd);
}

int getAudioIn()
{
	char buffer[BUF_LEN] = {0};
	if (s_audioInFd > 0)
	{
		memcpy(&buffer[0], "lk_test", 7);
		//		printf("get buffer = %s\n", buffer);
		int len = read(s_audioInFd, buffer, 7);
		//		printf("get buffer = %s, len = %d\n", buffer, len);
		return len;
	}

	return -1;
}

int setAudioIn(int micLine)
{
	char buffer[BUF_LEN] = {0};
	if (s_audioInFd > 0)
	{
		sprintf(buffer, "%d", micLine);
		int len = write(s_audioInFd, buffer, sizeof(buffer));
		if (len > 0)
			return 1;
	}

	return 0;
}

当中的read 和 write函数,可从驱动中获取一些返回值,也可将字符串传到驱动中。

驱动的入口为:

module_init(user_audioIN_init);

module_exit(user_audioIN_exit);

时间: 2024-10-26 19:52:52

Linux实现字符设备驱动的基础步骤的相关文章

Linux高级字符设备驱动 poll方法(select多路监控原理与实现)

1.什么是Poll方法,功能是什么? 2.Select系统调用(功能)      Select系统调用用于多路监控,当没有一个文件满足要求时,select将阻塞调用进程.      int select(int maxfd, fd_set *readfds, fd_set *writefds, fe_set *exceptfds, const struct timeval *timeout)     Select系统调用(参数)     1)Maxfd:           文件描述符的范围,比

linux 高级字符设备驱动 ioctl操作介绍 例程分析实现【转】

转自:http://my.oschina.net/u/274829/blog/285014 1,ioctl介绍 ioctl控制设备读写数据以及关闭等. 用户空间函数原型:int ioctl(int fd,unsinged long cmd,...) fd-文件描述符 cmd-对设备的发出的控制命令 ...表示这是一个可选的参数,存在与否依赖于cmd,如cmd为修改波特率,那么....就表示波特率的值.如果cmd表示关闭,则不需要参数 内核函数原型 file_operations结构体里面long

Linux内核分析(五)----字符设备驱动实现

原文:Linux内核分析(五)----字符设备驱动实现 Linux内核分析(五) 昨天我们对linux内核的子系统进行简单的认识,今天我们正式进入驱动的开发,我们今后的学习为了避免大家没有硬件的缺陷,我们都会以虚拟的设备为例进行学习,所以大家不必害怕没有硬件的问题. 今天我们会分析到以下内容: 1.      字符设备驱动基础 2.      简单字符设备驱动实现 3.      驱动测试 l  字符设备基础 1.       字符设备描述结构 在linux2.6内核中,使用cdev结构体描述一

Linux设备驱动之字符设备驱动

一.linux系统将设备分为3类:字符设备.块设备.网络设备. 应用程序调用的流程框图: 三种设备的定义分别如下, 字符设备:只能一个字节一个字节的读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后顺序进行.字符设备是面向流的设备,常见的字符设备如鼠标.键盘.串口.控制台.LED等. 块设备:是指可以从设备的任意位置读取一定长度的数据设备.块设备如硬盘.磁盘.U盘和SD卡等存储设备. 网络设备:网络设备比较特殊,不在是对文件进行操作,而是由专门的网络接口来实现.应用程序不能直接访

【转】linux设备驱动程序之简单字符设备驱动

原文网址:http://www.cnblogs.com/geneil/archive/2011/12/03/2272869.html 一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 1.字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流的设备,常见的字符设备有鼠标.键盘.串口.控制台和LED设备等.2.块设备:是指可以从设备的任意位置读取一定长度数据的设备.块设备包括硬盘.磁盘.U盘和SD卡等.

linux 字符设备驱动开发详解

一.设备的分类及特点 1.字符设备 字符设备是面向数据流的设备,没有请求缓冲区,对设备的存取只能按顺序按字节的存取而不能随机访问.    Linux下的大多设备都是字符设备.应用程序是通过字符设备节点来访问字符设备的.通常至少需要实现 open, close, read, 和 write 等系统调用.    设备节点一般都由mknod命令都创建在/dev目录下,包含了设备的类型.主/次设备号以及设备的访问权限控制等,如:crw-rw----  1 root  root 4, 64 Feb 18

linux设备驱动第三篇:写一个简单的字符设备驱动

在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存. 下面就开始学习如何写一个简单的字符设备驱动.首先我们来分解一下字符设备驱动都有那些结构或者方法组成,也就是说实现一个可以使用的字符设备驱动我们必须做些什么工作. 1.主设备号和次设备号 对于字符设备的访问是通过文件系统中的设备名称进行的.他们通常位于/dev目录下.如下: [plain] vie

深入浅出~Linux设备驱动之字符设备驱动

一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流的设备,常见的字符设备有鼠标.键盘.串口.控制台和LED设备等. 块设备:是指可以从设备的任意位置读取一定长度数据的设备.块设备包括硬盘.磁盘.U盘和SD卡等. 每一个字符设备或块设备都在/dev目录下对应一个设备文件.linux用户程序通过设备文件(或称设备节点)来使用驱动程序操作字符设备和块设备

linux设备驱动第三篇:如何写一个简单的字符设备驱动?

在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存. 下面就开始学习如何写一个简单的字符设备驱动.首先我们来分解一下字符设备驱动都有那些结构或者方法组成,也就是说实现一个可以使用的字符设备驱动我们必须做些什么工作. 1.主设备号和次设备号 对于字符设备的访问是通过文件系统中的设备名称进行的.他们通常位于/dev目录下.如下: [email prot