On Fri, Sep 20, 2024 at 08:57:23AM +0200, Ryan Roberts wrote: > On 19/09/2024 21:25, Russell King (Oracle) wrote: > > On Thu, Sep 19, 2024 at 07:49:09PM +0200, Ryan Roberts wrote: > >> On 19/09/2024 18:06, Russell King (Oracle) wrote: > >>> On Thu, Sep 19, 2024 at 05:48:58PM +0200, Ryan Roberts wrote: > >>>>> 32-bit arm uses, in some circumstances, an array because each level 1 > >>>>> page table entry is actually two descriptors. It needs to be this way > >>>>> because each level 2 table pointed to by each level 1 entry has 256 > >>>>> entries, meaning it only occupies 1024 bytes in a 4096 byte page. > >>>>> > >>>>> In order to cut down on the wastage, treat the level 1 page table as > >>>>> groups of two entries, which point to two consecutive 1024 byte tables > >>>>> in the level 2 page. > >>>>> > >>>>> The level 2 entry isn't suitable for the kernel's use cases (there are > >>>>> no bits to represent accessed/dirty and other important stuff that the > >>>>> Linux MM wants) so we maintain the hardware page tables and a separate > >>>>> set that Linux uses in the same page. Again, the software tables are > >>>>> consecutive, so from Linux's perspective, the level 2 page tables > >>>>> have 512 entries in them and occupy one full page. > >>>>> > >>>>> This is documented in arch/arm/include/asm/pgtable-2level.h > >>>>> > >>>>> However, what this means is that from the software perspective, the > >>>>> level 1 page table descriptors are an array of two entries, both of > >>>>> which need to be setup when creating a level 2 page table, but only > >>>>> the first one should ever be dereferenced when walking the tables, > >>>>> otherwise the code that walks the second level of page table entries > >>>>> will walk off the end of the software table into the actual hardware > >>>>> descriptors. > >>>>> > >>>>> I've no idea what the idea is behind introducing pgd_get() and what > >>>>> it's semantics are, so I can't comment further. > >>>> > >>>> The helper is intended to read the value of the entry pointed to by the passed > >>>> in pointer. And it shoiuld be read in a "single copy atomic" manner, meaning no > >>>> tearing. Further, the PTL is expected to be held when calling the getter. If the > >>>> HW can write to the entry such that its racing with the lock holder (i.e. HW > >>>> update of access/dirty) then READ_ONCE() should be suitable for most > >>>> architectures. If there is no possibility of racing (because HW doesn't write to > >>>> the entry), then a simple dereference would be sufficient, I think (which is > >>>> what the core code was already doing in most cases). > >>> > >>> The core code should be making no access to the PGD entries on 32-bit > >>> ARM since the PGD level does not exist. Writes are done at PMD level > >>> in arch code. Reads are done by core code at PMD level. > >>> > >>> It feels to me like pgd_get() just doesn't fit the model to which 32-bit > >>> ARM was designed to use decades ago, so I want full details about what > >>> pgd_get() is going to be used for and how it is going to be used, > >>> because I feel completely in the dark over this new development. I fear > >>> that someone hasn't understood the Linux page table model if they're > >>> wanting to access stuff at levels that effectively "aren't implemented" > >>> in the architecture specific kernel model of the page tables. > >> > >> This change isn't as big and scary as I think you fear. > > > > The situation is as I state above. Core code must _not_ dereference pgd > > pointers on 32-bit ARM. > > Let's just rewind a bit. This thread exists because the kernel test robot failed > to compile pgd_none_or_clear_bad() (a core-mm function) for the arm architecture > after Anshuman changed the direct pgd dereference to pgdp_get(). The reason > compilation failed is because arm defines its own pgdp_get() override, but it is > broken (there is a typo). Let's not rewind, because had you fully read and digested my reply, you would have seen why this isn't a problem... but let me spell it out. > > Code before Anshuman's change: > > static inline int pgd_none_or_clear_bad(pgd_t *pgd) > { > if (pgd_none(*pgd)) > return 1; > if (unlikely(pgd_bad(*pgd))) { > pgd_clear_bad(pgd); > return 1; > } > return 0; > } This isn't a problem as the code stands. While there is a dereference in C, that dereference is a simple struct copy, something that we use everywhere in the kernel. However, that is as far as it goes, because neither pgd_none() and pgd_bad() make use of their argument, and thus the compiler will optimise it away, resulting in no actual access to the page tables - _as_ _intended_. If these are going to be converted to pgd_get(), then we need pgd_get() to _also_ be optimised away, and if e.g. this is the only place that pgd_get() is going to be used, the suggestion I made in my previous email is entirely reasonable, since we know that the result of pgd_get() will not actually be used. > As an aside, the kernel also dereferences p4d, pud, pmd and pte pointers in > various circumstances. I already covered these in my previous reply. > And other changes in this series are also replacing those > direct dereferences with calls to similar helpers. The fact that these are all > folded (by a custom arm implementation if I've understood the below correctly) > just means that each dereference is returning what you would call the pmd from > the HW perspective, I think? It'll "return" the first of each pair of level-1 page table entries, which is pgd[0] or *p4d, *pud, *pmd - but all of these except *pmd need to be optimised away, so throwing lots of READ_ONCE() around this code without considering this is certainly the wrong approach. > >> The core-mm today > >> dereferences pgd pointers (and p4d, pud, pmd pointers) directly in its code. See > >> follow_pfnmap_start(), > > > > Doesn't seem to exist at least not in 6.11. > > Appologies, I'm on mm-unstable and that isn't upstream yet. See follow_pte() in > v6.11 or __apply_to_page_range(), or pgd_none_or_clear_bad() as per above. Looking at follow_pte(), it's not a problem. I think we wouldn't be having this conversation before: commit a32618d28dbe6e9bf8ec508ccbc3561a7d7d32f0 Author: Russell King <rmk+kernel@xxxxxxxxxxxxxxxx> Date: Tue Nov 22 17:30:28 2011 +0000 ARM: pgtable: switch to use pgtable-nopud.h where: -#define pgd_none(pgd) (0) -#define pgd_bad(pgd) (0) existed before this commit - and thus the dereference in things like: pgd_none(*pgd) wouldn't even be visible to beyond the preprocessor step. -- RMK's Patch system: https://www.armlinux.org.uk/developer/patches/ FTTP is here! 80Mbps down 10Mbps up. Decent connectivity at last!