diff options
author | Becky Bruce <beckyb@kernel.crashing.org> | 2011-10-10 06:50:39 -0400 |
---|---|---|
committer | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2011-12-07 00:26:22 -0500 |
commit | a1cd54198811e3c35fcaabdb94767b307f7ad1db (patch) | |
tree | 025b16589cab83d330fb2ae436dcdf70da813eac | |
parent | 8c1674de2b42d9d9f14a32445055aa525892c708 (diff) |
powerpc: Update hugetlb huge_pte_alloc and tablewalk code for FSL BOOKE
This updates the hugetlb page table code to handle 64-bit FSL_BOOKE.
The previous 32-bit work counted on the inner levels of the page table
collapsing.
Signed-off-by: Becky Bruce <beckyb@kernel.crashing.org>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
-rw-r--r-- | arch/powerpc/mm/hugetlbpage.c | 48 |
1 files changed, 42 insertions, 6 deletions
diff --git a/arch/powerpc/mm/hugetlbpage.c b/arch/powerpc/mm/hugetlbpage.c index 6b1cf6478851..96178e8fb046 100644 --- a/arch/powerpc/mm/hugetlbpage.c +++ b/arch/powerpc/mm/hugetlbpage.c | |||
@@ -155,11 +155,28 @@ static int __hugepte_alloc(struct mm_struct *mm, hugepd_t *hpdp, | |||
155 | hpdp->pd = 0; | 155 | hpdp->pd = 0; |
156 | kmem_cache_free(cachep, new); | 156 | kmem_cache_free(cachep, new); |
157 | } | 157 | } |
158 | #else | ||
159 | if (!hugepd_none(*hpdp)) | ||
160 | kmem_cache_free(cachep, new); | ||
161 | else | ||
162 | hpdp->pd = ((unsigned long)new & ~PD_HUGE) | pshift; | ||
158 | #endif | 163 | #endif |
159 | spin_unlock(&mm->page_table_lock); | 164 | spin_unlock(&mm->page_table_lock); |
160 | return 0; | 165 | return 0; |
161 | } | 166 | } |
162 | 167 | ||
168 | /* | ||
169 | * These macros define how to determine which level of the page table holds | ||
170 | * the hpdp. | ||
171 | */ | ||
172 | #ifdef CONFIG_PPC_FSL_BOOK3E | ||
173 | #define HUGEPD_PGD_SHIFT PGDIR_SHIFT | ||
174 | #define HUGEPD_PUD_SHIFT PUD_SHIFT | ||
175 | #else | ||
176 | #define HUGEPD_PGD_SHIFT PUD_SHIFT | ||
177 | #define HUGEPD_PUD_SHIFT PMD_SHIFT | ||
178 | #endif | ||
179 | |||
163 | pte_t *huge_pte_alloc(struct mm_struct *mm, unsigned long addr, unsigned long sz) | 180 | pte_t *huge_pte_alloc(struct mm_struct *mm, unsigned long addr, unsigned long sz) |
164 | { | 181 | { |
165 | pgd_t *pg; | 182 | pgd_t *pg; |
@@ -172,12 +189,13 @@ pte_t *huge_pte_alloc(struct mm_struct *mm, unsigned long addr, unsigned long sz | |||
172 | addr &= ~(sz-1); | 189 | addr &= ~(sz-1); |
173 | 190 | ||
174 | pg = pgd_offset(mm, addr); | 191 | pg = pgd_offset(mm, addr); |
175 | if (pshift >= PUD_SHIFT) { | 192 | |
193 | if (pshift >= HUGEPD_PGD_SHIFT) { | ||
176 | hpdp = (hugepd_t *)pg; | 194 | hpdp = (hugepd_t *)pg; |
177 | } else { | 195 | } else { |
178 | pdshift = PUD_SHIFT; | 196 | pdshift = PUD_SHIFT; |
179 | pu = pud_alloc(mm, pg, addr); | 197 | pu = pud_alloc(mm, pg, addr); |
180 | if (pshift >= PMD_SHIFT) { | 198 | if (pshift >= HUGEPD_PUD_SHIFT) { |
181 | hpdp = (hugepd_t *)pu; | 199 | hpdp = (hugepd_t *)pu; |
182 | } else { | 200 | } else { |
183 | pdshift = PMD_SHIFT; | 201 | pdshift = PMD_SHIFT; |
@@ -453,14 +471,23 @@ static void hugetlb_free_pmd_range(struct mmu_gather *tlb, pud_t *pud, | |||
453 | unsigned long start; | 471 | unsigned long start; |
454 | 472 | ||
455 | start = addr; | 473 | start = addr; |
456 | pmd = pmd_offset(pud, addr); | ||
457 | do { | 474 | do { |
475 | pmd = pmd_offset(pud, addr); | ||
458 | next = pmd_addr_end(addr, end); | 476 | next = pmd_addr_end(addr, end); |
459 | if (pmd_none(*pmd)) | 477 | if (pmd_none(*pmd)) |
460 | continue; | 478 | continue; |
479 | #ifdef CONFIG_PPC_FSL_BOOK3E | ||
480 | /* | ||
481 | * Increment next by the size of the huge mapping since | ||
482 | * there may be more than one entry at this level for a | ||
483 | * single hugepage, but all of them point to | ||
484 | * the same kmem cache that holds the hugepte. | ||
485 | */ | ||
486 | next = addr + (1 << hugepd_shift(*(hugepd_t *)pmd)); | ||
487 | #endif | ||
461 | free_hugepd_range(tlb, (hugepd_t *)pmd, PMD_SHIFT, | 488 | free_hugepd_range(tlb, (hugepd_t *)pmd, PMD_SHIFT, |
462 | addr, next, floor, ceiling); | 489 | addr, next, floor, ceiling); |
463 | } while (pmd++, addr = next, addr != end); | 490 | } while (addr = next, addr != end); |
464 | 491 | ||
465 | start &= PUD_MASK; | 492 | start &= PUD_MASK; |
466 | if (start < floor) | 493 | if (start < floor) |
@@ -487,8 +514,8 @@ static void hugetlb_free_pud_range(struct mmu_gather *tlb, pgd_t *pgd, | |||
487 | unsigned long start; | 514 | unsigned long start; |
488 | 515 | ||
489 | start = addr; | 516 | start = addr; |
490 | pud = pud_offset(pgd, addr); | ||
491 | do { | 517 | do { |
518 | pud = pud_offset(pgd, addr); | ||
492 | next = pud_addr_end(addr, end); | 519 | next = pud_addr_end(addr, end); |
493 | if (!is_hugepd(pud)) { | 520 | if (!is_hugepd(pud)) { |
494 | if (pud_none_or_clear_bad(pud)) | 521 | if (pud_none_or_clear_bad(pud)) |
@@ -496,10 +523,19 @@ static void hugetlb_free_pud_range(struct mmu_gather *tlb, pgd_t *pgd, | |||
496 | hugetlb_free_pmd_range(tlb, pud, addr, next, floor, | 523 | hugetlb_free_pmd_range(tlb, pud, addr, next, floor, |
497 | ceiling); | 524 | ceiling); |
498 | } else { | 525 | } else { |
526 | #ifdef CONFIG_PPC_FSL_BOOK3E | ||
527 | /* | ||
528 | * Increment next by the size of the huge mapping since | ||
529 | * there may be more than one entry at this level for a | ||
530 | * single hugepage, but all of them point to | ||
531 | * the same kmem cache that holds the hugepte. | ||
532 | */ | ||
533 | next = addr + (1 << hugepd_shift(*(hugepd_t *)pud)); | ||
534 | #endif | ||
499 | free_hugepd_range(tlb, (hugepd_t *)pud, PUD_SHIFT, | 535 | free_hugepd_range(tlb, (hugepd_t *)pud, PUD_SHIFT, |
500 | addr, next, floor, ceiling); | 536 | addr, next, floor, ceiling); |
501 | } | 537 | } |
502 | } while (pud++, addr = next, addr != end); | 538 | } while (addr = next, addr != end); |
503 | 539 | ||
504 | start &= PGDIR_MASK; | 540 | start &= PGDIR_MASK; |
505 | if (start < floor) | 541 | if (start < floor) |