[linux驱动]proc学习笔记(一)

1,在/proc文件系统下创建节点的API
http://www.cnblogs.com/ziziwu/archive/2011/10/20/2218975.html
  struct proc_dir_entry *create_proc_entry (const char *name, mode_t mode, struct proc_dir_entry *parent);
  其中参数分别是 /proc 文件名,掩码,父目录。

[cpp] view plaincopy

  1. 32 struct proc_dir_entry {
  2. 33         unsigned int low_ino;
  3. 34         umode_t mode;
  4. 35         nlink_t nlink;
  5. 36         kuid_t uid;
  6. 37         kgid_t gid;
  7. 38         loff_t size;
  8. 39         const struct inode_operations *proc_iops;
  9. 40         const struct file_operations *proc_fops;
  10. 41         struct proc_dir_entry *next, *parent, *subdir;
  11. 42         void *data;
  12. 43         atomic_t count;         /* use count */
  13. 44         atomic_t in_use;        /* number of callers into module in progress; */
  14. 45                         /* negative -> it‘s going away RSN */
  15. 46         struct completion *pde_unload_completion;
  16. 47         struct list_head pde_openers;   /* who did ->open, but not ->release */
  17. 48         spinlock_t pde_unload_lock; /* proc_fops checks and pde_users bumps */
  18. 49         u8 namelen;
  19. 50         char name[];
  20. 51 };

[cpp] view plaincopy

  1. 2,实现read和write proc文件系统节点的函数,分别为read_proc和write_proc函数
  2. 原型函数:int (*read_proc)(char *page, char **start, off_t offset, int count, int *eof, void *data)
  3. 原型函数:typedef int (write_proc_t)(struct file *file, const char __user *buffer, unsigned long count, void *data);

http://blog.csdn.net/wbd880419/article/details/6637102
  对于read_proc_t
 
1. 第一个参数:为啥叫page?答案就是如果对proc文件调用读操作,内核会分配一个页大小的缓冲区。如何输出大于一个页的数据呢,这得依赖于第二个和第三个参数了。
为了理解第二三个参数,回忆下与文件操作相关的系统调用:
int open(const char *pathname, int flags);
off_t lseek(int fildes, off_t offset, int whence);
ssize_t read(int fd, void *buf, size_t count);
ssize_t write(int fd, const void *buf, size_t count);
对于proc文件,一次read操作最多只能读取一个page的数据,如果需要读取大于一个页的数据需要保存read的返回值,
然后使用lseek设定offset,然后再次调用read。回到参数的说明:
2. start和off参数:off对应于lseek里面的offset(lseek whence为SEEK_END,offset为负 的情况下,传进来的off为零,具体原因待考古)。
如果不设置*start的值,off的取值只能在[0, count - 1]之间,且能够读取的数据大小为:count - off。可以理解系统拷贝了
 [page + off, page+count - 1]之间的数据到用户的buffer里。如果off的取值超出范围,read将读不到数据。
如果设置了*start的值,系统认为*start指向的地址就是off指定的地址,off的值会被忽略,系统会拷贝[start, start+count-1]之间的
数据到用户空间。当然我们在实现start地址的定位时,可能会需要off的值。
3. count 参数与read中的count一致
4. eof参数,设置了这个参数表明不想再提供数据了,神马意思呢?
如果不设置这个参数,对于上面说的start为空的情况,read_proc返回后,系统如果发现 (count - off) < count 
会接着下发写请求,读取off大小的数据。
例如:有这样的读取操作:
lseek(fd, 2, SEEK_SET); read(fd, buff_r, 30)
第一次调用read_proc, off=2, count=30, 由于我们没有设置start的值,将读取28字节的数据,由于没有设置*eof, 系统会再次下发read_proc
第二次调用read_proc, off=30, count=2, 在参数start的说明里提到,对于这个调用,系统默认读不到数据(怨念啊,为啥要下发)
于是read返回28
如果设置了这个参数就不会有第二次的下发了。
5. data参数,这个是给驱动程序预留的参数。
对于write_proc函数,参数是很简单的,需要说明的只有一点,就是write_proc的第二个参数buffer是用户态的地址,需要用copy_from_user
从用户态把数据拷到内核态的缓冲区里。

时间: 2024-08-25 14:35:20

[linux驱动]proc学习笔记(一)的相关文章

[linux内核]proc学习笔记(一)

