免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 7082 | 回复: 13
打印 上一主题 下一主题

[内存管理] mmap共享内存更新不及时 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-08-05 10:51 |只看该作者 |倒序浏览
使用mmap将内核空间内存映射到用户空间,对这块内存,内核来写,1ms更新一次,用户空间读,测试发现用户空间读取的数据有时会滞后内核更新的数据几个ms。
不知是否mmap参数设置不对,求助大虾帮忙。
内核代码:
  1. p = get_zeroed_page(GFP_KERNEL);
  2. static int int_mmap(struct file *filp, struct vm_area_struct *vma)
  3. {

  4.     unsigned long offset = vma->vm_pgoff << PAGE_SHIFT;
  5.     unsigned long physics = ((void *)p)-PAGE_OFFSET;
  6.     unsigned long mypfn = physics >> PAGE_SHIFT;
  7.     unsigned long vmsize = vma->vm_end-vma->vm_start;
  8.     unsigned long psize = PAGE_SIZE - offset;
  9.    
  10.     if(vmsize > psize) {
  11.         return -ENXIO;
  12.     }

  13.        
  14.         vma->vm_flags |= VM_IO | VM_SHARED;
  15.         vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);

  16.     if(remap_pfn_range(vma, vma->vm_start,  mypfn,  vmsize, vma->vm_page_prot) != 0) {
  17.         return -EAGAIN;
  18.     }   
  19.    
  20.     return   0;
  21. }
复制代码
用户空间:
  1. s32 test_init()
  2. {
  3.     fd = open(FILE_DEVICE, O_RDWR);
  4.         if (fd< 0)
  5.         {
  6.                 return OSP_ERROR;
  7.         }

  8.         p= mmap(NULL, getpagesize(),
  9.                 (u32)PROT_READ | (u32)PROT_WRITE, MAP_SHARED, fd, 0);
  10.         if(ospmmap == MAP_FAILED )
  11.         {
  12.                 close(fd);
  13.                 return -1;
  14.         }

  15.         return 0;
  16. }
复制代码

论坛徽章:
6
金牛座
日期:2013-10-08 10:19:10技术图书徽章
日期:2013-10-14 16:24:09CU十二周年纪念徽章
日期:2013-10-24 15:41:34狮子座
日期:2013-11-24 19:26:19未羊
日期:2014-01-23 15:50:002015年亚洲杯之阿联酋
日期:2015-05-09 14:36:15
2 [报告]
发表于 2013-08-05 13:12 |只看该作者
回复 1# tqyou85
使用mmap将内核空间内存映射到用户空间,对这块内存,内核来写,1ms更新一次,用户空间读,测试发现用户空间读取的数据有时会滞后内核更新的数据几个ms。
不知是否mmap参数设置不对,求助大虾帮忙。


内核空间和用户空间访问的是同一块内存,不会出现不一致的情况。之前的项目中曾经用过该机制进行用户态和内核态的数据同步,没出现过问题。

从你贴出来的代码看不出问题

论坛徽章:
0
3 [报告]
发表于 2013-08-05 14:43 |只看该作者
本帖最后由 tqyou85 于 2013-08-05 14:43 编辑

回复 2# 瀚海书香

修改了内核代码,将以下两行注释掉:
  1. vma->vm_flags |= VM_IO | VM_SHARED;
  2. vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
复制代码
问题就解决了,原因还得仔细分析。


   

论坛徽章:
16
2015亚冠之吉达阿赫利
日期:2015-08-17 11:21:462015年迎新春徽章
日期:2015-03-04 09:58:11酉鸡
日期:2014-12-07 09:06:19水瓶座
日期:2014-11-04 14:23:29天秤座
日期:2014-03-02 08:57:52双鱼座
日期:2014-02-22 13:07:56午马
日期:2014-02-14 11:08:18双鱼座
日期:2014-02-13 11:09:37卯兔
日期:2014-02-06 15:10:34子鼠
日期:2014-01-20 14:48:19戌狗
日期:2013-12-19 09:37:46射手座
日期:2013-12-19 09:33:47
4 [报告]
发表于 2013-08-05 15:18 |只看该作者
回复 3# tqyou85


只注释掉
  1. vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
复制代码
可以解决吗?

   

