linux驱动开发前奏

驱动开发概述:

1.驱动分类

2.驱动学习方法


常规分类法: 字符设备    块设备     网络设备

字符设备是一种按字节来访问的设备,字符驱动则负责驱动字符设备,这样的驱动通常支持open, close, read 和 write 系统调用。例如:串口,led,按键。

在大部分Unix系统中,块


设备定义为:以块(通常512字节)为最小传输单位的设备,块设备不能按字节读取数据。

网络接口可以是一个硬件设备,如网卡;但是可以是一个纯粹的软件设备,比如回环接口(lo),一个网络接口负责发送和接收数据报文。

总线分类方法:

USB设备,PCI设备,平台总线设备


        USB无线网卡:   网络接口,  USB设备。

 驱动学习方法:

1.驱动模型

2.硬件操作


#include <linux/module.h>
#include <linux/init.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/io.h>
#include <mach/gpio-bank-k.h>
#include "led.h"

#define LEDCON 0x7f008800
#define LEDDAT 0x7f008808
unsigned int *led_config; 
unsigned int *led_data; 

struct cdev cdev;
dev_t devno;

int led_open(struct inode *node, struct file *filp)
{
	led_config = ioremap(LEDCON,4);
	writel(0x11110000,led_config);
	
	led_data = ioremap(LEDDAT,4);
	
	return 0;
}

long led_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{
	switch (cmd)
	{
	    case LED_ON:
	        writel(0x00,led_data);
	        return 0;
	    
	    case LED_OFF:
	        writel(0xff,led_data);
	        return 0;
	    
	    default:
	    	return -EINVAL;
	}
}

static struct file_operations led_fops =
{
    .open = led_open,
    .unlocked_ioctl = led_ioctl,
};

static int led_init()
{
    cdev_init(&cdev,&led_fops);
    
    alloc_chrdev_region(&devno, 0 , 1 , "myled");
    cdev_add(&cdev, devno, 1);
    
    return 0;	
}

static void led_exit()
{
	cdev_del(&cdev);
	unregister_chrdev_region(devno,1);
}


module_init(led_init);
module_exit(led_exit);
 
 
  分析范例程序     ->    制作思维导图    ->  自己编写代码    ->  驱动程序框架   
</pre><pre name="code" class="html">             <span style="font-family: Arial, Helvetica, sans-serif;">初期不要过多的阅读内核代码。</span>
</pre><pre name="code" class="html">
硬件访问技术:
    访问流程:
                       驱动程序   --->    寄存器设备 
     驱动程序控制设备,主要是通过访问设备的寄存器来控制,因此我们讨论如何访问硬件,就成了访问这些寄存器了。
    地址映射:在linux系统中,无论是内核程序还是应用程序,都只能使用虚拟地址,而芯片手册给出的硬件寄存器地址
  或者RAM地址则是物理地址,无法寄存器读写:

                  1.所谓动态映射:是指在驱动程序中采用ioremap函数将物理地址映射为虚拟地址。

                            原型:  void*    ioremap(physaddr , size)

参数:    physaddr: 待映射的物理地址

size: 映射的区域长度

返回值:映射后的虚拟地址

       2.静态映射:是指linux系统根据用户事先指定的映射关系,在内核启动时,自动将物理地址映射为虚拟地址。


1.如何事先指定映射关系? 填充结构,告诉内核

在静态映射中,用户通过map_desc结构来指明物理地址与虚拟地址的映射关系。

struct  map_desc{

unsigned  long   virtual;     //映射后的虚拟地址

unsigned  long   pfn;        //物理地址所在的页帧号

unsigned  long  length;    //映射长度

unsigned  long type;      //映射的设备类型

                                 } ;

pfn: 利用__phys__to_pfn(物理地址)可以计算出物理地址所在物理也帧号

2.在linux内核中,什么地方完成自动映射?

在   内核代码: linux-2.6.39\arch\arm\mach-s3中的代码 iotable_init(s3c_iodesc,ARRAY_SIZE(s3c_iodesc));


 3.寄存器读写

在完成地址映射后,就可以读写寄存器了,linux提供了一些列函数,来读写寄存器


unsigned ioread8(void   *addr);

unsigned ioread16(void   *addr);

unsigned ioread32(void   *addr);

unsigned readb(void   *addr);

unsigned readw(void   *addr);

unsigned readl(void   *addr);




void iowrite8(u8 value, void *addr);

void iowrite16(u16 value,void *addr);

void iowrite32(u32 value,void* addr);


void writeb(unsigned vaule, address);

void writew(unsigned value ,address);

void writel(unsigned value,address);



unsigned ioread8(void   *addr);

unsigned ioread16(void   *addr);

unsigned ioread32(void   *addr);

unsigned readb(void   *addr);

unsigned readw(void   *addr);

unsigned readl(void   *addr);

你可能感兴趣的:(linux,C语言,驱动,Linux驱动开发)