[PATCH v2 06/20] ARM: LPAE: Introduce the 3-level page table format definitions
Russell King - ARM Linux
linux at arm.linux.org.uk
Mon Nov 15 13:34:43 EST 2010
On Fri, Nov 12, 2010 at 06:00:26PM +0000, Catalin Marinas wrote:
> This patch introduces the pgtable-3level*.h files with definitions
> specific to the LPAE page table format (3 levels of page tables).
>
> Each table is 4KB and has 512 64-bit entries. An entry can point to a
> 40-bit physical address. The young, write and exec software bits share
> the corresponding hardware bits (negated). Other software bits use spare
> bits in the PTE.
>
> The patch also changes some variable types from unsigned long or int to
> pteval_t or pgprot_t.
>
> Signed-off-by: Catalin Marinas <catalin.marinas at arm.com>
> ---
> arch/arm/include/asm/page.h | 4 +
> arch/arm/include/asm/pgtable-3level-hwdef.h | 78 ++++++++++++++++++
> arch/arm/include/asm/pgtable-3level-types.h | 55 +++++++++++++
> arch/arm/include/asm/pgtable-3level.h | 113 +++++++++++++++++++++++++++
> arch/arm/include/asm/pgtable-hwdef.h | 4 +
> arch/arm/include/asm/pgtable.h | 6 +-
> arch/arm/mm/mm.h | 8 +-
> arch/arm/mm/mmu.c | 2 +-
> 8 files changed, 264 insertions(+), 6 deletions(-)
> create mode 100644 arch/arm/include/asm/pgtable-3level-hwdef.h
> create mode 100644 arch/arm/include/asm/pgtable-3level-types.h
> create mode 100644 arch/arm/include/asm/pgtable-3level.h
>
> diff --git a/arch/arm/include/asm/page.h b/arch/arm/include/asm/page.h
> index 3848105..e5124db 100644
> --- a/arch/arm/include/asm/page.h
> +++ b/arch/arm/include/asm/page.h
> @@ -151,7 +151,11 @@ extern void __cpu_copy_user_highpage(struct page *to, struct page *from,
> #define clear_page(page) memset((void *)(page), 0, PAGE_SIZE)
> extern void copy_page(void *to, const void *from);
>
> +#ifdef CONFIG_ARM_LPAE
> +#include <asm/pgtable-3level-types.h>
> +#else
> #include <asm/pgtable-2level-types.h>
> +#endif
>
> #endif /* CONFIG_MMU */
>
> diff --git a/arch/arm/include/asm/pgtable-3level-hwdef.h b/arch/arm/include/asm/pgtable-3level-hwdef.h
> new file mode 100644
> index 0000000..2f99c3c
> --- /dev/null
> +++ b/arch/arm/include/asm/pgtable-3level-hwdef.h
> @@ -0,0 +1,78 @@
> +/*
> + * arch/arm/include/asm/pgtable-3level-hwdef.h
> + *
> + * Copyright (C) 2010 ARM Ltd.
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that it will be useful,
> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> + * GNU General Public License for more details.
> + *
> + * You should have received a copy of the GNU General Public License
> + * along with this program; if not, write to the Free Software
> + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
> + */
> +#ifndef _ASM_PGTABLE_3LEVEL_HWDEF_H
> +#define _ASM_PGTABLE_3LEVEL_HWDEF_H
> +
> +#include <linux/const.h>
> +#include <asm/pgtable-3level-types.h>
> +
> +/*
> + * Hardware page table definitions.
> + *
> + * + Level 1/2 descriptor
> + * - common
> + */
> +#define PMD_TYPE_MASK (_AT(pmd_t, 3) << 0)
> +#define PMD_TYPE_FAULT (_AT(pmd_t, 0) << 0)
> +#define PMD_TYPE_TABLE (_AT(pmd_t, 3) << 0)
> +#define PMD_TYPE_SECT (_AT(pmd_t, 1) << 0)
> +#define PMD_BIT4 (_AT(pmd_t, 0))
> +#define PMD_DOMAIN(x) (_AT(pmd_t, 0))
It is really not correct to have these constants type'd as pmd_t.
The idea behind pmd_t et.al. is to detect when normal arithmetic or
logical operations are performed on page table entries when the
accessors instead should be used.
By typing these as pmd_t, it means operations need to be:
u32 pmdval = pmd_val(foo) | pmd_val(PMD_TYE_TABLE);
which is obviously more complicated than is needed.
> diff --git a/arch/arm/mm/mm.h b/arch/arm/mm/mm.h
> index 6630620..a62f093 100644
> --- a/arch/arm/mm/mm.h
> +++ b/arch/arm/mm/mm.h
> @@ -16,10 +16,10 @@ static inline pmd_t *pmd_off_k(unsigned long virt)
> }
>
> struct mem_type {
> - unsigned int prot_pte;
> - unsigned int prot_l1;
> - unsigned int prot_sect;
> - unsigned int domain;
> + pgprot_t prot_pte;
> + pgprot_t prot_l1;
> + pgprot_t prot_sect;
> + pgprot_t domain;
Again, this is wrong. There's an accessor for pgprot_t typed data. This
causes code to violate it.
> diff --git a/arch/arm/mm/mmu.c b/arch/arm/mm/mmu.c
> index 0ca33dd..7c803c4 100644
> --- a/arch/arm/mm/mmu.c
> +++ b/arch/arm/mm/mmu.c
> @@ -292,7 +292,7 @@ static void __init build_mem_type_table(void)
> {
> struct cachepolicy *cp;
> unsigned int cr = get_cr();
> - unsigned int user_pgprot, kern_pgprot, vecs_pgprot;
> + pgprot_t user_pgprot, kern_pgprot, vecs_pgprot;
Ditto.
More information about the linux-arm-kernel
mailing list