论坛徽章:
6
金牛座
日期:2013-10-08 10:19:10技术图书徽章
日期:2013-10-14 16:24:09CU十二周年纪念徽章
日期:2013-10-24 15:41:34狮子座
日期:2013-11-24 19:26:19未羊
日期:2014-01-23 15:50:002015年亚洲杯之阿联酋
日期:2015-05-09 14:36:15
5 [报告]
发表于 2013-08-05 16:05 |只看该作者
回复 4# embeddedlwp

按理说应该设置nocached的啊?drivers/char/mem.c中的代码
   

论坛徽章:
0
6 [报告]
发表于 2013-08-05 16:10 |只看该作者
本帖最后由 tqyou85 于 2013-08-05 16:12 编辑
embeddedlwp 发表于 2013-08-05 15:18
回复 3# tqyou85


测试了下,只注释掉vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);可以解决问题。

论坛徽章:
16
2015亚冠之吉达阿赫利
日期:2015-08-17 11:21:462015年迎新春徽章
日期:2015-03-04 09:58:11酉鸡
日期:2014-12-07 09:06:19水瓶座
日期:2014-11-04 14:23:29天秤座
日期:2014-03-02 08:57:52双鱼座
日期:2014-02-22 13:07:56午马
日期:2014-02-14 11:08:18双鱼座
日期:2014-02-13 11:09:37卯兔
日期:2014-02-06 15:10:34子鼠
日期:2014-01-20 14:48:19戌狗
日期:2013-12-19 09:37:46射手座
日期:2013-12-19 09:33:47
7 [报告]
发表于 2013-08-05 16:27 |只看该作者
回复 6# tqyou85

神马平台?
   

论坛徽章:
6
金牛座
日期:2013-10-08 10:19:10技术图书徽章
日期:2013-10-14 16:24:09CU十二周年纪念徽章
日期:2013-10-24 15:41:34狮子座
日期:2013-11-24 19:26:19未羊
日期:2014-01-23 15:50:002015年亚洲杯之阿联酋
日期:2015-05-09 14:36:15
8 [报告]
发表于 2013-08-05 16:27 |只看该作者
回复 6# tqyou85

看看这个patch的注释,x86平台下加不加应该都可以。

From: Jean-Samuel Chenard <jsamch@gmail.com>
To: Greg KH <greg@kroah.com>
Cc: Hans J Koch <hjk@linutronix.de>, linux-kernel@vger.kernel.org,
Juergen Beisert <juergen127@kreuzholzen.de>
Date: Fri, 14 Mar 2008 11:19:49 +0100
Subject: Add pgprot_noncached() to UIO mmap code

Mapping of physical memory in UIO needs pgprot_noncached() to ensure
that IO memory is not cached. Without pgprot_noncached(), it (accidentally)
works on x86 and arm, but fails on PPC.

Signed-off-by: Jean-Samuel Chenard <jsamch@gmail.com>
Signed-off-by: Hans J Koch <hjk@linutronix.de>

---
drivers/uio/uio.c |    2 ++
1 file changed, 2 insertions(+)
Index: linux-2.6.25-rc/drivers/uio/uio.c
===================================================================
--- linux-2.6.25-rc.orig/drivers/uio/uio.c        2008-03-14 11:00:59.000000000 +0100
+++ linux-2.6.25-rc/drivers/uio/uio.c        2008-03-14 11:03:13.000000000 +0100
@@ -470,6 +470,8 @@

        vma->vm_flags |= VM_IO | VM_RESERVED;

+        vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
+
        return remap_pfn_range(vma,
                               vma->vm_start,
                               idev->info->mem[mi].addr >> PAGE_SHIFT,
   

论坛徽章:
0
9 [报告]
发表于 2013-08-05 17:07 |只看该作者
本帖最后由 tqyou85 于 2013-08-05 17:09 编辑
embeddedlwp 发表于 2013-08-05 16:27
回复 6# tqyou85

神马平台?


回复 8# 瀚海书香

ppc平台

论坛徽章:
0
10 [报告]
发表于 2013-08-08 13:42 |只看该作者
你把用户态的设成了no cache,直接读物理内存。
        vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
内核直接用的dcache,肯定又一段延迟才会写到物理内存。
这样用户态自然读的延迟了。


如果 vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);把这个去掉的话。用户态和内核态都用cache,虽然可能有cache aliasing的问题(用户态cache line和内核cache line都指向的同一块物理内存),但是一般体系都会自动解决这个alasing的,比如armv7就是自动处理vipt dcache aliasing的。

您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP