地址转换
如何根据逻辑地址打印物理地址?
思路:逻辑地址===》 线性地址 虚拟地址 ====》物理地址
Linux中逻辑地址和转换的线性地址相同
因为在Linux下,分一个段,相当于没分
在32bit下,线性地址空间为4G (0x0000 0000 ~ 0xffff ffff)
8086 (ALU)16bit (地址总线) 20bit
80386 (ALU)32bit (地址总线) 32bit
多级页表:一个程序在运行时其顶级页表常驻内存,次级页表按需要决定是否放在物理内存中(节省了内存空间)
查找具体地址的过程:
页表的一个记录包含很多的内容,例如缓存禁止、访问位、修改位、保护标识区、标志位(在内存否)、物理页面号……
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<stdint.h>
#include<assert.h>
#include<string.h>
void mem_addr(unsigned long vaddr,unsigned long *phy)
{
int pagesize=getpagesize(); //获取物理地址页的大小
unsigned long v_index=vaddr/pagesize; //计算偏移的页数
unsigned long v_offset=v_index*sizeof(uint64_t);
//每条页记录为8字节,计算偏移的字节数
uint64_t item=0;
unsigned long page_offset=vaddr%pagesize;//计算在最终确定页的偏移字节数
int fd=open("/proc/self/pagemap",O_RDONLY); //只读方式打开pagemap
if(fd==-1)
{
printf("open pagemap error\n");
return;
}
if(lseek(fd,v_offset,SEEK_SET)==-1) // 把v_offset设置为固定偏移值
{
printf("lseek error\n");
return;
}
if(read(fd,&item,sizeof(uint64_t))!=sizeof(uint64_t))
{
printf("read item error\n");
return;
}
if((((uint64_t)1<<63)&item)==0) //标志位判断
{
printf("flg filed\n");
return;
}
uint64_t phy_index=(((uint64_t)1<<55)-1)&item;
*phy=phy_index*pagesize+page_offset;
}
int a=10;
void fun()
{
printf("hello\n");
}
int main()
{
int b=20;
int addr=0;
char *s=(char*)malloc(1024*1024);
assert(s!=NULL);
memset(s,0,1024*1024);
pid_t pid=fork();
assert(pid!=-1);
// if(pid==0)
// {
// strcpy(s,"hello");
// }
// mem_addr(s,&addr);
mem_addr(&a,&addr);
// printf(" pid =%d ,vir addr=%x , phy addr=%x\n",getpid(),s,addr);
printf(" pid =%d ,vir addr=%x , phy addr=%x\n",getpid(),&a,addr);
sleep(5);
return 0;
}