- 论坛徽章:
- 0
|
在linux i586上. 要在KERNEL中调用, 同时要保证被转的地址有读权限
和正在内存. 因为仅仅为了演示, 权限检查不够.
paddr是一个kernel中的地址,可以用它来检查物理地址来验证转换正确. 因为物理地址无法在正常时候使用.
这不是通用的完善的程序,仅仅是为了玩而已.
[quote]原帖由 "flw"]下次贴代码请正确使用 code 标签,这次先帮你改过。[/quote 发表:
- #include <linux/module.h>;
- #include <linux/types.h>;
- #include <linux/errno.h>;
- #include <linux/slab.h>;
- #include <linux/romfs_fs.h>;
- #include <linux/fs.h>;
- #include <linux/init.h>;
- #include <linux/pagemap.h>;
- #include <linux/smp_lock.h>;
- #include <linux/buffer_head.h>;
- #include <linux/vfs.h>;
- #include <asm/uaccess.h>;
- //convert a virtual address to physical address, and returns a v-addr pointer used to access the memory
- long //physical memory address returned
- va2pa(
- struct mm_struct *mm, //user task mm
- unsigned long uaddr, //the user virtual address to convert
- char **kaddr) //the kernel virtual address
- {
- struct page *page;
- pgd_t *pgd;
- pte_t *pte;
- long pageno;
- unsigned long pg, paddr;
- char *cp;
-
- pgd = pgd_offset(mm, uaddr);
- pageno = ( (*(unsigned long*)&pgd[0]) / PAGE_SIZE );
-
- page = pfn_to_page(pageno); //get page from physical page frame no
- pte = lowmem_page_address(page); //pte_table
- pte = pte + pte_index(uaddr); //pte entry
- pg = ( *(unsigned long*)&pte[0] >;>; 12 ); //physical page no
- paddr = (pg << 12) + (uaddr & 0x00000fff);
-
- page = pfn_to_page(pg);
- cp = lowmem_page_address(page); //cp point to the vaddr page
- cp += (uaddr & 0x00000fff); //cp point to vaddr
- *kaddr = cp;
- return paddr;
- }
复制代码 |
|