创建虚拟文件系统设备文件节点_使用proc_create实例分析

proc虚拟文件系统也可以创建虚拟文件节点,实现用户空间与内核空间的交互。在驱动中创建节点,可以实现对硬件的控制。proc_create函数原型(在kernel-3.10/include/linux/proc_fs.h文件)如下所示:

static inline struct proc_dir_entry *proc_create(
const char *name, umode_t mode, struct proc_dir_entry *parent,
	const struct file_operations *proc_fops)
{
	return proc_create_data(name, mode, parent, proc_fops, NULL);
}

name:表示你要创建的设备节点的名称,可随意命名即可;
mode:表示节点的权限,一般赋值0644;
parent:表示父节点,如果直接在proc目录创建节点,直接赋值NULL即可;
proc_fops:表示与节点相关联的file_operations;

如下代码是我实现的一个test程序,可供参考学习proc_create的使用:

#include <linux/init.h>
#include <linux/slab.h>
#include <linux/module.h>
#include <linux/i2c.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/ioctl.h>
#include <linux/uaccess.h>
#include <linux/delay.h>
#include <linux/string.h>
#include <linux/wait.h>
#include <linux/platform_device.h>
#include <linux/gpio.h>
#include <linux/pinctrl/consumer.h>
#include <linux/of_gpio.h>
#include <linux/delay.h>
#include <linux/types.h>
#include <linux/proc_fs.h>

#define BUFSIZE  1024 

static char *buf;
static unsigned int len;

/***********************
 * file_operations->open
 * 无操作
 ***********************/

static int test_proc_open(struct inode *inode, struct file *file)
{
	return 0;
}
 
 
/************************
 * file_operations->read
 * 可以在adb工具进入机器的pro目录,执行adb shell && cd proc && cat tets_rw,
 * 即可读出节点test_rw的内容是12345
 ************************/

static ssize_t test_proc_read(struct file *file,
				char __user *buffer,size_t count, loff_t *f_pos) 
{
	if(*f_pos > 0)
		return 0;
 
printk("---start read---\n");
	printk("the	string is >>>>> %s\n", buf);
 
if(copy_to_user(buffer, buf, len))
		return -EFAULT;
	*f_pos = *f_pos + len;
	return len;
}


/************************
 * file_operations->write
 * 可以在adb工具进入机器的pro目录,
 * 执行adb shell && cd proc && echo 12345 > tets_rw,即可把12345写入节点test_rw
 ************************/

static ssize_t test_proc_write(struct file *file, const char __user *buffer,
										size_t count, loff_t *f_pos) 
{
	
if(count <= 0)
		return -EFAULT;
	printk("---start write---\n");
	
len = count > BUFSIZE ? BUFSIZE : count;
 
// kfree memory by kmalloc before
	if(buf != NULL)
		kfree(buf);
	buf = (char*)kmalloc(len+1, GFP_KERNEL);
	if(buf == NULL)
	{
		printk("test_proc_create kmalloc fail!\n");
		return -EFAULT;
	}
 
//memset(buf, 0, sizeof(buf));
	memset(buf, 0, len+1);
 
if(copy_from_user(buf, buffer, len))
		return -EFAULT;
	printk("test_proc_write writing :%s",buf);
	return len;
}

static struct file_operations test_fops = {
	.owner	= THIS_MODULE,
	.open	= test_proc_open,
//	.release = single_release,
	.read	= test_proc_read,
//	.llseek	= seq_lseek,
	.write 	= test_proc_write,
};

static int __init test_init(void)
{
	struct proc_dir_entry* file;
 
//创建proc文件并关联file_operations
	file = proc_create("test_rw", 0644, NULL, &test_fops);
	if (!file)
	    return -ENOMEM;
	printk("test_rw init success!\n");
	return 0;
}
 
static void __exit test_exit(void)
{
	remove_proc_entry("test_rw", NULL);
	printk("test_exit\n");

}

module_init(test_init);
module_exit(test_exit);

MODULE_AUTHOR("caizd");
MODULE_DESCRIPTION("Proc_create Test Driver");
MODULE_LICENSE("GPL");

代码编进kernel之后,可以在adb工具验证是否可行。

C:\Users\asus>adb shell
root@inwatch_portal:/ # cd proc
cd proc
root@inwatch_portal:/proc # echo 12345 > test_rw
echo 12345 > test_rw
root@inwatch_portal:/proc # cat test_rw
cat test_rw
12345
root@inwatch_portal:/proc #

---------- 爱生活,爱安卓,爱Linux ----------

  • 1
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

零意-

您的打赏将是我继续创作的动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值