linux下uboot kernel操作cpu寄存器

简介: 大多数的内核里面都有会对GPIO的操作,而且内核里面对GPIO进行配置也很方便,要什么功能就配置成什么就可以了。 还有一些寄存器是内核没有配置到的,但是我们要操作怎么办,内核里面也定义了相关的接口函数。

大多数的内核里面都有会对GPIO的操作,而且内核里面对GPIO进行配置也很方便,要什么功能就配置成什么就可以了。

还有一些寄存器是内核没有配置到的,但是我们要操作怎么办,内核里面也定义了相关的接口函数。

在u-boot中操作某个寄存器:

 

[cpp]  view plain  copy
 
 print?在CODE上查看代码片派生到我的代码片
  1. reg = readl(IOMUXC_BASE_ADDR + IOMUXC_REG_GPR1);  
  2. reg &= ~IOMUXC_REG_GPR1_ACTCS0_MASK;  
  3. writel(reg, IOMUXC_BASE_ADDR + IOMUXC_REG_GPR1);  

其中IOMUXC_BASE_ADDR是物理地址,跟踪代码发现writel操作如下:

 

#define writel(v,a) __arch_putl(v,a)

#define __arch_putl(v,a) (*(volatile unsigned int *)(a) = (v))

所以在uboot里面配置寄存相当于是给物理地址直接赋值,volatile的意思是提醒编译器需要存储或读取这个变量的时候,都会直接从变量地址中读取数据

 

而在内核中,上面的写法是无法运行的,会提示虚拟地址错误。在内核中通常是通过虚拟地址来给物理地址赋值,所以需要将物理地址转换成虚拟地址

 

[cpp]  view plain  copy
 
 print?在CODE上查看代码片派生到我的代码片
  1. reg = __raw_readl(ioremap(IOMUXC_BASE_ADDR + IOMUXC_REG_GPR1,4));  
  2. reg &= ~IOMUXC_REG_GPR1_ACTCS0_MASK;  
  3. reg &= ~IOMUXC_REG_GPR1_ADDRS0_MASK;  
  4. reg |= ((CS0_NORFLASH_SIZE | IOMUXC_REG_GPR1_ACTCS0));  
  5. __raw_writel(reg, ioremap(IOMUXC_BASE_ADDR + IOMUXC_REG_GPR1,4));  

这里的ioremap是将物理地址IOMUXC_BASE_ADDR转换得到对应的虚拟地址,4表示4个字节,即32位的地址。

 

u-boot下读写gpio:

与读写寄存器类似,u-boot下读写GPIO口是直接给GPIO赋值:

 

[cpp]  view plain  copy
 
 print?在CODE上查看代码片派生到我的代码片
  1. mxc_request_iomux(MX53_PIN_GPIO_8, IOMUX_CONFIG_ALT1);  
  2. mxc_iomux_set_pad(MX53_PIN_GPIO_8, 0x1E4);  
  3.   
  4. reg = readl(GPIO1_BASE_ADDR + 0x0);  
  5.    <span style="white-space:pre"> </span>reg |= 0x100;  
  6. writel(reg, GPIO1_BASE_ADDR + 0x0);  
  7.   
  8. // Set pin direction as output  
  9. reg = readl(GPIO1_BASE_ADDR + 0x4);  
  10. reg |= 0x100;  
  11. writel(reg, GPIO1_BASE_ADDR + 0x4);  


GPIO_8 是GPIO1_8,前面两个配置GPIO_8的功能。

 

查看datasheet可以看到GPIO1的地址配置

53F8_4000 GPIO data register (GPIO-1_DR) 32 R/W 0000_0000h 
53F8_4004 GPIO direction register (GPIO-1_GDIR) 32 R/W 0000_0000h 
53F8_4008 GPIO pad status register (GPIO-1_PSR) 32 R 0000_0000h 
53F8_400C GPIO interrupt configuration register1 (GPIO-1_ICR1) 32 R/W 0000_0000h 
53F8_4010 GPIO interrupt configuration register2 (GPIO-1_ICR2) 32 R/W 0000_0000h
53F8_4014 GPIO interrupt mask register (GPIO-1_IMR) 32 R/W 0000_0000h 
53F8_4018 GPIO interrupt status register (GPIO-1_ISR) 32 w1c 0000_0000h
53F8_401C GPIO edge select register (GPIO-1_EDGE_SEL) 32 R/W 0000_0000h 

 

