Search code examples
operating-systemxv6page-tablesvirtual-address-space

How to loop through all page table entries of a process in xv6?


I'm trying to loop through all pages for a process in xv6. I've looked at this diagram to understand how it works: enter image description here

but my code is getting:

unexpected trap 14 from cpu 0 eip 801045ff (cr2=0xdfbb000)

Code:

  pde_t *  physPgDir = (void *)V2P(p->pgdir);
  int c = 0;
  for(unsigned int  i =0; i<NPDENTRIES;i++){
    pde_t * pde = &physPgDir[i];
    
    if(*pde & PTE_P){//page directory valid entry
      int pde_ppn = (int)((PTE_ADDR(*pde)) >> PTXSHIFT);

      pte_t * physPgtab = (void *)(PTE_ADDR(*pde));//grab 20 MSB for inner page table phys pointer;
      // go through inner page table
      for(unsigned int j =0;j<NPDENTRIES;j++){
        pte_t * pte = &physPgtab[j];
        if(*pte & PTE_P){//valid entry
          c++;
          unsigned int pte_ppn = (PTE_ADDR(*pte)) >> PTXSHIFT;//grab 20 MSB for inner page table phys pointer;
          //do thing
        }
      }
    }
}

This is in some custom function in proc.c that gets called elsewhere. p is a process pointer. As far as I understand, cr3 contains the physical address of the current process. However in my case I need to get the page table for a given process pointer. The xv6 code seems to load V2P(p->pgdir) in cr3. That is why I tried to get V2P(p->pgdir). However,the trap happens before pde is dereferenced. Which means there is a problem there. Am I not supposed to use the physical address?

Edit: As Brendan answered, the virtual address p->pgdir is what should be dereferenced. Additionally, the PPN from the Page directory should also be converted via P2V to properly dereference into the page table. If someone else also gets confused about this aspect of xv6 in the future, I hope that helps.


Solution

  • When dealing with paging the golden rule is "never store physical addresses in any kind of pointer". The reasons are:

    a) They aren't virtual addresses and can't be dereferenced, so it's better to make bugs obvious by ensuring you get compile time errors if you try to use a physical address as a pointer.

    b) In some cases physical addresses are a different size to virtual addresses (e.g. "PAE paging" in 80x86 where virtual addresses are still 32-bit but physical addresses are potentially up to 52 bits); and it's better (for portability - e.g. so that PAE support can be added to XV6 easier at some point).

    With this in mind your first line of code is an obvious bug (it breaks the "golden rule"). It should either be pde_t physPgDir = V2P(p->pgdir); or pde_t * pgDir = p->pgdir;. I'll let you figure out which (as I suspect it's homework, and I'm confident that by adhering to the "golden rule" you'll solve your own problem).