stuman 发表于 2014-04-19 17:36

如何对中断处理函数进行线性映射?

vmalloc_fault:
        {
               
                int index = pgd_index(address);
                unsigned long pgd_paddr;
                pgd_t *pgd, *pgd_k;
                pud_t *pud, *pud_k;
                pmd_t *pmd, *pmd_k;
                pte_t *pte_k;

                asm("movl %%cr3,%0":"=r" (pgd_paddr));
                pgd = index + (pgd_t *)__va(pgd_paddr);
                pgd_k = init_mm.pgd + index;

                if (!pgd_present(*pgd_k))
                        goto no_context;

                pud = pud_offset(pgd, address);
                pud_k = pud_offset(pgd_k, address);
                if (!pud_present(*pud_k))
                        goto no_context;
               
                pmd = pmd_offset(pud, address);
                pmd_k = pmd_offset(pud_k, address);
                if (!pmd_present(*pmd_k))
                        goto no_context;
                set_pmd(pmd, *pmd_k);

                pte_k = pte_offset_kernel(pmd_k, address);
                if (!pte_present(*pte_k))
                        goto no_context;
                return;
        }用户态的程序通过中断进入内核态后会执行中断处理函数,那么中断处理函数的线性地址如何转换成物理地址呢?
我在缺页异常中找到如上程序,看程序似乎是把主内核表中pmd项拷贝到用户程序的pmd中。假设这里使用了4级映射,代码pud = pud_offset(pgd, address);是查找线性地址address对应的pud项的线性地址,那么会不会出现pud表根本没有建立的情况呢?与此类似的还有pmd表。如果pud表与pmd表没有建立,这段代码就无法执行了。

humjb_1983 发表于 2014-04-21 09:13

不会出现pud和pmd没有建立的情况,vmalloc区缺页异常发生时,只会缺页表项。
另外,这个函数的作用是从内核主页表同步内核页表。

asuka2001 发表于 2014-04-21 09:51

回复 1# stuman

据我所知,中断向量是存放在固定的物理地址上,应该不会走用户进程的页表转换这一过程!

stuman 发表于 2014-04-25 20:56

中断向量是存放在固定的物理地址上,这不错,可是对于调用中断处理程序还是要进行线性映射呀,这里的线性映射怎么处理呢?

humjb_1983 发表于 2014-04-28 13:45

中断处理程序运行在内核态,照内核通用处理方法就行了吧~~

arm-linux-gcc 发表于 2014-04-28 16:53

本帖最后由 arm-linux-gcc 于 2014-04-28 16:54 编辑

向量的映射,我只清楚ARM的

paging_init
        devicemaps_init
                vectors = early_alloc(PAGE_SIZE * 2);
                early_trap_init(vectors);                           将向量处理相关的代码(地址无关的)拷贝到刚才分配的内存中
                map.pfn = __phys_to_pfn(virt_to_phys(vectors));
                map.virtual = 0xffff0000;                           将刚才分配的页映射到虚拟地址0xffff0000——ARM的向量基地址
                create_mapping(&map);

页: [1]
查看完整版本: 如何对中断处理函数进行线性映射?