Message ID | 91159d49bcbee0526ca6235ff7ef1ee7d378d013.1715971869.git.christophe.leroy@csgroup.eu (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | Reimplement huge pages without hugepd on powerpc (8xx, e500, book3s/64) | expand |
On Fri, May 17, 2024 at 08:59:55PM +0200, Christophe Leroy wrote: > Unlike many architectures, powerpc 8xx hardware tablewalk requires > a two level process for all page sizes, allthough second level only > has one entry when pagesize is 8M. So, I went on a quick reading on https://www.nxp.com/docs/en/application-note-software/AN3066.pdf to get more insight, and I realized that some of the questions I made in v1 were quite dump. > > To fit with Linux page table topology and without requiring special > page directory layout like hugepd, the page entry will be replicated > 1024 times in the standard page table. However for large pages it is You only have to replicate 1024 times in case the page size is 4KB, and you will have to replicate that twice and have 2 PMDs pointing to it, right? For 16KB, you will have the PMD containing 512 entries of 16KB. > necessary to set bits in the level-1 (PMD) entry. At the time being, > for 512k pages the flag is kept in the PTE and inserted in the PMD > entry at TLB miss exception, that is necessary because we can have rlwimi r11, r10, 32 - 9, _PMD_PAGE_512K mtspr SPRN_MI_TWC, r11 So we shift the value and compare it to _PMD_PAGE_512K to see if the PTE is a 512K page, and then we set it to SPRN_MI_TWC which I guess is some CPU special register? > pages of different sizes in a page table. However the 12 PTE bits are > fully used and there is no room for an additional bit for page size. You are referring to the bits in arch/powerpc/include/asm/nohash/32/pte-8xx.h ? > For 8M pages, there will be only one page per PMD entry, it is > therefore possible to flag the pagesize in the PMD entry, with the I am confused, and it might be just terminology, or I am getting wrong the design. You say that for 8MB pages, there will one page per PMD entry, but based on the above, you will have 1024 entries (replicated)? So, maybe this wanted to be read as "there will be only one page size per PMD entry". > advantage that the information will already be at the right place for > the hardware. > > To do so, add a new helper called pmd_populate_size() which takes the > page size as an additional argument, and modify __pte_alloc() to also "page size" makes me thing of the standart page size the kernel is operating on (aka PAGE_SIZE), but it is actually the size of the huge page, so I think we should clarify it. > take that argument. pte_alloc() is left unmodified in order to > reduce churn on callers, and a pte_alloc_size() is added for use by > pte_alloc_huge(). > > When an architecture doesn't provide pmd_populate_size(), > pmd_populate() is used as a fallback. It is a bit unfortunate that we have to touch the code for other architectures (in patch#2) > Signed-off-by: Christophe Leroy <christophe.leroy@csgroup.eu> So far I only looked at this patch and patch#2, and code-wise looks good and makes sense, but I find it a bit unfortunate that we have to touch general definitons and arch code (done in patch#2 and patch#3), and I hoped we could somehow isolate this, but I could not think of a way. I will give it some more though. > --- > include/linux/mm.h | 12 +++++++----- > mm/filemap.c | 2 +- > mm/internal.h | 2 +- > mm/memory.c | 19 ++++++++++++------- > mm/pgalloc-track.h | 2 +- > mm/userfaultfd.c | 4 ++-- > 6 files changed, 24 insertions(+), 17 deletions(-) > > diff --git a/include/linux/mm.h b/include/linux/mm.h > index b6bdaa18b9e9..158cb87bc604 100644 > --- a/include/linux/mm.h > +++ b/include/linux/mm.h > @@ -2803,8 +2803,8 @@ static inline void mm_inc_nr_ptes(struct mm_struct *mm) {} > static inline void mm_dec_nr_ptes(struct mm_struct *mm) {} > #endif > > -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd); > -int __pte_alloc_kernel(pmd_t *pmd); > +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz); > +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz); > > #if defined(CONFIG_MMU) > > @@ -2989,7 +2989,8 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, > pte_unmap(pte); \ > } while (0) > > -#define pte_alloc(mm, pmd) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd)) > +#define pte_alloc_size(mm, pmd, sz) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd, sz)) > +#define pte_alloc(mm, pmd) pte_alloc_size(mm, pmd, PAGE_SIZE) > > #define pte_alloc_map(mm, pmd, address) \ > (pte_alloc(mm, pmd) ? NULL : pte_offset_map(pmd, address)) > @@ -2998,9 +2999,10 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, > (pte_alloc(mm, pmd) ? \ > NULL : pte_offset_map_lock(mm, pmd, address, ptlp)) > > -#define pte_alloc_kernel(pmd, address) \ > - ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd))? \ > +#define pte_alloc_kernel_size(pmd, address, sz) \ > + ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd, sz))? \ > NULL: pte_offset_kernel(pmd, address)) > +#define pte_alloc_kernel(pmd, address) pte_alloc_kernel_size(pmd, address, PAGE_SIZE) > > #if USE_SPLIT_PMD_PTLOCKS > > diff --git a/mm/filemap.c b/mm/filemap.c > index 30de18c4fd28..5a783063d1f6 100644 > --- a/mm/filemap.c > +++ b/mm/filemap.c > @@ -3428,7 +3428,7 @@ static bool filemap_map_pmd(struct vm_fault *vmf, struct folio *folio, > } > > if (pmd_none(*vmf->pmd) && vmf->prealloc_pte) > - pmd_install(mm, vmf->pmd, &vmf->prealloc_pte); > + pmd_install(mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); > > return false; > } > diff --git a/mm/internal.h b/mm/internal.h > index 07ad2675a88b..4a01bbf55264 100644 > --- a/mm/internal.h > +++ b/mm/internal.h > @@ -206,7 +206,7 @@ void folio_activate(struct folio *folio); > void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, > struct vm_area_struct *start_vma, unsigned long floor, > unsigned long ceiling, bool mm_wr_locked); > -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte); > +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz); > > struct zap_details; > void unmap_page_range(struct mmu_gather *tlb, > diff --git a/mm/memory.c b/mm/memory.c > index d2155ced45f8..2a9eba13a95f 100644 > --- a/mm/memory.c > +++ b/mm/memory.c > @@ -409,7 +409,12 @@ void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, > } while (vma); > } > > -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) > +#ifndef pmd_populate_size > +#define pmd_populate_size(mm, pmdp, pte, sz) pmd_populate(mm, pmdp, pte) > +#define pmd_populate_kernel_size(mm, pmdp, pte, sz) pmd_populate_kernel(mm, pmdp, pte) > +#endif > + > +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz) > { > spinlock_t *ptl = pmd_lock(mm, pmd); > > @@ -429,25 +434,25 @@ void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) > * smp_rmb() barriers in page table walking code. > */ > smp_wmb(); /* Could be smp_wmb__xxx(before|after)_spin_lock */ > - pmd_populate(mm, pmd, *pte); > + pmd_populate_size(mm, pmd, *pte, sz); > *pte = NULL; > } > spin_unlock(ptl); > } > > -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd) > +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz) > { > pgtable_t new = pte_alloc_one(mm); > if (!new) > return -ENOMEM; > > - pmd_install(mm, pmd, &new); > + pmd_install(mm, pmd, &new, sz); > if (new) > pte_free(mm, new); > return 0; > } > > -int __pte_alloc_kernel(pmd_t *pmd) > +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz) > { > pte_t *new = pte_alloc_one_kernel(&init_mm); > if (!new) > @@ -456,7 +461,7 @@ int __pte_alloc_kernel(pmd_t *pmd) > spin_lock(&init_mm.page_table_lock); > if (likely(pmd_none(*pmd))) { /* Has another populated it ? */ > smp_wmb(); /* See comment in pmd_install() */ > - pmd_populate_kernel(&init_mm, pmd, new); > + pmd_populate_kernel_size(&init_mm, pmd, new, sz); > new = NULL; > } > spin_unlock(&init_mm.page_table_lock); > @@ -4740,7 +4745,7 @@ vm_fault_t finish_fault(struct vm_fault *vmf) > } > > if (vmf->prealloc_pte) > - pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte); > + pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); > else if (unlikely(pte_alloc(vma->vm_mm, vmf->pmd))) > return VM_FAULT_OOM; > } > diff --git a/mm/pgalloc-track.h b/mm/pgalloc-track.h > index e9e879de8649..90e37de7ab77 100644 > --- a/mm/pgalloc-track.h > +++ b/mm/pgalloc-track.h > @@ -45,7 +45,7 @@ static inline pmd_t *pmd_alloc_track(struct mm_struct *mm, pud_t *pud, > > #define pte_alloc_kernel_track(pmd, address, mask) \ > ((unlikely(pmd_none(*(pmd))) && \ > - (__pte_alloc_kernel(pmd) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ > + (__pte_alloc_kernel(pmd, PAGE_SIZE) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ > NULL: pte_offset_kernel(pmd, address)) > > #endif /* _LINUX_PGALLOC_TRACK_H */ > diff --git a/mm/userfaultfd.c b/mm/userfaultfd.c > index 3c3539c573e7..0f129d5c5aa2 100644 > --- a/mm/userfaultfd.c > +++ b/mm/userfaultfd.c > @@ -764,7 +764,7 @@ static __always_inline ssize_t mfill_atomic(struct userfaultfd_ctx *ctx, > break; > } > if (unlikely(pmd_none(dst_pmdval)) && > - unlikely(__pte_alloc(dst_mm, dst_pmd))) { > + unlikely(__pte_alloc(dst_mm, dst_pmd, PAGE_SIZE))) { > err = -ENOMEM; > break; > } > @@ -1687,7 +1687,7 @@ ssize_t move_pages(struct userfaultfd_ctx *ctx, unsigned long dst_start, > err = -ENOENT; > break; > } > - if (unlikely(__pte_alloc(mm, src_pmd))) { > + if (unlikely(__pte_alloc(mm, src_pmd, PAGE_SIZE))) { > err = -ENOMEM; > break; > } > -- > 2.44.0 >
Le 20/05/2024 à 11:01, Oscar Salvador a écrit : > On Fri, May 17, 2024 at 08:59:55PM +0200, Christophe Leroy wrote: >> Unlike many architectures, powerpc 8xx hardware tablewalk requires >> a two level process for all page sizes, allthough second level only >> has one entry when pagesize is 8M. > > So, I went on a quick reading on > > https://www.nxp.com/docs/en/application-note-software/AN3066.pdf > > to get more insight, and I realized that some of the questions I made > in v1 were quite dump. I had a quick look at that document and it seems to provide a good summary of MMU features and principles. However there are some theoritical information which is not fully right in practice. For instance when they say "Segment attributes. These fields define attributes common to all pages in this segment.". This is right in theory if you consider it from Linux page table topology point of view, hence what they call a segment is a PMD entry for Linux. However, in practice each page has its own L1 and L2 attributes and there is not requirement at HW level to have all L1 attributes of all pages of a segment the same. > >> >> To fit with Linux page table topology and without requiring special >> page directory layout like hugepd, the page entry will be replicated >> 1024 times in the standard page table. However for large pages it is > > You only have to replicate 1024 times in case the page size is 4KB, and you > will have to replicate that twice and have 2 PMDs pointing to it, right? Indeed. > > For 16KB, you will have the PMD containing 512 entries of 16KB. Exactly. > >> necessary to set bits in the level-1 (PMD) entry. At the time being, >> for 512k pages the flag is kept in the PTE and inserted in the PMD >> entry at TLB miss exception, that is necessary because we can have > > rlwimi r11, r10, 32 - 9, _PMD_PAGE_512K rlwimi = Rotate Left Word Immediate then Mask Insert. Here it rotates r10 by 23 bits to the left (or 9 to the right) then masks with _PMD_PAGE_512K and inserts it into r11. It means _PAGE_HUGE bit is copied into lower bit of PS attribute. PS takes the following values: PS = 00 ==> Small page (4k or 16k) PS = 01 ==> 512k page PS = 10 ==> Undefined PS = 11 ==> 8M page > mtspr SPRN_MI_TWC, r11 > > So we shift the value and compare it to _PMD_PAGE_512K to see if the PTE > is a 512K page, and then we set it to SPRN_MI_TWC which I guess is some > CPU special register? TWC is where you store the Level 1 attributes, see figure 3 in the document you mentioned. > >> pages of different sizes in a page table. However the 12 PTE bits are >> fully used and there is no room for an additional bit for page size. > > You are referring to the bits in > arch/powerpc/include/asm/nohash/32/pte-8xx.h ? Yes, page are 4k so only the 12 lower bits are available to encode PTE bits and all are used. > >> For 8M pages, there will be only one page per PMD entry, it is >> therefore possible to flag the pagesize in the PMD entry, with the > > I am confused, and it might be just terminology, or I am getting wrong > the design. > You say that for 8MB pages, there will one page per PMD entry, but > based on the above, you will have 1024 entries (replicated)? > So, maybe this wanted to be read as "there will be only one page size per PMD > entry". You have 1024 entries in the PTE table. The PMD entry points to that table were all 1024 entries are the same because they all define the same (half) of a 8M page. So you are also right, there is only one page size because there is only one 8M page. > >> advantage that the information will already be at the right place for >> the hardware. >> >> To do so, add a new helper called pmd_populate_size() which takes the >> page size as an additional argument, and modify __pte_alloc() to also > > "page size" makes me thing of the standart page size the kernel is > operating on (aka PAGE_SIZE), but it is actually the size of the huge > page, so I think we should clarify it. Page size means "size of the page". > >> take that argument. pte_alloc() is left unmodified in order to >> reduce churn on callers, and a pte_alloc_size() is added for use by >> pte_alloc_huge(). >> >> When an architecture doesn't provide pmd_populate_size(), >> pmd_populate() is used as a fallback. > > It is a bit unfortunate that we have to touch the code for other > architectures (in patch#2) That's a RFC, all ideas are welcome, I needed something to replace hugepd_populate() > >> Signed-off-by: Christophe Leroy <christophe.leroy@csgroup.eu> > > So far I only looked at this patch and patch#2, and code-wise looks good and > makes sense, but I find it a bit unfortunate that we have to touch general > definitons and arch code (done in patch#2 and patch#3), and I hoped we could > somehow isolate this, but I could not think of a way. > > I will give it some more though. > >> --- >> include/linux/mm.h | 12 +++++++----- >> mm/filemap.c | 2 +- >> mm/internal.h | 2 +- >> mm/memory.c | 19 ++++++++++++------- >> mm/pgalloc-track.h | 2 +- >> mm/userfaultfd.c | 4 ++-- >> 6 files changed, 24 insertions(+), 17 deletions(-) >> >> diff --git a/include/linux/mm.h b/include/linux/mm.h >> index b6bdaa18b9e9..158cb87bc604 100644 >> --- a/include/linux/mm.h >> +++ b/include/linux/mm.h >> @@ -2803,8 +2803,8 @@ static inline void mm_inc_nr_ptes(struct mm_struct *mm) {} >> static inline void mm_dec_nr_ptes(struct mm_struct *mm) {} >> #endif >> >> -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd); >> -int __pte_alloc_kernel(pmd_t *pmd); >> +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz); >> +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz); >> >> #if defined(CONFIG_MMU) >> >> @@ -2989,7 +2989,8 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, >> pte_unmap(pte); \ >> } while (0) >> >> -#define pte_alloc(mm, pmd) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd)) >> +#define pte_alloc_size(mm, pmd, sz) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd, sz)) >> +#define pte_alloc(mm, pmd) pte_alloc_size(mm, pmd, PAGE_SIZE) >> >> #define pte_alloc_map(mm, pmd, address) \ >> (pte_alloc(mm, pmd) ? NULL : pte_offset_map(pmd, address)) >> @@ -2998,9 +2999,10 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, >> (pte_alloc(mm, pmd) ? \ >> NULL : pte_offset_map_lock(mm, pmd, address, ptlp)) >> >> -#define pte_alloc_kernel(pmd, address) \ >> - ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd))? \ >> +#define pte_alloc_kernel_size(pmd, address, sz) \ >> + ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd, sz))? \ >> NULL: pte_offset_kernel(pmd, address)) >> +#define pte_alloc_kernel(pmd, address) pte_alloc_kernel_size(pmd, address, PAGE_SIZE) >> >> #if USE_SPLIT_PMD_PTLOCKS >> >> diff --git a/mm/filemap.c b/mm/filemap.c >> index 30de18c4fd28..5a783063d1f6 100644 >> --- a/mm/filemap.c >> +++ b/mm/filemap.c >> @@ -3428,7 +3428,7 @@ static bool filemap_map_pmd(struct vm_fault *vmf, struct folio *folio, >> } >> >> if (pmd_none(*vmf->pmd) && vmf->prealloc_pte) >> - pmd_install(mm, vmf->pmd, &vmf->prealloc_pte); >> + pmd_install(mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); >> >> return false; >> } >> diff --git a/mm/internal.h b/mm/internal.h >> index 07ad2675a88b..4a01bbf55264 100644 >> --- a/mm/internal.h >> +++ b/mm/internal.h >> @@ -206,7 +206,7 @@ void folio_activate(struct folio *folio); >> void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, >> struct vm_area_struct *start_vma, unsigned long floor, >> unsigned long ceiling, bool mm_wr_locked); >> -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte); >> +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz); >> >> struct zap_details; >> void unmap_page_range(struct mmu_gather *tlb, >> diff --git a/mm/memory.c b/mm/memory.c >> index d2155ced45f8..2a9eba13a95f 100644 >> --- a/mm/memory.c >> +++ b/mm/memory.c >> @@ -409,7 +409,12 @@ void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, >> } while (vma); >> } >> >> -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) >> +#ifndef pmd_populate_size >> +#define pmd_populate_size(mm, pmdp, pte, sz) pmd_populate(mm, pmdp, pte) >> +#define pmd_populate_kernel_size(mm, pmdp, pte, sz) pmd_populate_kernel(mm, pmdp, pte) >> +#endif >> + >> +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz) >> { >> spinlock_t *ptl = pmd_lock(mm, pmd); >> >> @@ -429,25 +434,25 @@ void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) >> * smp_rmb() barriers in page table walking code. >> */ >> smp_wmb(); /* Could be smp_wmb__xxx(before|after)_spin_lock */ >> - pmd_populate(mm, pmd, *pte); >> + pmd_populate_size(mm, pmd, *pte, sz); >> *pte = NULL; >> } >> spin_unlock(ptl); >> } >> >> -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd) >> +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz) >> { >> pgtable_t new = pte_alloc_one(mm); >> if (!new) >> return -ENOMEM; >> >> - pmd_install(mm, pmd, &new); >> + pmd_install(mm, pmd, &new, sz); >> if (new) >> pte_free(mm, new); >> return 0; >> } >> >> -int __pte_alloc_kernel(pmd_t *pmd) >> +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz) >> { >> pte_t *new = pte_alloc_one_kernel(&init_mm); >> if (!new) >> @@ -456,7 +461,7 @@ int __pte_alloc_kernel(pmd_t *pmd) >> spin_lock(&init_mm.page_table_lock); >> if (likely(pmd_none(*pmd))) { /* Has another populated it ? */ >> smp_wmb(); /* See comment in pmd_install() */ >> - pmd_populate_kernel(&init_mm, pmd, new); >> + pmd_populate_kernel_size(&init_mm, pmd, new, sz); >> new = NULL; >> } >> spin_unlock(&init_mm.page_table_lock); >> @@ -4740,7 +4745,7 @@ vm_fault_t finish_fault(struct vm_fault *vmf) >> } >> >> if (vmf->prealloc_pte) >> - pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte); >> + pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); >> else if (unlikely(pte_alloc(vma->vm_mm, vmf->pmd))) >> return VM_FAULT_OOM; >> } >> diff --git a/mm/pgalloc-track.h b/mm/pgalloc-track.h >> index e9e879de8649..90e37de7ab77 100644 >> --- a/mm/pgalloc-track.h >> +++ b/mm/pgalloc-track.h >> @@ -45,7 +45,7 @@ static inline pmd_t *pmd_alloc_track(struct mm_struct *mm, pud_t *pud, >> >> #define pte_alloc_kernel_track(pmd, address, mask) \ >> ((unlikely(pmd_none(*(pmd))) && \ >> - (__pte_alloc_kernel(pmd) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ >> + (__pte_alloc_kernel(pmd, PAGE_SIZE) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ >> NULL: pte_offset_kernel(pmd, address)) >> >> #endif /* _LINUX_PGALLOC_TRACK_H */ >> diff --git a/mm/userfaultfd.c b/mm/userfaultfd.c >> index 3c3539c573e7..0f129d5c5aa2 100644 >> --- a/mm/userfaultfd.c >> +++ b/mm/userfaultfd.c >> @@ -764,7 +764,7 @@ static __always_inline ssize_t mfill_atomic(struct userfaultfd_ctx *ctx, >> break; >> } >> if (unlikely(pmd_none(dst_pmdval)) && >> - unlikely(__pte_alloc(dst_mm, dst_pmd))) { >> + unlikely(__pte_alloc(dst_mm, dst_pmd, PAGE_SIZE))) { >> err = -ENOMEM; >> break; >> } >> @@ -1687,7 +1687,7 @@ ssize_t move_pages(struct userfaultfd_ctx *ctx, unsigned long dst_start, >> err = -ENOENT; >> break; >> } >> - if (unlikely(__pte_alloc(mm, src_pmd))) { >> + if (unlikely(__pte_alloc(mm, src_pmd, PAGE_SIZE))) { >> err = -ENOMEM; >> break; >> } >> -- >> 2.44.0 >> >
On Mon, May 20, 2024 at 04:24:51PM +0000, Christophe Leroy wrote: > I had a quick look at that document and it seems to provide a good > summary of MMU features and principles. However there are some > theoritical information which is not fully right in practice. For > instance when they say "Segment attributes. These fields define > attributes common to all pages in this segment.". This is right in > theory if you consider it from Linux page table topology point of view, > hence what they call a segment is a PMD entry for Linux. However, in > practice each page has its own L1 and L2 attributes and there is not > requirement at HW level to have all L1 attributes of all pages of a > segment the same. Thanks for taking the time Christophe, highly appreciated. > rlwimi = Rotate Left Word Immediate then Mask Insert. Here it rotates > r10 by 23 bits to the left (or 9 to the right) then masks with > _PMD_PAGE_512K and inserts it into r11. > > It means _PAGE_HUGE bit is copied into lower bit of PS attribute. > > PS takes the following values: > > PS = 00 ==> Small page (4k or 16k) > PS = 01 ==> 512k page > PS = 10 ==> Undefined > PS = 11 ==> 8M page I see, thanks for the explanation. > That's a RFC, all ideas are welcome, I needed something to replace > hugepd_populate() The only user interested in pmd_populate() having a sz parameter is 8xx because it will toggle _PMD_PAGE_8M in case of a 8MB mapping. Would it be possible for 8xx to encode the 'sz' in the *pmd pointer prior to calling down the chain? (something like as we do for PTR_ERR()). Then pmd_populate_{kernel_}size() from 8xx, would extract it like: unsigned long sz = PTR_SIZE(pmd) Then we would not need all these 'sz' parameters scattered. Can that work? PD: Do you know a way to emulate a 8xx VM? qemu seems to not have support support. Thanks
Le 21/05/2024 à 13:57, Oscar Salvador a écrit : > On Mon, May 20, 2024 at 04:24:51PM +0000, Christophe Leroy wrote: >> I had a quick look at that document and it seems to provide a good >> summary of MMU features and principles. However there are some >> theoritical information which is not fully right in practice. For >> instance when they say "Segment attributes. These fields define >> attributes common to all pages in this segment.". This is right in >> theory if you consider it from Linux page table topology point of view, >> hence what they call a segment is a PMD entry for Linux. However, in >> practice each page has its own L1 and L2 attributes and there is not >> requirement at HW level to have all L1 attributes of all pages of a >> segment the same. > > Thanks for taking the time Christophe, highly appreciated. > > >> rlwimi = Rotate Left Word Immediate then Mask Insert. Here it rotates >> r10 by 23 bits to the left (or 9 to the right) then masks with >> _PMD_PAGE_512K and inserts it into r11. >> >> It means _PAGE_HUGE bit is copied into lower bit of PS attribute. >> >> PS takes the following values: >> >> PS = 00 ==> Small page (4k or 16k) >> PS = 01 ==> 512k page >> PS = 10 ==> Undefined >> PS = 11 ==> 8M page > > I see, thanks for the explanation. > >> That's a RFC, all ideas are welcome, I needed something to replace >> hugepd_populate() > > The only user interested in pmd_populate() having a sz parameter > is 8xx because it will toggle _PMD_PAGE_8M in case of a 8MB mapping. > > Would it be possible for 8xx to encode the 'sz' in the *pmd pointer > prior to calling down the chain? (something like as we do for PTR_ERR()). > Then pmd_populate_{kernel_}size() from 8xx, would extract it like: > > unsigned long sz = PTR_SIZE(pmd) > > Then we would not need all these 'sz' parameters scattered. > > Can that work? Indeed _PMD_PAGE_8M can be set in set_huge_pte_at(), no need to do it atomically as part of pmd_populate, so I'll drop patches 1 and 2. > > > PD: Do you know a way to emulate a 8xx VM? qemu seems to not have > support support. > I don't know any way. You are right that 8xx is not supported by QEMU unfortunately. I don't know how difficult it would be to add it to QEMU. Christophe
diff --git a/include/linux/mm.h b/include/linux/mm.h index b6bdaa18b9e9..158cb87bc604 100644 --- a/include/linux/mm.h +++ b/include/linux/mm.h @@ -2803,8 +2803,8 @@ static inline void mm_inc_nr_ptes(struct mm_struct *mm) {} static inline void mm_dec_nr_ptes(struct mm_struct *mm) {} #endif -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd); -int __pte_alloc_kernel(pmd_t *pmd); +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz); +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz); #if defined(CONFIG_MMU) @@ -2989,7 +2989,8 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, pte_unmap(pte); \ } while (0) -#define pte_alloc(mm, pmd) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd)) +#define pte_alloc_size(mm, pmd, sz) (unlikely(pmd_none(*(pmd))) && __pte_alloc(mm, pmd, sz)) +#define pte_alloc(mm, pmd) pte_alloc_size(mm, pmd, PAGE_SIZE) #define pte_alloc_map(mm, pmd, address) \ (pte_alloc(mm, pmd) ? NULL : pte_offset_map(pmd, address)) @@ -2998,9 +2999,10 @@ pte_t *pte_offset_map_nolock(struct mm_struct *mm, pmd_t *pmd, (pte_alloc(mm, pmd) ? \ NULL : pte_offset_map_lock(mm, pmd, address, ptlp)) -#define pte_alloc_kernel(pmd, address) \ - ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd))? \ +#define pte_alloc_kernel_size(pmd, address, sz) \ + ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd, sz))? \ NULL: pte_offset_kernel(pmd, address)) +#define pte_alloc_kernel(pmd, address) pte_alloc_kernel_size(pmd, address, PAGE_SIZE) #if USE_SPLIT_PMD_PTLOCKS diff --git a/mm/filemap.c b/mm/filemap.c index 30de18c4fd28..5a783063d1f6 100644 --- a/mm/filemap.c +++ b/mm/filemap.c @@ -3428,7 +3428,7 @@ static bool filemap_map_pmd(struct vm_fault *vmf, struct folio *folio, } if (pmd_none(*vmf->pmd) && vmf->prealloc_pte) - pmd_install(mm, vmf->pmd, &vmf->prealloc_pte); + pmd_install(mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); return false; } diff --git a/mm/internal.h b/mm/internal.h index 07ad2675a88b..4a01bbf55264 100644 --- a/mm/internal.h +++ b/mm/internal.h @@ -206,7 +206,7 @@ void folio_activate(struct folio *folio); void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, struct vm_area_struct *start_vma, unsigned long floor, unsigned long ceiling, bool mm_wr_locked); -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte); +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz); struct zap_details; void unmap_page_range(struct mmu_gather *tlb, diff --git a/mm/memory.c b/mm/memory.c index d2155ced45f8..2a9eba13a95f 100644 --- a/mm/memory.c +++ b/mm/memory.c @@ -409,7 +409,12 @@ void free_pgtables(struct mmu_gather *tlb, struct ma_state *mas, } while (vma); } -void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) +#ifndef pmd_populate_size +#define pmd_populate_size(mm, pmdp, pte, sz) pmd_populate(mm, pmdp, pte) +#define pmd_populate_kernel_size(mm, pmdp, pte, sz) pmd_populate_kernel(mm, pmdp, pte) +#endif + +void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte, unsigned long sz) { spinlock_t *ptl = pmd_lock(mm, pmd); @@ -429,25 +434,25 @@ void pmd_install(struct mm_struct *mm, pmd_t *pmd, pgtable_t *pte) * smp_rmb() barriers in page table walking code. */ smp_wmb(); /* Could be smp_wmb__xxx(before|after)_spin_lock */ - pmd_populate(mm, pmd, *pte); + pmd_populate_size(mm, pmd, *pte, sz); *pte = NULL; } spin_unlock(ptl); } -int __pte_alloc(struct mm_struct *mm, pmd_t *pmd) +int __pte_alloc(struct mm_struct *mm, pmd_t *pmd, unsigned long sz) { pgtable_t new = pte_alloc_one(mm); if (!new) return -ENOMEM; - pmd_install(mm, pmd, &new); + pmd_install(mm, pmd, &new, sz); if (new) pte_free(mm, new); return 0; } -int __pte_alloc_kernel(pmd_t *pmd) +int __pte_alloc_kernel(pmd_t *pmd, unsigned long sz) { pte_t *new = pte_alloc_one_kernel(&init_mm); if (!new) @@ -456,7 +461,7 @@ int __pte_alloc_kernel(pmd_t *pmd) spin_lock(&init_mm.page_table_lock); if (likely(pmd_none(*pmd))) { /* Has another populated it ? */ smp_wmb(); /* See comment in pmd_install() */ - pmd_populate_kernel(&init_mm, pmd, new); + pmd_populate_kernel_size(&init_mm, pmd, new, sz); new = NULL; } spin_unlock(&init_mm.page_table_lock); @@ -4740,7 +4745,7 @@ vm_fault_t finish_fault(struct vm_fault *vmf) } if (vmf->prealloc_pte) - pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte); + pmd_install(vma->vm_mm, vmf->pmd, &vmf->prealloc_pte, PAGE_SIZE); else if (unlikely(pte_alloc(vma->vm_mm, vmf->pmd))) return VM_FAULT_OOM; } diff --git a/mm/pgalloc-track.h b/mm/pgalloc-track.h index e9e879de8649..90e37de7ab77 100644 --- a/mm/pgalloc-track.h +++ b/mm/pgalloc-track.h @@ -45,7 +45,7 @@ static inline pmd_t *pmd_alloc_track(struct mm_struct *mm, pud_t *pud, #define pte_alloc_kernel_track(pmd, address, mask) \ ((unlikely(pmd_none(*(pmd))) && \ - (__pte_alloc_kernel(pmd) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ + (__pte_alloc_kernel(pmd, PAGE_SIZE) || ({*(mask)|=PGTBL_PMD_MODIFIED;0;})))?\ NULL: pte_offset_kernel(pmd, address)) #endif /* _LINUX_PGALLOC_TRACK_H */ diff --git a/mm/userfaultfd.c b/mm/userfaultfd.c index 3c3539c573e7..0f129d5c5aa2 100644 --- a/mm/userfaultfd.c +++ b/mm/userfaultfd.c @@ -764,7 +764,7 @@ static __always_inline ssize_t mfill_atomic(struct userfaultfd_ctx *ctx, break; } if (unlikely(pmd_none(dst_pmdval)) && - unlikely(__pte_alloc(dst_mm, dst_pmd))) { + unlikely(__pte_alloc(dst_mm, dst_pmd, PAGE_SIZE))) { err = -ENOMEM; break; } @@ -1687,7 +1687,7 @@ ssize_t move_pages(struct userfaultfd_ctx *ctx, unsigned long dst_start, err = -ENOENT; break; } - if (unlikely(__pte_alloc(mm, src_pmd))) { + if (unlikely(__pte_alloc(mm, src_pmd, PAGE_SIZE))) { err = -ENOMEM; break; }
Unlike many architectures, powerpc 8xx hardware tablewalk requires a two level process for all page sizes, allthough second level only has one entry when pagesize is 8M. To fit with Linux page table topology and without requiring special page directory layout like hugepd, the page entry will be replicated 1024 times in the standard page table. However for large pages it is necessary to set bits in the level-1 (PMD) entry. At the time being, for 512k pages the flag is kept in the PTE and inserted in the PMD entry at TLB miss exception, that is necessary because we can have pages of different sizes in a page table. However the 12 PTE bits are fully used and there is no room for an additional bit for page size. For 8M pages, there will be only one page per PMD entry, it is therefore possible to flag the pagesize in the PMD entry, with the advantage that the information will already be at the right place for the hardware. To do so, add a new helper called pmd_populate_size() which takes the page size as an additional argument, and modify __pte_alloc() to also take that argument. pte_alloc() is left unmodified in order to reduce churn on callers, and a pte_alloc_size() is added for use by pte_alloc_huge(). When an architecture doesn't provide pmd_populate_size(), pmd_populate() is used as a fallback. Signed-off-by: Christophe Leroy <christophe.leroy@csgroup.eu> --- include/linux/mm.h | 12 +++++++----- mm/filemap.c | 2 +- mm/internal.h | 2 +- mm/memory.c | 19 ++++++++++++------- mm/pgalloc-track.h | 2 +- mm/userfaultfd.c | 4 ++-- 6 files changed, 24 insertions(+), 17 deletions(-)