arm64架构的linux 配置vm_page_prot方式

在 ARM64 架构上,通过 `vm_page_prot` 属性可以修改 `UIO` 映射内存的访问权限及缓存策略,常见的有非缓存(Non-cached)、写合并(Write Combine)等。下面是 ARM64 常用的 `vm_page_prot` 设置及其对应的操作方式。

1. **非缓存模式 (`pgprot_noncached`)**

ARM64 系统通常使用 `pgprot_noncached` 来设置非缓存模式,适用于访问要求严格顺序且不使用缓存的设备内存,例如 MMIO 区域。

用法

```c

vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);

```

此设置确保所有访问都是直接进行,不经过 CPU 缓存,以避免数据不一致的问题。

2. **写合并模式 (`pgprot_writecombine`)**

对于一些需要快速写入、但不强制严格顺序的设备内存(如帧缓冲区),可以使用 `pgprot_writecombine` 设置写合并模式。在写合并模式下,写操作会被合并以提高性能,但数据写入顺序不保证严格一致。

用法

```c

vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);

```

写合并模式适合需要高效写入的数据区域,但在读写时要注意顺序一致性问题。

3. **默认缓存模式(`pgprot_cached`)**

在 ARM64 上,`pgprot_cached` 是内存的默认缓存模式,适合普通的内存映射,不适合 MMIO 等不要求缓存的设备。通常情况下,`UIO` 映射设备内存时不使用此模式,但有些驱动需要时也可以指定:

```c

vma->vm_page_prot = pgprot_cached(vma->vm_page_prot);

```

4. **完全禁用缓存和写缓冲**

在一些特殊情况下,可能需要完全禁用缓存和写缓冲,这可以通过 `pgprot_device` 来实现。这种模式下,所有的读写顺序和数据完整性都严格受控,适合敏感的硬件寄存器访问。

用法

```c

vma->vm_page_prot = pgprot_device(vma->vm_page_prot);

```

示例:ARM64 `UIO` 驱动中配置 `vm_page_prot`

下面是一个在 `UIO` 驱动 `mmap` 函数中,根据设备需要设置 `vm_page_prot` 属性的示例:

```c

static int my_uio_mmap(struct uio_info *info, struct vm_area_struct *vma)

{

// 设置非缓存模式,适用于 MMIO 区域

vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);

// 配置权限

vma->vm_flags |= (VM_READ | VM_WRITE);

// 映射物理地址

if (remap_pfn_range(vma, vma->vm_start,

info->mem[0].addr >> PAGE_SHIFT,

vma->vm_end - vma->vm_start,

vma->vm_page_prot)) {

return -EAGAIN;

}

return 0;

}

```

注意事项

  • **根据设备需求选择合适的模式**:例如,对于顺序要求严格的设备寄存器,使用非缓存模式(`pgprot_noncached`)或设备模式(`pgprot_device`)。

  • **确保页面对齐**:`remap_pfn_range` 中的物理地址和大小应按页面大小对齐,以避免 `bus error`。

  • **调整权限和缓存模式**:根据具体需求配置 `vm_page_prot`,以满足设备内存的访问需求。

相关推荐
7yewh3 小时前
嵌入式硬件杂谈(四)-高速板PCB设计 高速信号全面讲解 蛇形线 等长线 差分对 阻抗对
驱动开发·嵌入式硬件·mcu·物联网·硬件工程·pcb工艺·精益工程
lishing616 小时前
Linux驱动开发(9):pinctrl子系统和gpio子系统--led实验
linux·运维·驱动开发
lishing616 小时前
Linux驱动开发(7):使用设备树实现RGB 灯驱动
linux·驱动开发
TeYiToKu20 小时前
笔记整理—linux驱动开发部分(13)块设备
linux·c语言·驱动开发·笔记·嵌入式硬件·arm
网易独家音乐人Mike Zhou21 小时前
【Linux驱动开发】irq中断配置API及中断应用 阻塞休眠和非阻塞的驱动操作
linux·c语言·驱动开发·stm32·单片机·mcu·iot
believe、悠闲1 天前
GetVolumeInformation函数使用记录
c++·windows·驱动开发
不怕犯错,就怕不做1 天前
修复kernel编译栈帧大小异常问题error: the frame size of 1928 bytes is larger than 1024 bytes
linux·arm开发·驱动开发
bigbig猩猩1 天前
Gin 框架中的表单处理与数据绑定
驱动开发·gin
rayylee2 天前
kvm-dmesg:从宿主机窥探虚拟机内核dmesg日志
linux·运维·服务器·驱动开发·后端·安全
白书宇3 天前
13.100ASK_T113-PRO RTC实验
linux·arm开发·驱动开发·嵌入式硬件·物联网·硬件工程