忘记密码   免费注册 查看新帖 | 论坛精华区
ChinaUnix.net
  平台论坛 博客 微博 读书 人才 精华 文库 自测 | 频道操作系统 开发 数据库 存储 服务器 网络 IT新闻 Linux 下载 Power用户组
最近访问板块 发新帖
查看: 10716 | 回复: 30

请教Godbach版主(mmap用户态和内核态共享内存) [复制链接]
更多

帖子
2838
主题
668
精华
9
可用积分
41
专家积分
100
在线时间
1600 小时
注册时间
2008-08-15
最后登录
2011-12-01
论坛徽章:
0
发表于 2009-01-01 20:05:26 |显示全部楼层
拜读了你的帖子
    http://linux.chinaunix.net/bbs/thread-1034670-1-5.html
我在我自己电脑上尝试了下:
   发现没了create_proc_info_entry这个函数,我自己修改了下代码如下:可以make 也可以insmod但是运行用户态程序时

./mmap_user 'hello'
phymem_addr=0, phymem_size=0
Segmentation fault

dmesg的信息:
[18065.048108] mmap_user[11329]: segfault at ffffffff ip b7ed4da4 sp bfaec690 error 6 in libc-2.8.90.so[b7e5e000+158000]
[18068.692529] mmap_user[11330]: segfault at ffffffff ip b7e6fda4 sp bfe89a30 error 6 in libc-2.8.90.so[b7df9000+158000]


#include <linux/version.h>
#include <linux/module.h>
#include <linux/proc_fs.h>
#include <linux/mm.h>

#define PROC_MEMSHARE_DIR               "memshare"
#define PROC_MEMSHARE_PHYADDR           "phymem_addr"
#define PROC_MEMSHARE_SIZE              "phymem_size"

/*alloc one page. 4096 bytes*/
#define PAGE_ORDER                                0
/*this value can get from PAGE_ORDER*/
#define PAGES_NUMBER                                1

struct proc_dir_entry *proc_memshare_dir ;
unsigned long kernel_memaddr = 0;
unsigned long kernel_memsize= 0;

/*static int proc_read_phymem_addr(char *page, char **start, off_t off, int count)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return sprintf(page, "%08lx\n", __pa(kernel_memaddr));
}
static int proc_read_phymem_size(char *page, char **start, off_t off, int count)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return sprintf(page, "%lu\n", kernel_memsize);
}*/


static int __init init(void)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/*build proc dir "memshare"and two proc files: phymem_addr, phymem_size in the dir*/
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;proc_memshare_dir = proc_mkdir(PROC_MEMSHARE_DIR, NULL);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;create_proc_entry(PROC_MEMSHARE_PHYADDR, 0644, proc_memshare_dir);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;create_proc_entry(PROC_MEMSHARE_SIZE, 0644, proc_memshare_dir);

&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/*alloc one page*/
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;kernel_memaddr =__get_free_pages(GFP_KERNEL, PAGE_ORDER);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if(!kernel_memaddr)
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("Allocate memory failure!\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SetPageReserved(virt_to_page(kernel_memaddr));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;kernel_memsize = PAGES_NUMBER * PAGE_SIZE;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("Allocate memory success!. The phy mem addr=%08lx, size=%lu\n", __pa(kernel_memaddr), kernel_memsize);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return 0;
}

static void __exit fini(void)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("The content written by user is: %s\n", (unsigned char *) kernel_memaddr);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ClearPageReserved(virt_to_page(kernel_memaddr));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;free_pages(kernel_memaddr, PAGE_ORDER);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;remove_proc_entry(PROC_MEMSHARE_PHYADDR, proc_memshare_dir);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;remove_proc_entry(PROC_MEMSHARE_SIZE, proc_memshare_dir);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;remove_proc_entry(PROC_MEMSHARE_DIR, NULL);

&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;
}
module_init(init);
module_exit(fini);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Godbach (nylzhaowei@163.com)");
MODULE_DESCRIPTION("Kernel memory share module.");


先谢谢了
http://ubuntuer.cublog.cn欢迎做做
To be 千里马!

Rank: 8Rank: 8

帖子
14516
主题
366
精华
21
可用积分
58752
专家积分
79
在线时间
5761 小时
注册时间
2007-03-09
最后登录
2012-02-09
论坛徽章:
0
发表于 2009-01-01 20:25:49 |显示全部楼层
如果不使用proc文件的话,就不要保留这三行了
        proc_memshare_dir = proc_mkdir(PROC_MEMSHARE_DIR, NULL);
        create_proc_entry(PROC_MEMSHARE_PHYADDR, 0644, proc_memshare_dir);
        create_proc_entry(PROC_MEMSHARE_SIZE, 0644, proc_memshare_dir);
----------
欢迎光临Godbach的博客交流技术问题:
Godbach's Blog
---------
明犯我强汉天威者,穷搜天下,万里追杀,覆其巢,断其苗裔,戮其身,追其魂,屠其魄,虽远必诛!
To be 千里马!

Rank: 8Rank: 8

帖子
14516
主题
366
精华
21
可用积分
58752
专家积分
79
在线时间
5761 小时
注册时间
2007-03-09
最后登录
2012-02-09
论坛徽章:
0
发表于 2009-01-01 20:28:11 |显示全部楼层
另外,insmod之后,看一下日志信息,日志信息里面应该记录了分配内存成功与否等相关信息
----------
欢迎光临Godbach的博客交流技术问题:
Godbach's Blog
---------
明犯我强汉天威者,穷搜天下,万里追杀,覆其巢,断其苗裔,戮其身,追其魂,屠其魄,虽远必诛!

