Linux系统“/dev/mem”设备使用详解(Hi3520D)

2023-11-04 23:10

本文主要是介绍Linux系统“/dev/mem”设备使用详解(Hi3520D),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1 前言
  • 2 “/dev/mem”设备
    • 2.1 设备使用优点
    • 2.2 设备使用不足
    • 2.3 使用方法
  • 3 应用例子
  • 4 参考文章


1 前言

  linux系统用户态访问内核态通常有这几种方式:

  • 设备文件,“read/write/ioctl”,常用的方式
  • prcfs进程文件系统
  • sysfs虚拟文件系统,
  • 内存映射(mmap)
  • netlink socket

  本文描述的是“设备文件”与“内存映射(mmap)”的一个应用范畴,linux系统提供了一个虚拟设备“/dev/mem”,结合mmap函数,用户态可以直接访问内核物理地址空间。


2 “/dev/mem”设备

  “/dev/mem”是linux系统的一个虚拟字符设备,无论是标准linux系统还是嵌入式linux系统,都支持该设备。

在这里插入图片描述

  “/dev/mem”设备是内核所有物理地址空间的全映像,这些地址包括:

  • 物理内存(RAM)空间
  • 物理存储(ROM)空间
  • cpu总线地址
  • cpu寄存器地址
  • 外设寄存器地址,GPIO、定时器、ADC

  “/dev/mem”设备通常与“mmap”结合使用,将该设备的物理内存映射到用户态,这样用户空间可以直接访问内存态。

  因为涉及访问内核空间,因此只有root用户才有访问“/dev/mem”设备的权限。


2.1 设备使用优点

  • 用户可以直接访问内核物理空间,省略内存拷贝过程,效率高
  • 访问灵活,一般用于前期BSP、驱动、SDK调试,

2.2 设备使用不足

  由于把内核态内存访问权限直接交给用户,灵活性和高效率的同时,可能带来安全性的隐患。

  • 增加内核不稳定性,进程崩溃可能导致内核崩溃
  • 访问非法空间时可能导致内核崩溃

2.3 使用方法

  “/dev/mem”设备通常与“mmap”结合使用,将该设备的物理内存映射到用户态,在用户态直接访问内核态物理内存。

【1】第一步,open一个“/dev/mem”文件描述符,访问权限可以为只读(O_RDONLY )、只写(O_WRONLY )、读写(O_RDWR )的阻塞或者非阻塞方式。

int fd = 0;
fd = open("/dev/mem", O_RDWR | O_NDELAY);   /* 读写权限,非阻塞 */

【2】第二步,通过mmap把需访问的目标物理地址与“/dev/mem”文件描述符建立映射。

char *mmap_addr = NULL;mmap_addr=(char *)mmap(NULL, MMAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, MMAP_ADDR);  

注:
关于mmap函数的使用方法,参考“mmap内存映射”文章。


【3】第三步,地址读写访问。

int a = 0*(int*)mmap_addr = 0x10;	/* 写地址 */
a = *(int)mmap_addr;		/* 读地址 */

3 应用例子

  上一篇文章中描述“嵌入式linux下获取cpu温度的方法”;后面发现海思Hi3520DV400提供的SDK linux系统并未支持标准cpu温度读取接口。下面我们通过“/dev/mem”设备,在用户态直接读取Hi3520D的cpu温度寄存器,来获取当前cpu温度值。

  首先查阅Hi3520D的寄存器手册,理解cpu温度寄存器使用与温度读取方法。
在这里插入图片描述

Hi3520D cpu温度读取与计算方法


在这里插入图片描述

Hi3520D cpu温度控制寄存器


在这里插入图片描述

Hi3520D cpu温度数据寄存器0


  根据寄存器手册,Hi3520D cpu温度支持单次采样和循环采样,我们选择单次采样,从数据寄存器0获取温度数据值,然后换算为实际温度。实现源代码如下。

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/mman.h>#define REG_PERT_BASE_ADDR		0x120E0000	/* 寄存器基地址 */
#define REG_PERT_PMC68_ADDR 	0x110		/* 温度控制寄存器 */
#define REG_PERT_PMC70_ADDR 	0x118		/* 温度值寄存器0 */
#define REG_PERT_PMC71_ADDR 	0x11C		/* 温度值寄存器1 */
#define REG_PERT_PMC72_ADDR 	0x120		/* 温度值寄存器2 */
#define REG_PERT_PMC73_ADDR 	0x124		/* 温度值寄存器3 */#define VALUE_ENABLE_TSENSOR	(0x01<<30)	/* 使能温度转换 */
#define VALUE_DISABLE_TSENSOR	(0x0)		/* 失能温度转换 */#define MMAP_SIZE 0x1000					/* 映射内存大小,通常为一个内存页(4096)整数 */
#define MMAP_ADDR REG_PERT_BASE_ADDR		/* 映射物理地址 */int main(int argc, char *argv[])
{int fd = 0;int *preg_pmc68 = NULL;int *preg_pmc70 = NULL;char *mmap_base = NULL;int   temp = 0;fd = open("/dev/mem", O_RDWR | O_NDELAY);        if (fd < 0)    {  printf("open mem fd failed,%s\n", strerror(errno));      return -1;  }  mmap_base=(char *)mmap(NULL, MMAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, MMAP_ADDR);  if (NULL == mmap_base){printf("mmap failed,%s\n", strerror(errno));return -1;}preg_pmc68 = (int*)(mmap_base + REG_PERT_PMC68_ADDR);preg_pmc70 = (int*)(mmap_base + REG_PERT_PMC70_ADDR);for(;;){*preg_pmc68 = VALUE_ENABLE_TSENSOR;		 /* 单次转换 */*preg_pmc68 = VALUE_DISABLE_TSENSOR;temp = *preg_pmc70 & 0x3ff;				 /* 单次转换温度值存于code0 */temp = 10 * (temp-125) * 165 / 806 - 400;/* 扩大10倍取1位小数 */printf("cpu temperature: [%d.%d C]\n", temp/10, temp%10);sleep(2);	}	munmap(mmap_base, MMAP_SIZE);close(fd);return 0;
}

编译运行

  • 交叉编译
  • 通过nfs或者u盘将执行文件传输至Hi3520D板端执行
/* 编译 */
# arm-hisiv500-linux hi_temp.c -o hi_temp/* 执行 */
# ./hi_temp 
cpu temperature: [54.3 C]
cpu temperature: [54.3 C]
cpu temperature: [54.1 C]

4 参考文章

【1】/dev/mem可没那么简单

这篇关于Linux系统“/dev/mem”设备使用详解(Hi3520D)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/345821

相关文章

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

Python使用vllm处理多模态数据的预处理技巧

《Python使用vllm处理多模态数据的预处理技巧》本文深入探讨了在Python环境下使用vLLM处理多模态数据的预处理技巧,我们将从基础概念出发,详细讲解文本、图像、音频等多模态数据的预处理方法,... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

C#中Guid类使用小结

《C#中Guid类使用小结》本文主要介绍了C#中Guid类用于生成和操作128位的唯一标识符,用于数据库主键及分布式系统,支持通过NewGuid、Parse等方法生成,感兴趣的可以了解一下... 目录前言一、什么是 Guid二、生成 Guid1. 使用 Guid.NewGuid() 方法2. 从字符串创建

Python使用python-can实现合并BLF文件

《Python使用python-can实现合并BLF文件》python-can库是Python生态中专注于CAN总线通信与数据处理的强大工具,本文将使用python-can为BLF文件合并提供高效灵活... 目录一、python-can 库:CAN 数据处理的利器二、BLF 文件合并核心代码解析1. 基础合