linux mmap驱动实现

时间:2022-09-05 13:59:55

在实现驱动程序的mmap函数时,要注意映射地址的转换问题,见代码。

 

定义一个设备结构体:

  1. struct leedriver  
  2. {  
  3.     struct cdev cdev;  
  4.     unsigned char mem[MEMSIZE];  
  5. };  

这里面这个MEMSIZE,最小都要是4096,因为内存映射是以页为单位的。

在实现simple_remap_mmap函数时,代码如下

  1. static int simple_remap_mmap(struct file *filp, struct vm_area_struct *vma)  
  2. {  
  3.     vma->vm_pgoff = ((u32)virt_to_phys(leedriverp->mem)) >> PAGE_SHIFT;  
  4.     printk("vma->vm_pgoff = %lx/n",vma->vm_pgoff);  
  5.     if (remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff,  
  6.                 vma->vm_end - vma->vm_start,  
  7.                 vma->vm_page_prot))  
  8.         return -EAGAIN;  
  9.     vma->vm_ops = &simple_remap_vm_ops;  
  10.     simple_vma_open(vma);  
  11.     return 0;  
  12. }  

注意函数开头的vma->vm_pgoff = ((u32)virt_to_phys(leedriverp->mem)) >> PAGE_SHIFT;

leedriverp->mem是个内核虚拟地址,应该通过virt_to_phys转换成物理地址,再进行左移操作,就OK了。

之后定义一下vm_ops

[c-sharp] view plain copy
  1. static struct vm_operations_struct simple_remap_vm_ops = {  
  2.     .open =  simple_vma_open,  
  3.     .close = simple_vma_close,  
  4. };  

并且在file_operations中添加.mmap=simple_remap_mmap,整个过程就完成了