成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux虛擬內存地址轉化成物理內存地址

系統 Linux
多核共享同一個地址空間也有個弊端,就是如果程序有問題(野指針,數組越界)可能會寫別的core管理的內存空間,這樣給我們帶來的問題就是程序的值莫名其妙的被改變了。我們為了排查這種問題,才考慮把應用程序的虛擬地址轉化為物理地址,進行print debug以便于統一分析。

[[197961]]

背景

現代手機這種SOC(system on chip),因為功耗、Modem等功能soc上集成了很多core,他們還可以是獨立的系統在運轉。

比如ADSP簡介ADSP(Application Digital Signal Processing)就是高通的Hexagon DSP ,就是獨立運轉的一個core+system。這樣做不僅可以使用soc上的專用核處理專業的事情,比如上面說的ADSP就可以處理音頻解碼,當然它的DSP特性還可以處理sensor融合算法,比起通用處理器(cortex a72 a53 a17 a9 a8這些核)處理效率更高,更省電。

當然出于成本因素我們不會為它單獨焊上一個內存顆粒,它共享了主存的一部分,比如從地址0xc0000000 - 0xc0100000 1MB的空間,此時內核(Linux運行在通用處理器上)將不再觸碰這塊內存。

但是多核共享同一個地址空間也有個弊端,就是如果程序有問題(野指針,數組越界)可能會寫別的core管理的內存空間,這樣給我們帶來的問題就是程序的值莫名其妙的被改變了。我們為了排查這種問題,才考慮把應用程序的虛擬地址轉化為物理地址,進行print debug以便于統一分析。

實現

kernel 在2.6.25的時候加入了這樣一個功能/proc/self/pagemap 也就是在每個進程的/proc里面都有一個pagemap通過讀取里面的內容就可以算出當前虛擬地址對應的物理頁,然后加入page_offset就可以知道當前虛擬地址對應的物理地址。

