ubuntu/linux mint 创建proc文件

18 篇文章 0 订阅

在做内核驱动开发的时候,可以使用/proc下的文件,获取相应的信息,以便调试。

大多数/proc下的文件是只读的,但为了示例的完整性,都提供了写方法。

方法一:使用create_proc_entry创建proc文件(简单,但写操作有缓冲区溢出的危险);

方法二:使用proc_create和seq_file创建proc文件(较方法三简洁);

方法三:使用proc_create_data和seq_file创建proc文件(较麻烦,但比较完整);

示例四:在proc文件中使用内核链表的一个示例(用的方法三)。

--------------------------------------------------------------------------------------------------------------------

三、

proc_test03.c 源码
 

#include <linux/module.h>
#include <linux/sched.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/uaccess.h>
#include <linux/slab.h>
 
// global var
static char *str = NULL;
 
// linux/seq_file.h
// void * (*start) (struct seq_file *m, loff_t *pos);
// void (*stop) (struct seq_file *m, void *v);
// void * (*next) (struct seq_file *m, void *v, loff_t *pos);
// int (*show) (struct seq_file *m, void *v);
 
/**
* author:  aran
* fuction: seq_operations -> start
*/
static void *my_seq_start(struct seq_file *m, loff_t *pos)
{
	if (0 == *pos)
	{
		++*pos;
		return (void *)1; // return anything but NULL, just for test
	}
	return NULL;
}
 
/**
* author:  aran
* fuction: seq_operations -> next
*/
static void *my_seq_next(struct seq_file *m, void *v, loff_t *pos)
{
	// only once, so no next
	return NULL;
}
 
/**
* author:  aran
* fuction: seq_operations -> stop
*/
static void my_seq_stop(struct seq_file *m, void *v)
{
	// clean sth.
	// nothing to do
}
 
/**
* author:  aran
* fuction: seq_operations -> show
*/
static int my_seq_show(struct seq_file *m, void *v)
{
	seq_printf(m, "current kernel time is %llu\n", (unsigned long long) get_jiffies_64());
	seq_printf(m, "str is %s\n", str);
 
	return 0; //!! must be 0, or will show nothing T.T
}
 
// global var
static struct seq_operations my_seq_fops = 
{
	.start	= my_seq_start,
	.next	= my_seq_next,
	.stop	= my_seq_stop,
	.show	= my_seq_show,
};
 
// file_operations
// int (*open) (struct inode *, struct file *)
// ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *)
 
/**
* author:  aran
* fuction: file_operations -> open
*/
static int proc_seq_open(struct inode *inode, struct file *file)
{
	return seq_open(file, &my_seq_fops);
}
 
/**
* author:  aran
* fuction: file_operations -> write
*/
static ssize_t proc_seq_write(struct file *file, const char __user *buffer, size_t count, loff_t *f_pos)
{
	//分配临时缓冲区
	char *tmp = kzalloc((count+1), GFP_KERNEL);
	if (!tmp)
		return -ENOMEM;
 
	//将用户态write的字符串拷贝到内核空间
	//copy_to|from_user(to,from,cnt)
	if (copy_from_user(tmp, buffer, count)) {
		kfree(tmp);
		return -EFAULT;
	}
 
	//将str的旧空间释放,然后将tmp赋值给str
	kfree(str);
	str = tmp;
 
	return count;
}
 
// global var
static struct file_operations proc_seq_fops = 
{
	.owner		= THIS_MODULE,
	.open		= proc_seq_open,
	.read		= seq_read,
	.write		= proc_seq_write,
	.llseek		= seq_lseek,
	.release	= seq_release,
};
 
static int __init my_init(void)
{
	struct proc_dir_entry *file;
 
	// create "/proc/proc_seq" file
	file = proc_create_data(
		"jif",		// name
		0666,		// mode
		NULL,		// parent dir_entry
		&proc_seq_fops,	// file_operations
		NULL		// data
		);
	if (NULL == file)
	{
		printk("Count not create /proc/jif file!\n");
		return -ENOMEM;
	}
 
	return 0;
}
 
static void __exit my_exit(void)
{
	remove_proc_entry("jif", NULL);
	kfree(str);
}
 
module_init(my_init);
module_exit(my_exit);
 
MODULE_AUTHOR("aran");
MODULE_LICENSE("GPL");

Makefile文件:

obj-m	:= proc_test03.o
KERNEL	:= /lib/modules/`uname -r`/build #for mint/ubuntu
#KERNEL	:= /lib/modules/`uname -r`/source #for redhat
 
all:
	make -C $(KERNEL) M=`pwd` modules
 
install:
	make -C $(KERNEL) M=`pwd` modules_install
	depmod -A
 
clean:
	make -C $(KERNEL) M=`pwd` clean

 
static int Touch_proc_open(struct inode *inode, struct file *file)
{
    LOGD("touch_proc_open");
	return 0;
}
 
static int Touch_proc_release(struct inode *inode, struct file *file)
{
    LOGD("Touch_proc_release");
	return 0;
}
 
static ssize_t Touch_proc_write(struct file *file, const char __user *data,
						size_t len, loff_t *ppos)
{
    LOGD("touch proc write");
    if(0 != __copy_from_user((void *)touch_proc_str, data, len))
    {
        LOGE("copy user buffer error");
        return -EPERM;
    }
}


/*
 *	Kernel Interfaces
 */
 
static const struct file_operations Touch_proc_fops = {
	.owner	 = THIS_MODULE,
	.open    = Touch_proc_open,
	.write   = Touch_proc_write,
	.release = Touch_proc_release,
};

    
    	if(NULL == proc_create(TOUCH_PROC_NAME, 0, NULL, &Touch_proc_fops))
    {
        LOGE("create touch porc error");
        return -EBUSY;
    }


static ssize_t gt91xx_config_read_proc(struct file *file, char __user *page, size_t size, loff_t *ppos)
{
    *ppos += ptr - kmem;
    copy_to_user(page, kmem, ptr - kmem);
    kfree(kmem);
    return (ptr - kmem);
}
 
static ssize_t gt91xx_config_write_proc(struct file *filp, const char __user *buffer, size_t count, loff_t *off)
{
    s32 ret = 0;
 
    GTP_DEBUG("write count %d\n", count);
    if (copy_from_user(&config[2], buffer, count))
    {
        GTP_ERROR("copy from user fail\n");
        return -EFAULT;
    }
    return count;
}
static const struct file_operations config_proc_ops = {
    .owner = THIS_MODULE,
    .read = gt91xx_config_read_proc,
    .write = gt91xx_config_write_proc,
};
// Create proc file system
gt91xx_config_proc = proc_create(GT91XX_CONFIG_PROC_FILE, 0666, NULL, &config_proc_ops);
if (gt91xx_config_proc == NULL)
{
    GTP_ERROR("create_proc_entry %s failed\n", GT91XX_CONFIG_PROC_FILE);
}
else
{
    GTP_INFO("create proc entry %s success", GT91XX_CONFIG_PROC_FILE);
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值