天天看点

linux 0.11 内核学习 -- char_dev.c

#include <errno.h>

#include <sys/types.h> // 定义了基本的系统数据类型

#include <linux/sched.h>

#include <linux/kernel.h> // 含有一些内核常用函数的原形定义

#include <asm/segment.h>

#include <asm/io.h>

extern int tty_read(unsigned minor,char * buf,int count);

extern int tty_write(unsigned minor,char * buf,int count);

typedef (*crw_ptr)(int rw,unsigned minor,char * buf,int count,off_t * pos);

static int rw_ttyx(int rw,unsigned minor,char * buf,int count,off_t * pos)

{

return ((rw==READ)?tty_read(minor,buf,count):

tty_write(minor,buf,count));

}

static int rw_tty(int rw,unsigned minor,char * buf,int count, off_t * pos)

{

// 若进程没有对应的控制终端,则返回出错号

if (current->tty<0)

return -EPERM;

return rw_ttyx(rw,current->tty,buf,count,pos);

}

static int rw_ram(int rw,char * buf, int count, off_t *pos)

{

return -EIO;

}

static int rw_mem(int rw,char * buf, int count, off_t * pos)

{

return -EIO;

}

static int rw_kmem(int rw,char * buf, int count, off_t * pos)

{

return -EIO;

}

static int rw_port(int rw,char * buf, int count, off_t * pos)

{

int i=*pos; // 端口地址

// 对于所要求读写的字节数,并且端口地址小于64k

while (count-->0 && i<65536) 

{

// 若是读命令,则从端口i 中读取一字节内容并放到用户缓冲区中

if (rw==READ)

put_fs_byte(inb(i),buf++);

// 若是写命令,则从用户数据缓冲区中取一字节输出到端口i

else

outb(get_fs_byte(buf++),i);

i++; // 前移一个端口

}

// 计算读/写的字节数,并相应调整读写指针

i -= *pos;

*pos += i;

// 返回读/写的字节数

return i;

}

static int rw_memory(int rw, unsigned minor, char * buf, int count, off_t * pos)

{

// 根据内存设备子设备号,分别调用不同的内存读写函数

switch(minor) 

{

case 0:

return rw_ram(rw,buf,count,pos);

case 1:

return rw_mem(rw,buf,count,pos);

case 2:

return rw_kmem(rw,buf,count,pos);

case 3:

return (rw==READ)?0:count;

case 4:

return rw_port(rw,buf,count,pos);

default:

return -EIO;

}

}

#define NRDEVS ((sizeof (crw_table))/(sizeof (crw_ptr)))

static crw_ptr crw_table[]={

NULL,

rw_memory,

NULL,

NULL,

rw_ttyx,

rw_tty,

NULL,

NULL

};

int rw_char(int rw,int dev, char * buf, int count, off_t * pos)

{

crw_ptr call_addr;

// 如果设备号超出系统设备数,则返回出错码

if (MAJOR(dev)>=NRDEVS)

return -ENODEV;

// 若该设备没有对应的读/写函数,则返回出错码

if (!(call_addr=crw_table[MAJOR(dev)]))

return -ENODEV;

// 调用对应设备的读写操作函数,并返回实际读/写的字节数

return call_addr(rw,MINOR(dev),buf,count,pos);

}

参考《linux内核完全注释》和网上相关资料

继续阅读