pagemap需要你的應用有root權限才能使用。

  1. #include <errno.h> 
  2.  
  3. #include <stdio.h> 
  4.  
  5. #include <sys/stat.h> 
  6.  
  7. #include <string.h> 
  8.  
  9. #include <fcntl.h> 
  10.  
  11. #include <stdlib.h> 
  12.  
  13. #include <stdint.h> 
  14.  
  15. #include <sys/types.h> 
  16.  
  17. #include <sys/stat.h> 
  18.  
  19. #include <fcntl.h> 
  20.  
  21. #include <unistd.h> 
  22.  
  23. #include <sys/mman.h> 
  24.  
  25. // 參考 
  26.  
  27. // https://www.kernel.org/doc/Documentation/vm/pagemap.txt 
  28.  
  29. #define    page_map_file     "/proc/self/pagemap" 
  30.  
  31. #define    PFN_MASK          ((((uint64_t)1)<<55)-1) 
  32.  
  33. #define    PFN_PRESENT_FLAG  (((uint64_t)1)<<63) 
  34.  
  35. int mem_addr_vir2phy(unsigned long vir, unsigned long *phy) 
  36.  
  37.  
  38. int fd; 
  39.  
  40. int page_size=getpagesize(); 
  41.  
  42. unsigned long vir_page_idx = vir/page_size; 
  43.  
  44. unsigned long pfn_item_offset = vir_page_idx*sizeof(uint64_t); 
  45.  
  46. uint64_t pfn_item; 
  47.  
  48. fd = open(page_map_file, O_RDONLY); 
  49.  
  50. if (fd<0) 
  51.  
  52.  
  53. fprintf(stderr, "open %s failed", page_map_file); 
  54.  
  55. return -1; 
  56.  
  57.  
  58. if ((off_t)-1 == lseek(fd, pfn_item_offset, SEEK_SET)) 
  59.  
  60.  
  61. fprintf(stderr, "lseek %s failed", page_map_file); 
  62.  
  63. return -1; 
  64.  
  65.  
  66. if (sizeof(uint64_t) != read(fd, &pfn_item, sizeof(uint64_t))) 
  67.  
  68.  
  69. fprintf(stderr, "read %s failed", page_map_file); 
  70.  
  71. return -1; 
  72.  
  73.  
  74. if (0==(pfn_item & PFN_PRESENT_FLAG)) 
  75.  
  76.  
  77. fprintf(stderr, "page is not present"); 
  78.  
  79. return -1; 
  80.  
  81.  
  82. *phy = (pfn_item & PFN_MASK)*page_size + vir % page_size; 
  83.  
  84. return 0; 
  85.  
  86.  
  87. int main(int argc, char* argv[]) { 
  88.  
  89. unsigned long a = 0xffbbccaa; 
  90.  
  91. unsigned long vir = reinterpret_cast<unsigned long>(&a); 
  92.  
  93. unsigned long phy = 0; 
  94.  
  95. fprintf(stderr, "sizeof(unsigned long):%lu, sizeof(unsigned long*):%lu\n", sizeof(unsigned long), sizeof(unsigned long*)); 
  96.  
  97. mem_addr_vir2phy(vir, &phy); 
  98.  
  99. fprintf(stderr, "1 vir:0x%lx, phy: 0x%lx getchar to continue\n", vir, phy); 
  100.  
  101. getchar(); 
  102.  
  103. a = 0x11111111; 
  104.  
  105. fprintf(stderr, "2 vir:0x%lx, phy: 0x%lx getchar to continue\n", vir, phy); 
  106.  
  107. getchar(); 
  108.  
  109. fprintf(stderr, "3 vir:0x%lx, phy: 0x%lx a:0x%lx\n", vir, phy, a); 
  110.  

 

如何驗證

你需要開啟kernel如下模塊

CONFIG_DEVMEM=y

關閉如下模塊

CONFIG_STRICT_DEVMEM=n

一般的Android 都有/system/bin/r(源碼在system/core/toolbox/r.c)這個命令,這個命令類似devmem之類的嵌入式工具,通過/dev/mem(物理內存)mmap來讀取物理內存的值,當然你也可以修改該地址的值

上面的例子他們通過getchar() 阻止程序的運行,以便你有足夠的時間來敲/system/bin/r命令和參數

命令用法,上面的例子我們取了一個棧上變量的虛擬地址,轉換成物理地址。然后你就可以通過/system/bin/r來讀取和修改這個地址的值了。

讀取0x9a6f0b20地址的值

  1. adb shell /system/bin/r 0x9a6f0b20 

修改0x9a6f0b20地址的值為0xffbbccaa

  1. adb shell /system/bin/r 0x9a6f0b20 0xffbbccaa 

源碼可以直接git clone git@github.com:green130181/kernel-study.git

工程里的 pagemap直接拷貝到aosp的任意目錄

然后aosp的根目錄執行

  1. source build/envsetup.sh 
  2.  
  3. lunch "your select" 
  4.  
  5. cd pagemap dir 
  6.  
  7. mm 

 

之后adb push 到你的機器,即可開始驗證。

當然還有很多先進的比如ramdump Trace32來實現內存地址查看,不過上面的對于一個應用來講足夠輕量級,夠用就好! 

 

責任編輯:龐桂玉 來源: 嵌入式Linux中文站
相關推薦

2013-08-05 15:44:36

C語言基礎

2021-04-27 13:56:49

內存.映射地址

2021-10-06 20:23:08

Linux共享內存

2019-12-26 08:45:46

Linux虛擬內存

2022-10-24 08:48:07

虛擬內存Linux

2023-10-18 13:25:00

操作系統進程

2020-11-11 08:25:45

虛擬內存模型

2010-06-10 17:12:23

Linux 內存監控

2020-04-14 16:03:31

Linux虛擬內存操作系統

2010-06-02 11:33:26

Linux 內存監控

2009-10-19 09:45:06

linux內存存管理

2019-03-20 14:29:46

Linux虛擬內存

2021-06-01 08:20:55

Linux虛擬內存命令

2022-08-21 16:52:27

Linux虛擬內存

2019-03-14 09:29:02

Linux系統內存

2022-08-02 09:02:17

虛擬內存操作系統

2015-09-29 08:51:59

內存地址主引導

2010-06-02 12:47:12

Linux 內存監控

2014-01-14 10:52:06

Linux vmsta虛擬內存

2019-07-10 12:40:29

Linux虛擬地址空間物理地址空間
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文在线一区二区 | 国产精品欧美一区喷水 | 99re在线 | 久久这里只有精品首页 | 一区二区三区成人 | 韩国av影院 | 欧美乱操 | 麻豆视频国产在线观看 | 国产精品久久久久久一区二区三区 | 中文字幕在线精品 | 日韩国产中文字幕 | 91中文视频 | 亚洲国产一区二区三区四区 | 国产精品国产精品国产专区不蜜 | 国产一区 | 久久久久无码国产精品一区 | 欧美日韩亚洲国产综合 | 激情免费视频 | 看毛片的网站 | 久久久国产一区 | 欧美亚洲另类丝袜综合网动图 | 狠狠操在线 | 日韩av电影在线观看 | 日韩精品专区在线影院重磅 | 亚洲国产视频一区二区 | 国产一区二区三区四区五区3d | 夜操| 国产精品夜夜夜一区二区三区尤 | 亚洲一区国产 | 国产一区二区 | 久久久国产一区二区三区 | 人操人免费视频 | 亚洲欧美日韩在线一区二区 | 亚洲成人自拍网 | 欧美日韩三级视频 | 99精品一区二区三区 | 亚洲精品久久 | 日韩高清国产一区在线 | 欧美日韩第一页 | 麻豆久久久| 国产精品欧美一区二区 |