MOV EAX, ESI
AND EAX, 0xFFFFF3FF
SHR EAX, 0A
SUB EAX, 40000000
MOV EAX, ESI
AND EAX, FFCFFFFF
SHR EAX, 14
SUB EAX,3FD00000
首先看下PTE的计算公式:
PTE_Addr = (VirtualAddr >> 12) * 4 + 0xC0000000
上面的推导过程如下:
PTE_Addr=((VA&0xffc00000)>>22)*4096+((VA&0x3ff000)>>12)*4+0xc0000000
=(((VA&0xffc00000)>>22)<<12)+(((VA&0x3ff000)>>12)<<2)+0xc0000000
=((VA&0xffc00000)>>10)+((VA&0x3ff000)>>10)+0xc0000000
=((VA&0xffc00000)+(VA&0x3ff000)>>10)+0xc0000000
=((VA&(0xffc00000|0x3ff000))>>10)+0xc0000000
=((VA&0xfffff000)>>10)+0xc0000000
=(((VA&0xfffff000)>>12)<<2)+0xc0000000
=((VA&0xfffff000)>>12)*4+0xc0000000
下面分析下上面汇编的优化代码:
MOV EAX, ESI
AND EAX, 0xFFFFF3FF ;下面是右移10位,原来右移12位。保持一致其中的最后两位需清0
SHR EAX, 0A ;右移12位再左移2位,结果就是右移10位。
SUB EAX, 40000000 ;对EAX而言(不考虑标志位),减去0x40000000和加上0xC0000000是一样的,之后
;EAX便是ESI对应的VA的页表的虚拟地址。
MOV EAX, ESI
AND EAX, FFCFFFFF;和上面的道理一样
SHR EAX, 14 ;右移12+10位再左移2位,结果就是右移20位。
SUB EAX,3FD00000 ;对EAX而言,减去0x3FD00000和加上0xC0300000是一样的