Linux 內核中操作寄存器
其實對於 Linux 內核中操作寄存器是一件特別簡單的事情,首先你需要知道你要操作寄存器的地址,有可能是網卡狀態寄存器,有可能是門鈴寄存器等等,不管是什么寄存器他都是在內存中映射出來的一段地址。
#include <linux/module.h> #include <linux/init.h> #include <linux/io.h> static int __init hello_init(void) { /* ioremap 由物理地址轉換為虛擬地址 */ unsigned long register_address = (unsigned long)ioremap(0x800286020c4, 4); /* 操作該寄存器 */ *(unsigned int *)register_address = 0x1000; /* 讀取該寄存器 */ pr_err("*register_address = %x\n", *(unsigned int *)register_address); return 0; } static void __exit hello_exit(void) { printk(KERN_ALERT "Bye man, I have been unload.\n"); } module_init(hello_init); module_exit(hello_exit); MODULE_AUTHOR("Jackie Liu <liuyun01@kylinos.cn>"); MODULE_LICENSE("GPL");
Linux 應用層操作寄存器
應用層操作寄存器首先需要將內核映射到核外空間,內核已經提供了一個 /dev/mem
的文件接口,這個文件相當於整個系統內存所在,將該文件打開然后指定需要映射的內存的位置即可。需要注意的是,一般 mmap 映射的是一段內存地址空間,譬如一段 PCIe 寄存器空間等等,而不是一個單獨的寄存器地址
#include <stdio.h> #include <stdlib.h> #include <time.h> #include <unistd.h> #include <fcntl.h> #include <unistd.h> #include <sys/mman.h> #define AUDIO_REG_BASE (0x80028780000) #define MAP_SIZE 0x80000 static int dev_fd; int main(int argc, char **argv) { dev_fd = open("/dev/mem", O_RDWR | O_NDELAY); if (dev_fd < 0) { printf("open(/dev/mem) failed."); return 0; } unsigned char *map_base=(unsigned char * )mmap(NULL, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, dev_fd, AUDIO_REG_BASE ); if (map_base == MAP_FAILED) return -1; printf("%x \n", *(volatile unsigned int *)(map_base)); printf("%x \n", *(volatile unsigned int *)(map_base+0x38)); //打印該寄存器地址的value #if 1 // LINE IN printf("%x \n", *(volatile unsigned int *)(map_base+0x30)); *(volatile unsigned int *)(map_base + 0x30) = 0x208121bc; //修改該寄存器地址的value usleep(1000000); *(volatile unsigned int *)(map_base + 0x30) &= ~(0x1<<16); //修改該寄存器地址的value usleep(1000000); printf("%x \n", *(volatile unsigned int *)(map_base+0x30)); #endif if(dev_fd) close(dev_fd); munmap(map_base,MAP_SIZE);//解除映射關系 return 0; }