1,在/proc文件系统下创建节点的APIhttp://www.cnblogs.com/ziziwu/archive/2011/10/20/2218975.html  struct proc_dir_entry *create_proc_entry (const char *name, mode_t mode, struct proc_dir_entry *parent);  其中参数分别是 /proc 文件名,掩码,父目录. [cpp] view plaincopy 32 struct pro

Introduction the naive“scull” 《linux设备驱动》 学习笔记

Introduction the naive "scull" 首先,什么是scull? scull (Simple Character Utility for Loading Localities). scull is a char driver that acts on a memory area as though it were a device. 和第一个C程序Hello world一样,他什么都不能干,却能很好的阐释怎么一步步进阶的去写驱动 blog的最后,我会给出这对于sc

linux系统驱动基础学习笔记

Linux驱动: 角色:应用程序 API      操作系统      驱动       实际硬件 功能:1.对设备进行初始化和释放2.把数据从内核传送到硬件和从硬件读取数据3.检测和处理设备出现的错误 Linux驱动程序类型:字符设备:由文件系统管理    (通过设备文件访问)块设备:由文件系统管理网络设备:由协议栈管理      (通过socket访问) 查看系统设备文件ls -l /devcat /proc/devicescat /sys/power/state 属性:文件类型   主设备

Android深度探索(卷1)HAL与驱动开发学习笔记(2)

Android深度探索(卷1)HAL与驱动开发学习笔记(2) 第二章搭建Android开发环境 书中介绍了两种JDK的安装方法, 方法一: 从官网下载JDK并进行配置,解压后在终端打开profile文件来设置PATH环境变量(# soure /etc/profile),打开profile文件后输入下面的内容 export PATH=.:developer/jdk6/bin:$PATH 保存profile文件以后,有两种方法可以重新加载profile文件. 1.# sourse  /etc/pro

linux系统构建学习笔记

嵌入式系统构架:(硬件+软件)应用软件层: Application GNU C Library(glibc)文件系统: 系统层: API(Systern Call Interface) OS Core + Power Mannager+ File Manager + GUI Mannager TCP/IP HTTP WAP DataBase Browser DDI(Device Drver Interface) 板级支持:BSP:Board Support Package       OEM A

Android深度探索(卷1)HAL与驱动开发学习笔记(4)

Android深度探索(卷1)HAL与驱动开发学习笔记(4) 第四章  源代码的下载与编译 一.源代码配置Android源代码下载环境 1.建一个用于存放下载脚本文件(repo)的目录 # mkdir ~/bin # PATH=~.bin:$PATH 2.下载repo脚本文件 # curl https://dl-ssl.google.com/dl/googlesource/git-repo/repo > ~/bin/repo #chmod a+x ~/bin/repo 3.创建用于存放Andro

Android深度探索(卷1)HAL与驱动开发学习笔记(6)

Android深度探索(卷1)HAL与驱动开发学习笔记(6) 第六章 第一个Linux驱动程序 统计单词个数   Linux系统将每一个驱动都映射成一个文件.这些文件称为设备文件或驱动文件,都保存在/dev目录中.这种设计理念使得与Linux驱动进行交互就像与普通文件进行交互一样容易.虽然C语言里没有事件的概念,但却有与事件类似的概念,这就是回调(c a l l b a c k)函数.因此,编写Lin u x驱动最重要的一步就是编写阴调函数,否则与设备文件交互的数据将无法得到处理. 6.1编写L

Linux第一周学习笔记(2)

Linux第一周学习笔记(2) 1.3认识Linux (1).什么是Linux Linux是一个操作系统,比如说微软的winodws.苹果电脑的masOS.早于Linux发行的unix: 我们平时使用的PPT.浏览网站.玩游戏这些都是需要操作系统的层面上来完成的: 也包括了我们现在手机使用的系统有三个版本一个是Andriod和苹果的Los以及微软的windows,Los其实也就是Unix系统而Andriod是Linux系统: 我们生活当中常用的一些网站.游戏.QQ.微信这些应用都是在Linux操

linux网络编程学习笔记之二 -----错误异常处理和各种碎碎(更新中)

errno 在unix系统中对大部分系统调用非正常返回时,通常返回值为-1,并设置全局变量errno(errno.h),如socket(), bind(), accept(), listen().erron存放一个正整数来保存上次出错的错误值. 对线程而言,每个线程都有专用的errno变量,不必考虑同步问题. strerror converts to English (Note: use strerror_r for thread safety) perror is simplified str