#define PAGE_SHIFT 12 // 页偏移 #define PAGE_SIZE (_AC(1,UL) << PAGE_SHIFT) //页大小 #define PAGE_MASK (~((1 << PAGE_SHIFT) - 1)) //取页
还有PAGE_OFFSET,内核中通常为3G。
这几个宏把无符号整数转换成对应的类型
#define __pte(x) ((pte_t) { (x) } ) #define __pmd(x) ((pmd_t) { (x) } ) #define __pgd(x) ((pgd_t) { (x) } ) #define __pgprot(x) ((pgprot_t) { (x) } )
根据x把它转换成对应的无符号整数
#define pte_val(x) ((x).pte_low) #define pmd_val(x) ((x).pmd) #define pgd_val(x) ((x).pgd) #define pgprot_val(x) ((x).pgprot)
与下一个页框对齐
#define PAGE_ALIGN(addr) (((addr)+PAGE_SIZE-1)&PAGE_MASK)
把内核空间的线性地址转换为物理地址
#define __pa(x) ((unsigned long)(x)-PAGE_OFFSET)
把物理地址转化为线性地址
#define __va(x) ((void *)((unsigned long)(x)+PAGE_OFFSET))
把pfn页框号转化为内核线性地址, 然后左移12位, 得到对应的内核线性地址
#define pfn_to_kaddr(pfn) __va((pfn) << PAGE_SHIFT)
根据页框号得到对应的物理页描述符
#define pfn_to_page(pfn) (mem_map + (pfn))
根据物理描述符得到对应的页框号
#define page_to_pfn(page) ((unsigned long)((page) - mem_map))
根据内核线性地址得到其对应的物理页描述符
#define virt_to_page(kaddr) pfn_to_page(__pa(kaddr) >> PAGE_SHIFT)
清除一个页表项并返回前一个值
#define ptep_get_and_clear(xp) __pte(xchg(&(xp)->pte_low, 0))
判断一个页表项是否指向同一页并且访问优先级相同
#define pte_same(a, b) ((a).pte_low == (b).pte_low)
x是页表项值, 通过pte_pfn得到其对应的物理页框号, 最后通过pfn_to_page得到对应的物理页描述符
#define pte_page(x) pfn_to_page(pte_pfn(x))
如果对应的表项值为0, 返回1
#define pte_none(x) (!(x).pte_low)
x是页表项值, 右移12位后得到其对应的物理页框号
#define pte_pfn(x) ((unsigned long)(((x).pte_low >> PAGE_SHIFT)))
根据页框号和页表项的属性值合并成一个页表项值
#define pfn_pte(pfn, prot) __pte(((pfn) << PAGE_SHIFT) | pgprot_val(prot))
根据页框号和页表项的属性值合并成一个中间表项值
#define pfn_pmd(pfn, prot) __pmd(((pfn) << PAGE_SHIFT) | pgprot_val(prot))
向一个表项中写入指定的值
#define set_pte(pteptr, pteval) (*(pteptr) = pteval) #define set_pte_atomic(pteptr, pteval) set_pte(pteptr,pteval) #define set_pmd(pmdptr, pmdval) (*(pmdptr) = pmdval) #define set_pgd(pgdptr, pgdval) (*(pgdptr) = pgdval)
根据线性地址得到高10位值, 也就是在目录表中的索引
#define pgd_index(address) (((address) >> PGDIR_SHIFT) & (PTRS_PER_PGD-1))
根据mm和线性地址address得到目录项值
#define pgd_offset(mm, address) ((mm)->pgd+pgd_index(address))
根据线性地址address提取在页表中的索引
#define pte_index(address) (((address) >> PAGE_SHIFT) & (PTRS_PER_PTE - 1))
根据页描述符和属性得到一个页表项值
#define mk_pte(page, pgprot) pfn_pte(page_to_pfn(page), (pgprot))