帖子
2838
主题
668
精华
9
可用积分
41
专家积分
100
在线时间
1600 小时
注册时间
2008-08-15
最后登录
2011-12-01
论坛徽章:
0
发表于 2009-01-02 14:55:54 |显示全部楼层
感谢你的恢复,那不用proc后,用户态如何知道怎么调用内存呢?
用户态程序何以知道该用那块内存?

[ 本帖最后由 ubuntuer 于 2009-1-2 14:56 编辑 ]
http://ubuntuer.cublog.cn欢迎做做

帖子
2838
主题
668
精华
9
可用积分
41
专家积分
100
在线时间
1600 小时
注册时间
2008-08-15
最后登录
2011-12-01
论坛徽章:
0
发表于 2009-01-02 15:01:42 |显示全部楼层
我觉得一定有替代的函数的...
http://ubuntuer.cublog.cn欢迎做做

帖子
2838
主题
668
精华
9
可用积分
41
专家积分
100
在线时间
1600 小时
注册时间
2008-08-15
最后登录
2011-12-01
论坛徽章:
0
发表于 2009-01-02 15:42:39 |显示全部楼层
res = create_proc_entry (PROC_MEMSHARE_PHYADDR, 0, proc_memshare_dir);
&nbsp;&nbsp;if (res)
&nbsp;&nbsp;&nbsp;&nbsp;res->read_proc = proc_read_phymem_addr;

&nbsp;&nbsp;res = create_proc_entry (PROC_MEMSHARE_SIZE, 0, proc_memshare_dir);
&nbsp;&nbsp;if (res)
&nbsp;&nbsp;&nbsp;&nbsp;res->read_proc = proc_read_phymem_size;

我查了下那个函数实际上相当于read_proc
这样dmesg的信息如下:
[ 3342.728521] Allocate memory success!. The phy mem addr=37b55000, size=4096
[ 3362.064879] Program mmap_user tried to access /dev/mem between 37b55000->37b56000.
[ 3362.064916] mmap_user[8763]: segfault at ffffffff ip b7ebbda4 sp bf8d3c70 error 6 in libc-2.8.90.so[b7e45000+158000]
http://ubuntuer.cublog.cn欢迎做做
To be 千里马!

Rank: 8Rank: 8

帖子
14516
主题
366
精华
21
可用积分
58752
专家积分
79
在线时间
5761 小时
注册时间
2007-03-09
最后登录
2012-02-09
论坛徽章:
0
发表于 2009-01-02 15:47:35 |显示全部楼层
原帖由 ubuntuer 于 2009-1-2 15:01 发表
我觉得一定有替代的函数的...


应该是有替代函数的
----------
欢迎光临Godbach的博客交流技术问题:
Godbach's Blog
---------
明犯我强汉天威者,穷搜天下,万里追杀,覆其巢,断其苗裔,戮其身,追其魂,屠其魄,虽远必诛!
To be 千里马!

Rank: 8Rank: 8

帖子
14516
主题
366
精华
21
可用积分
58752
专家积分
79
在线时间
5761 小时
注册时间
2007-03-09
最后登录
2012-02-09
论坛徽章:
0
发表于 2009-01-02 15:56:24 |显示全部楼层
原帖由 ubuntuer 于 2009-1-2 14:55 发表
感谢你的恢复,那不用proc后,用户态如何知道怎么调用内存呢?
用户态程序何以知道该用那块内存?


我是看你代码中不用proc了,所以创建也就没有必要了。
本来就是需要通过一些机制把这两个值传出来了。要么使用proc,要么使用其他机制.
----------
欢迎光临Godbach的博客交流技术问题:
Godbach's Blog
---------
明犯我强汉天威者,穷搜天下,万里追杀,覆其巢,断其苗裔,戮其身,追其魂,屠其魄,虽远必诛!
To be 千里马!

Rank: 8Rank: 8

帖子
14516
主题
366
精华
21
可用积分
58752
专家积分
79
在线时间
5761 小时
注册时间
2007-03-09
最后登录
2012-02-09
论坛徽章:
0
发表于 2009-01-02 15:59:43 |显示全部楼层
res = create_proc_entry (PROC_MEMSHARE_PHYADDR, 0, proc_memshare_dir);
  if (res)
    res->read_proc = proc_read_phymem_addr;

  res = create_proc_entry (PROC_MEMSHARE_SIZE, 0, proc_memshare_dir);
  if (res)
    res->read_proc = proc_read_phymem_size;


res->read_proc是什么类型,能把proc_read_phymem_addr直接赋值吗??
----------
欢迎光临Godbach的博客交流技术问题:
Godbach's Blog
---------
明犯我强汉天威者,穷搜天下,万里追杀,覆其巢,断其苗裔,戮其身,追其魂,屠其魄,虽远必诛!

帖子
2838
主题
668
精华
9
可用积分
41
专家积分
100
在线时间
1600 小时
注册时间
2008-08-15
最后登录
2011-12-01
论坛徽章:
0
发表于 2009-01-02 16:55:05 |显示全部楼层
你看dmesg的信息可以看到感觉都是对的阿
分配的以及用户态尝试写的
http://ubuntuer.cublog.cn欢迎做做
您需要登录后才可以回帖 登录 | 注册

北京皓辰网域网络信息技术有限公司. 版权所有 京ICP证:060528号 北京市公安局海淀分局网监中心备案编号:1101082001
广播电视节目制作经营许可证(京) 字第1234号 中国互联网协会会员  联系我们:
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP