forked from mirrors/linux
		
	ARM: mm: introduce present, faulting entries for PAGE_NONE
PROT_NONE mappings apply the page protection attributes defined by _P000 which translate to PAGE_NONE for ARM. These attributes specify an XN, RDONLY pte that is inaccessible to userspace. However, on kernels configured without support for domains, such a pte *is* accessible to the kernel and can be read via get_user, allowing tasks to read PROT_NONE pages via syscalls such as read/write over a pipe. This patch introduces a new software pte flag, L_PTE_NONE, that is set to identify faulting, present entries. Signed-off-by: Will Deacon <will.deacon@arm.com>
This commit is contained in:
		
							parent
							
								
									dbf62d5006
								
							
						
					
					
						commit
						26ffd0d43b
					
				
					 6 changed files with 16 additions and 3 deletions
				
			
		| 
						 | 
				
			
			@ -124,6 +124,7 @@
 | 
			
		|||
#define L_PTE_USER		(_AT(pteval_t, 1) << 8)
 | 
			
		||||
#define L_PTE_XN		(_AT(pteval_t, 1) << 9)
 | 
			
		||||
#define L_PTE_SHARED		(_AT(pteval_t, 1) << 10)	/* shared(v6), coherent(xsc3) */
 | 
			
		||||
#define L_PTE_NONE		(_AT(pteval_t, 1) << 11)
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * These are the memory types, defined to be compatible with
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -77,6 +77,7 @@
 | 
			
		|||
#define L_PTE_XN		(_AT(pteval_t, 1) << 54)	/* XN */
 | 
			
		||||
#define L_PTE_DIRTY		(_AT(pteval_t, 1) << 55)	/* unused */
 | 
			
		||||
#define L_PTE_SPECIAL		(_AT(pteval_t, 1) << 56)	/* unused */
 | 
			
		||||
#define L_PTE_NONE		(_AT(pteval_t, 1) << 57)	/* PROT_NONE */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * To be used in assembly code with the upper page attributes.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -73,7 +73,7 @@ extern pgprot_t		pgprot_kernel;
 | 
			
		|||
 | 
			
		||||
#define _MOD_PROT(p, b)	__pgprot(pgprot_val(p) | (b))
 | 
			
		||||
 | 
			
		||||
#define PAGE_NONE		_MOD_PROT(pgprot_user, L_PTE_XN | L_PTE_RDONLY)
 | 
			
		||||
#define PAGE_NONE		_MOD_PROT(pgprot_user, L_PTE_XN | L_PTE_RDONLY | L_PTE_NONE)
 | 
			
		||||
#define PAGE_SHARED		_MOD_PROT(pgprot_user, L_PTE_USER | L_PTE_XN)
 | 
			
		||||
#define PAGE_SHARED_EXEC	_MOD_PROT(pgprot_user, L_PTE_USER)
 | 
			
		||||
#define PAGE_COPY		_MOD_PROT(pgprot_user, L_PTE_USER | L_PTE_RDONLY | L_PTE_XN)
 | 
			
		||||
| 
						 | 
				
			
			@ -83,7 +83,7 @@ extern pgprot_t		pgprot_kernel;
 | 
			
		|||
#define PAGE_KERNEL		_MOD_PROT(pgprot_kernel, L_PTE_XN)
 | 
			
		||||
#define PAGE_KERNEL_EXEC	pgprot_kernel
 | 
			
		||||
 | 
			
		||||
#define __PAGE_NONE		__pgprot(_L_PTE_DEFAULT | L_PTE_RDONLY | L_PTE_XN)
 | 
			
		||||
#define __PAGE_NONE		__pgprot(_L_PTE_DEFAULT | L_PTE_RDONLY | L_PTE_XN | L_PTE_NONE)
 | 
			
		||||
#define __PAGE_SHARED		__pgprot(_L_PTE_DEFAULT | L_PTE_USER | L_PTE_XN)
 | 
			
		||||
#define __PAGE_SHARED_EXEC	__pgprot(_L_PTE_DEFAULT | L_PTE_USER)
 | 
			
		||||
#define __PAGE_COPY		__pgprot(_L_PTE_DEFAULT | L_PTE_USER | L_PTE_RDONLY | L_PTE_XN)
 | 
			
		||||
| 
						 | 
				
			
			@ -240,7 +240,7 @@ static inline pte_t pte_mkspecial(pte_t pte) { return pte; }
 | 
			
		|||
 | 
			
		||||
static inline pte_t pte_modify(pte_t pte, pgprot_t newprot)
 | 
			
		||||
{
 | 
			
		||||
	const pteval_t mask = L_PTE_XN | L_PTE_RDONLY | L_PTE_USER;
 | 
			
		||||
	const pteval_t mask = L_PTE_XN | L_PTE_RDONLY | L_PTE_USER | L_PTE_NONE;
 | 
			
		||||
	pte_val(pte) = (pte_val(pte) & ~mask) | (pgprot_val(newprot) & mask);
 | 
			
		||||
	return pte;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -167,6 +167,10 @@
 | 
			
		|||
	tst	r1, #L_PTE_YOUNG
 | 
			
		||||
	tstne	r1, #L_PTE_PRESENT
 | 
			
		||||
	moveq	r3, #0
 | 
			
		||||
#ifndef CONFIG_CPU_USE_DOMAINS
 | 
			
		||||
	tstne	r1, #L_PTE_NONE
 | 
			
		||||
	movne	r3, #0
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	str	r3, [r0]
 | 
			
		||||
	mcr	p15, 0, r0, c7, c10, 1		@ flush_pte
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -101,6 +101,10 @@ ENTRY(cpu_v7_set_pte_ext)
 | 
			
		|||
 | 
			
		||||
	tst	r1, #L_PTE_YOUNG
 | 
			
		||||
	tstne	r1, #L_PTE_VALID
 | 
			
		||||
#ifndef CONFIG_CPU_USE_DOMAINS
 | 
			
		||||
	eorne	r1, r1, #L_PTE_NONE
 | 
			
		||||
	tstne	r1, #L_PTE_NONE
 | 
			
		||||
#endif
 | 
			
		||||
	moveq	r3, #0
 | 
			
		||||
 | 
			
		||||
 ARM(	str	r3, [r0, #2048]! )
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -67,6 +67,9 @@ ENTRY(cpu_v7_set_pte_ext)
 | 
			
		|||
#ifdef CONFIG_MMU
 | 
			
		||||
	tst	r2, #L_PTE_VALID
 | 
			
		||||
	beq	1f
 | 
			
		||||
	tst	r3, #1 << (57 - 32)		@ L_PTE_NONE
 | 
			
		||||
	bicne	r2, #L_PTE_VALID
 | 
			
		||||
	bne	1f
 | 
			
		||||
	tst	r3, #1 << (55 - 32)		@ L_PTE_DIRTY
 | 
			
		||||
	orreq	r2, #L_PTE_RDONLY
 | 
			
		||||
1:	strd	r2, r3, [r0]
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue