编写字符设备驱动实现内核态与用户态通信

本次主要实现的是通过编写字符设备,实现从用户态获得只有内核太才有权限访问的进程段地址(比如代码段,其他同理)

初稿主要代码如下:

firstdriver.c

 

/*字符注册模块实现如下*/
#include<linux/slab.h>
#include<linux/module.h>
#include<linux/init.h>
#include<linux/types.h>
#include<linux/errno.h>
#include<linux/fs.h>
#include<linux/mm.h>
#include<linux/cdev.h>
#include<asm/io.h>
#include<linux/sched.h>
#include<asm/uaccess.h>
#include<linux/kernel.h>
#include<linux/list.h>
#include<linux/sem.h>
#include<linux/pid.h>


#define MAJOR_NUM 260
#define MINOR_NUM 0
#define MINIOR_NUM 0
#define DEVICE_NUM 1


//以下为驱动需要实现的四个函数的生命,注册之后,用户态程序即调用此对应函数
static int firstdriver_open(struct inode *, struct file *);
static int firstdriver_release(struct inode *,struct file *);
static ssize_t firstdriver_read(struct file *, char __user *,size_t, loff_t *);
static ssize_t firstdriver_write(struct file *,char __user *,size_t ,loff_t *);
struct ret_type   //定义返回用户态的函数类型
{
        unsigned long start_code;
        unsigned long end_code;
};

static int firstdriver_major = MAJOR_NUM;  //定义主设备号
static int firstdriver_minor = MINOR_NUM;  //定义次设备号
struct file_operations firstdriver_fops =
        {
                .owner = THIS_MODULE,
                .open = firstdriver_open,
                .write = firstdriver_write,
                .release = firstdriver_release,
                .read = firstdriver_read,
        };
struct cdev *cdev;
static pid_t pidno = 1;
static int __init firstdriver_init(void)
{
                int ret = 0;
                dev_t devno = MKDEV(MAJOR_NUM,MINOR_NUM);
                cdev = cdev_alloc();//分配设备空间,这里为了简单,静态指定了设备号,最好动态申请
        if(register_chrdev_region(devno,DEVICE_NUM,"firstdriver"))
        {
                printk(KERN_ALERT"register_chrdev_region fail\n");
                return -1;
        }
        else
        {
                cdev_init(cdev,&firstdriver_fops);
                cdev->owner = THIS_MODULE;
                cdev->ops = &firstdriver_fops;
                if((ret = cdev_add(cdev,devno,1)))
                        printk(KERN_ALERT"Error in adding firstdriver\n");
                else
                        printk(KERN_ALERT"register success\n");
        }
        return ret;
}
static void __exit firstdriver_exit(void)
{
        dev_t devno = MKDEV(MAJOR_NUM,0);
        cdev_del(cdev);
        unregister_chrdev_region(devno,1);
        if(cdev)
                kfree(cdev);
        cdev = NULL;
}
static int firstdriver_open(struct inode * inode ,struct file * filp)
{
        printk(KERN_ALERT"open succedd\n");
        return 0;
}
static int firstdriver_release(struct inode* inode, struct file *filp)
{
        printk(KERN_ALERT"release success\n");
        return 0;
}
static ssize_t firstdriver_read(struct file *filp,char *buf,size_t len, loff_t * off)
{
        printk("READING:\n");
        struct task_struct *p = NULL;
        struct ret_type *tmp;
        struct pid *kpid = find_get_pid(pidno);
        tmp = kmalloc(sizeof(*tmp),GFP_KERNEL);
        if(tmp == NULL){
                return -ENOMEM;
        }
        p = pid_task(kpid,PIDTYPE_PID);
        if(p == NULL){
  kfree(tmp);
                printk(KERN_ALERT"find task failed\n");
                return -1;
        }
        if(p->mm == NULL){
                kfree(tmp);
                printk(KERN_ALERT"mm = NULL error\n");
                return -1;
        } else {
                tmp->start_code = p->mm->start_code;
                tmp->end_code = p->mm->end_code;
                printk(KERN_ALERT"start_code: %lu\n",tmp->start_code);
                printk(KERN_ALERT"end_code: %li\n",tmp->end_code);
        }
        if(copy_to_user(buf,tmp,sizeof(*tmp)))
                return -EFAULT;
        return sizeof(*tmp);
}
static ssize_t firstdriver_write(struct file *filp,char * buf,size_t len,loff_t * off)
{
        printk("writing:\n");
        if(copy_from_user(&pidno,buf,sizeof(int)))
                return -EFAULT;
        return sizeof(int);
}
module_init(firstdriver_init);
module_exit(firstdriver_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("firstdriver");
 


test.c

 

 

#include<stdlib.h>
#include<unistd.h>
#include<stdio.h>
#include<fcntl.h>
#include<malloc.h>
struct ret_type
{
        unsigned long start_code;
        unsigned long end_code;
};
int main(int argc,char *argv[])
{
        struct ret_type *tmp;
        tmp = (struct ret_type *)malloc(sizeof(struct ret_type));
        if(tmp == NULL)
        {
                printf("malloc error\n");
                return -1;
        }
        int fd,num;
        num = atoi(argv[1]);
        printf("num is %d\n",num);

        fd = open("/dev/firstdriver",O_RDWR,S_IRUSR|S_IWUSR);
        if(fd != -1){
                int n;
                n = write(fd,&num,sizeof(int));
                if(n != sizeof(int)){
                        printf("write error\n");
                        goto  out;
                }
                n = read(fd,tmp,sizeof(*tmp));
                if(n != sizeof(*tmp)){
                        printf("read error, n =  %d\n",n);
                        goto out;
  }
                n = read(fd,tmp,sizeof(*tmp));
                if(n != sizeof(*tmp)){
                        printf("read error, n =  %d\n",n);
                        goto out;
                }
                printf("start_code is  %lu \n",tmp->start_code);
                printf("end_code is  %lu \n",tmp->end_code);

                close(fd);
        }
        else
                printf("device open failed \n");

        out:free(tmp);
        return 0;
}

getchar.c

 

 

#include<stdio.h>
int main()
{
        getchar();
        return 0;
}

 

 

 

以超级用户身份执行以下事项:
1.make 生成模块文件,firstdriver.ko;

2.insmod firstdriver.ko;

3.cd /dev,mknod  firstdriver  c 260 0

4.gcc -o getchar getchar.c

5../getchar &

6.gcc -o test test.c

7../test (5生成的后台运行的进程号)

8.可看到结果如下:

 start_code: 4194304
 end_code: 4196284
 

 

参考:

1.http://www.360doc.com/content/14/0123/19/14451193_347404829.shtml

2.http://blog.chinaunix.net/uid-20662363-id-1904086.html

3.http://blog.csdn.net/rainbolide/article/details/7335888

4.http://www.360doc.com/content/10/1130/20/1378815_73825191.shtml

5.http://www.360doc.com/content/12/0901/16/10588621_233591860.shtml

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值