可以看到它的数据寄存器的偏移地址是0x0,输入输出寄存器的偏移地址是0x4。而reg |= 0x100;是GPIO_8的所在的偏移,即(0x1 << 8)。

读取一个gpio的值,只需要读取它的状态寄存器就可以了,

reg = readl( GPIO1_BASE_ADDR + 0x08 );

if(reg & (0x1 << 8))

printf("it is high\n");

else

printf("it is low\n");

目录
相关文章
|
8天前
|
Linux 网络安全 开发工具
Linux 管理远程会话 screen:掌握终端的多任务操作
`Linux screen` 命令让多任务管理变得更简单,尤其在SSH连接远程服务器时。创建新会话如`screen -S backup`,查看会话`screen -ls`,退出`exit`。高级功能包括直接在会话中运行命令,如`screen vim memo.txt`,会话共享以协同工作,以及通过`screen -r`或`-D -r`重新连接或强制恢复断开的会话。提高效率,确保任务不间断运行。
7 1
|
14天前
|
Linux 数据处理
Linux中的nproc命令:轻松查看系统CPU核心数
`nproc`命令在Linux中用于查看CPU核心数,简洁高效,无参数直接运行。它读取`/proc/cpuinfo`获取信息,适用于资源分配。例如,`nproc`显示核心数,`nproc --all`(非标准选项,可能需结合其他命令)展示更多详情。在脚本中,可将`nproc`输出赋值给变量以适应动态资源管理。使用时注意文件访问权限,检查结果准确性,并结合其他工具如`lscpu`获取更全面硬件信息。
|
2天前
|
Linux 调度
部署02-我们一般接触的是Mos和Wimdows这两款操作系统,很少接触到Linux,操作系统的概述,硬件是由计算机系统中由电子和机械,光电元件所组成的,CPU,内存,硬盘,软件是用户与计算机接口之间
部署02-我们一般接触的是Mos和Wimdows这两款操作系统,很少接触到Linux,操作系统的概述,硬件是由计算机系统中由电子和机械,光电元件所组成的,CPU,内存,硬盘,软件是用户与计算机接口之间
|
4天前
|
安全 固态存储 Linux
服务器linux操作系统重装的完整流程-傻瓜式教学
服务器linux操作系统重装的完整流程-傻瓜式教学
|
9天前
|
Linux Perl
如何在Linux系统中确定CPU架构
如何在Linux系统中确定CPU架构
12 0
|
1月前
|
Linux 应用服务中间件 网络安全
linux 初始化全部操作
linux 初始化全部操作
20 1
|
18天前
|
缓存 监控 Linux
深入了解Linux的`lscpu`命令:你的CPU信息专家
`lscpu`是Linux下的命令行工具,用于获取CPU详细信息,如架构、核心、线程、缓存和型号。它从系统文件读取数据,提供实时信息,支持多种输出格式,如扩展视图、解析格式。常用参数包括显示所有CPU (`-a`)、在线CPU (`-b`) 和可解析格式 (`--parseable`)。结合其他工具,`lscpu`在系统管理和性能调优中十分有用。
|
1月前
|
Linux C++
Linux C/C++目录和文件的更多操作
Linux C/C++目录和文件的更多操作
|
1月前
|
监控 Shell Linux
shell linux中用shell写一个占用CPU的脚本
shell linux中用shell写一个占用CPU的脚本
29 0
|
1月前
|
Ubuntu Linux
服务器硬件 做raid操作 ubuntu linux做raid
服务器硬件 做raid操作 ubuntu linux做raid
21 0