diff options
Diffstat (limited to 'arch/arc/mm')
| -rw-r--r-- | arch/arc/mm/cache_arc700.c | 126 | ||||
| -rw-r--r-- | arch/arc/mm/fault.c | 18 | ||||
| -rw-r--r-- | arch/arc/mm/init.c | 42 | ||||
| -rw-r--r-- | arch/arc/mm/tlb.c | 38 | ||||
| -rw-r--r-- | arch/arc/mm/tlbex.S | 35 | 
5 files changed, 129 insertions, 130 deletions
| diff --git a/arch/arc/mm/cache_arc700.c b/arch/arc/mm/cache_arc700.c index aedce190544..f415d851b76 100644 --- a/arch/arc/mm/cache_arc700.c +++ b/arch/arc/mm/cache_arc700.c @@ -73,6 +73,33 @@  #include <asm/cachectl.h>  #include <asm/setup.h> +/* Instruction cache related Auxiliary registers */ +#define ARC_REG_IC_BCR		0x77	/* Build Config reg */ +#define ARC_REG_IC_IVIC		0x10 +#define ARC_REG_IC_CTRL		0x11 +#define ARC_REG_IC_IVIL		0x19 +#if (CONFIG_ARC_MMU_VER > 2) +#define ARC_REG_IC_PTAG		0x1E +#endif + +/* Bit val in IC_CTRL */ +#define IC_CTRL_CACHE_DISABLE   0x1 + +/* Data cache related Auxiliary registers */ +#define ARC_REG_DC_BCR		0x72	/* Build Config reg */ +#define ARC_REG_DC_IVDC		0x47 +#define ARC_REG_DC_CTRL		0x48 +#define ARC_REG_DC_IVDL		0x4A +#define ARC_REG_DC_FLSH		0x4B +#define ARC_REG_DC_FLDL		0x4C +#if (CONFIG_ARC_MMU_VER > 2) +#define ARC_REG_DC_PTAG		0x5C +#endif + +/* Bit val in DC_CTRL */ +#define DC_CTRL_INV_MODE_FLUSH  0x40 +#define DC_CTRL_FLUSH_STATUS    0x100 +  char *arc_cache_mumbojumbo(int cpu_id, char *buf, int len)  {  	int n = 0; @@ -89,8 +116,10 @@ char *arc_cache_mumbojumbo(int cpu_id, char *buf, int len)  			enb ?  "" : "DISABLED (kernel-build)");		\  } -	PR_CACHE(&cpuinfo_arc700[c].icache, __CONFIG_ARC_HAS_ICACHE, "I-Cache"); -	PR_CACHE(&cpuinfo_arc700[c].dcache, __CONFIG_ARC_HAS_DCACHE, "D-Cache"); +	PR_CACHE(&cpuinfo_arc700[c].icache, IS_ENABLED(CONFIG_ARC_HAS_ICACHE), +			"I-Cache"); +	PR_CACHE(&cpuinfo_arc700[c].dcache, IS_ENABLED(CONFIG_ARC_HAS_DCACHE), +			"D-Cache");  	return buf;  } @@ -100,17 +129,23 @@ char *arc_cache_mumbojumbo(int cpu_id, char *buf, int len)   * the cpuinfo structure for later use.   * No Validation done here, simply read/convert the BCRs   */ -void __cpuinit read_decode_cache_bcr(void) +void read_decode_cache_bcr(void)  { -	struct bcr_cache ibcr, dbcr;  	struct cpuinfo_arc_cache *p_ic, *p_dc;  	unsigned int cpu = smp_processor_id(); +	struct bcr_cache { +#ifdef CONFIG_CPU_BIG_ENDIAN +		unsigned int pad:12, line_len:4, sz:4, config:4, ver:8; +#else +		unsigned int ver:8, config:4, sz:4, line_len:4, pad:12; +#endif +	} ibcr, dbcr;  	p_ic = &cpuinfo_arc700[cpu].icache;  	READ_BCR(ARC_REG_IC_BCR, ibcr); -	if (ibcr.config == 0x3) -		p_ic->assoc = 2; +	BUG_ON(ibcr.config != 3); +	p_ic->assoc = 2;		/* Fixed to 2w set assoc */  	p_ic->line_len = 8 << ibcr.line_len;  	p_ic->sz = 0x200 << ibcr.sz;  	p_ic->ver = ibcr.ver; @@ -118,8 +153,8 @@ void __cpuinit read_decode_cache_bcr(void)  	p_dc = &cpuinfo_arc700[cpu].dcache;  	READ_BCR(ARC_REG_DC_BCR, dbcr); -	if (dbcr.config == 0x2) -		p_dc->assoc = 4; +	BUG_ON(dbcr.config != 2); +	p_dc->assoc = 4;		/* Fixed to 4w set assoc */  	p_dc->line_len = 16 << dbcr.line_len;  	p_dc->sz = 0x200 << dbcr.sz;  	p_dc->ver = dbcr.ver; @@ -132,14 +167,12 @@ void __cpuinit read_decode_cache_bcr(void)   * 3. Enable the Caches, setup default flush mode for D-Cache   * 3. Calculate the SHMLBA used by user space   */ -void __cpuinit arc_cache_init(void) +void arc_cache_init(void)  { -	unsigned int temp;  	unsigned int cpu = smp_processor_id();  	struct cpuinfo_arc_cache *ic = &cpuinfo_arc700[cpu].icache;  	struct cpuinfo_arc_cache *dc = &cpuinfo_arc700[cpu].dcache; -	int way_pg_ratio = way_pg_ratio; -	int dcache_does_alias; +	unsigned int dcache_does_alias, temp;  	char str[256];  	printk(arc_cache_mumbojumbo(0, str, sizeof(str))); @@ -149,20 +182,11 @@ void __cpuinit arc_cache_init(void)  #ifdef CONFIG_ARC_HAS_ICACHE  	/* 1. Confirm some of I-cache params which Linux assumes */ -	if ((ic->assoc != ARC_ICACHE_WAYS) || -	    (ic->line_len != ARC_ICACHE_LINE_LEN)) { +	if (ic->line_len != ARC_ICACHE_LINE_LEN)  		panic("Cache H/W doesn't match kernel Config"); -	} -#if (CONFIG_ARC_MMU_VER > 2) -	if (ic->ver != 3) { -		if (running_on_hw) -			panic("Cache ver doesn't match MMU ver\n"); - -		/* For ISS - suggest the toggles to use */ -		pr_err("Use -prop=icache_version=3,-prop=dcache_version=3\n"); -	} -#endif +	if (ic->ver != CONFIG_ARC_MMU_VER) +		panic("Cache ver doesn't match MMU ver\n");  #endif  	/* Enable/disable I-Cache */ @@ -181,14 +205,12 @@ chk_dc:  		return;  #ifdef CONFIG_ARC_HAS_DCACHE -	if ((dc->assoc != ARC_DCACHE_WAYS) || -	    (dc->line_len != ARC_DCACHE_LINE_LEN)) { +	if (dc->line_len != ARC_DCACHE_LINE_LEN)  		panic("Cache H/W doesn't match kernel Config"); -	} - -	dcache_does_alias = (dc->sz / ARC_DCACHE_WAYS) > PAGE_SIZE;  	/* check for D-Cache aliasing */ +	dcache_does_alias = (dc->sz / dc->assoc) > PAGE_SIZE; +  	if (dcache_does_alias && !cache_is_vipt_aliasing())  		panic("Enable CONFIG_ARC_CACHE_VIPT_ALIASING\n");  	else if (!dcache_does_alias && cache_is_vipt_aliasing()) @@ -239,11 +261,9 @@ static inline void wait_for_flush(void)   */  static inline void __dc_entire_op(const int cacheop)  { -	unsigned long flags, tmp = tmp; +	unsigned int tmp = tmp;  	int aux; -	local_irq_save(flags); -  	if (cacheop == OP_FLUSH_N_INV) {  		/* Dcache provides 2 cmd: FLUSH or INV  		 * INV inturn has sub-modes: DISCARD or FLUSH-BEFORE @@ -267,8 +287,6 @@ static inline void __dc_entire_op(const int cacheop)  	/* Switch back the DISCARD ONLY Invalidate mode */  	if (cacheop == OP_FLUSH_N_INV)  		write_aux_reg(ARC_REG_DC_CTRL, tmp & ~DC_CTRL_INV_MODE_FLUSH); - -	local_irq_restore(flags);  }  /* @@ -459,8 +477,15 @@ static void __ic_line_inv_vaddr(unsigned long paddr, unsigned long vaddr,  	local_irq_restore(flags);  } +static inline void __ic_entire_inv(void) +{ +	write_aux_reg(ARC_REG_IC_IVIC, 1); +	read_aux_reg(ARC_REG_IC_CTRL);	/* blocks */ +} +  #else +#define __ic_entire_inv()  #define __ic_line_inv_vaddr(pstart, vstart, sz)  #endif /* CONFIG_ARC_HAS_ICACHE */ @@ -487,7 +512,7 @@ void flush_dcache_page(struct page *page)  	struct address_space *mapping;  	if (!cache_is_vipt_aliasing()) { -		set_bit(PG_arch_1, &page->flags); +		clear_bit(PG_dc_clean, &page->flags);  		return;  	} @@ -501,7 +526,7 @@ void flush_dcache_page(struct page *page)  	 * Make a note that K-mapping is dirty  	 */  	if (!mapping_mapped(mapping)) { -		set_bit(PG_arch_1, &page->flags); +		clear_bit(PG_dc_clean, &page->flags);  	} else if (page_mapped(page)) {  		/* kernel reading from page with U-mapping */ @@ -629,26 +654,13 @@ void ___flush_dcache_page(unsigned long paddr, unsigned long vaddr)  	__dc_line_op(paddr, vaddr & PAGE_MASK, PAGE_SIZE, OP_FLUSH_N_INV);  } -void flush_icache_all(void) -{ -	unsigned long flags; - -	local_irq_save(flags); - -	write_aux_reg(ARC_REG_IC_IVIC, 1); - -	/* lr will not complete till the icache inv operation is not over */ -	read_aux_reg(ARC_REG_IC_CTRL); -	local_irq_restore(flags); -} -  noinline void flush_cache_all(void)  {  	unsigned long flags;  	local_irq_save(flags); -	flush_icache_all(); +	__ic_entire_inv();  	__dc_entire_op(OP_FLUSH_N_INV);  	local_irq_restore(flags); @@ -667,7 +679,12 @@ void flush_cache_page(struct vm_area_struct *vma, unsigned long u_vaddr,  {  	unsigned int paddr = pfn << PAGE_SHIFT; -	__sync_icache_dcache(paddr, u_vaddr, PAGE_SIZE); +	u_vaddr &= PAGE_MASK; + +	___flush_dcache_page(paddr, u_vaddr); + +	if (vma->vm_flags & VM_EXEC) +		__inv_icache_page(paddr, u_vaddr);  }  void flush_cache_range(struct vm_area_struct *vma, unsigned long start, @@ -717,7 +734,7 @@ void copy_user_highpage(struct page *to, struct page *from,  	 * non copied user pages (e.g. read faults which wire in pagecache page  	 * directly).  	 */ -	set_bit(PG_arch_1, &to->flags); +	clear_bit(PG_dc_clean, &to->flags);  	/*  	 * if SRC was already usermapped and non-congruent to kernel mapping @@ -725,15 +742,16 @@ void copy_user_highpage(struct page *to, struct page *from,  	 */  	if (clean_src_k_mappings) {  		__flush_dcache_page(kfrom, kfrom); +		set_bit(PG_dc_clean, &from->flags);  	} else { -		set_bit(PG_arch_1, &from->flags); +		clear_bit(PG_dc_clean, &from->flags);  	}  }  void clear_user_page(void *to, unsigned long u_vaddr, struct page *page)  {  	clear_page(to); -	set_bit(PG_arch_1, &page->flags); +	clear_bit(PG_dc_clean, &page->flags);  } diff --git a/arch/arc/mm/fault.c b/arch/arc/mm/fault.c index 689ffd86d5e..0fd1f0d515f 100644 --- a/arch/arc/mm/fault.c +++ b/arch/arc/mm/fault.c @@ -15,6 +15,7 @@  #include <linux/uaccess.h>  #include <linux/kdebug.h>  #include <asm/pgalloc.h> +#include <asm/mmu.h>  static int handle_vmalloc_fault(struct mm_struct *mm, unsigned long address)  { @@ -51,14 +52,14 @@ bad_area:  	return 1;  } -void do_page_fault(struct pt_regs *regs, int write, unsigned long address, -		   unsigned long cause_code) +void do_page_fault(struct pt_regs *regs, unsigned long address)  {  	struct vm_area_struct *vma = NULL;  	struct task_struct *tsk = current;  	struct mm_struct *mm = tsk->mm;  	siginfo_t info;  	int fault, ret; +	int write = regs->ecr_cause & ECR_C_PROTV_STORE;  /* ST/EX */  	unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE |  				(write ? FAULT_FLAG_WRITE : 0); @@ -109,7 +110,8 @@ good_area:  	/* Handle protection violation, execute on heap or stack */ -	if (cause_code == ((ECR_V_PROTV << 16) | ECR_C_PROTV_INST_FETCH)) +	if ((regs->ecr_vec == ECR_V_PROTV) && +	    (regs->ecr_cause == ECR_C_PROTV_INST_FETCH))  		goto bad_area;  	if (write) { @@ -176,7 +178,6 @@ bad_area_nosemaphore:  	/* User mode accesses just cause a SIGSEGV */  	if (user_mode(regs)) {  		tsk->thread.fault_address = address; -		tsk->thread.cause_code = cause_code;  		info.si_signo = SIGSEGV;  		info.si_errno = 0;  		/* info.si_code has been set above */ @@ -197,7 +198,7 @@ no_context:  	if (fixup_exception(regs))  		return; -	die("Oops", regs, address, cause_code); +	die("Oops", regs, address);  out_of_memory:  	if (is_global_init(tsk)) { @@ -206,8 +207,10 @@ out_of_memory:  	}  	up_read(&mm->mmap_sem); -	if (user_mode(regs)) -		do_group_exit(SIGKILL);	/* This will never return */ +	if (user_mode(regs)) { +		pagefault_out_of_memory(); +		return; +	}  	goto no_context; @@ -218,7 +221,6 @@ do_sigbus:  		goto no_context;  	tsk->thread.fault_address = address; -	tsk->thread.cause_code = cause_code;  	info.si_signo = SIGBUS;  	info.si_errno = 0;  	info.si_code = BUS_ADRERR; diff --git a/arch/arc/mm/init.c b/arch/arc/mm/init.c index 4a177365b2c..a08ce718542 100644 --- a/arch/arc/mm/init.c +++ b/arch/arc/mm/init.c @@ -74,7 +74,7 @@ void __init setup_arch_memory(void)  	/* Last usable page of low mem (no HIGHMEM yet for ARC port) */  	max_low_pfn = max_pfn = PFN_DOWN(end_mem); -	max_mapnr = num_physpages = max_low_pfn - min_low_pfn; +	max_mapnr = max_low_pfn - min_low_pfn;  	/*------------- reserve kernel image -----------------------*/  	memblock_reserve(CONFIG_LINUX_LINK_BASE, @@ -84,7 +84,7 @@ void __init setup_arch_memory(void)  	/*-------------- node setup --------------------------------*/  	memset(zones_size, 0, sizeof(zones_size)); -	zones_size[ZONE_NORMAL] = num_physpages; +	zones_size[ZONE_NORMAL] = max_low_pfn - min_low_pfn;  	/*  	 * We can't use the helper free_area_init(zones[]) because it uses @@ -106,39 +106,9 @@ void __init setup_arch_memory(void)   */  void __init mem_init(void)  { -	int codesize, datasize, initsize, reserved_pages, free_pages; -	int tmp; -  	high_memory = (void *)(CONFIG_LINUX_LINK_BASE + arc_mem_sz); - -	totalram_pages = free_all_bootmem(); - -	/* count all reserved pages [kernel code/data/mem_map..] */ -	reserved_pages = 0; -	for (tmp = 0; tmp < max_mapnr; tmp++) -		if (PageReserved(mem_map + tmp)) -			reserved_pages++; - -	/* XXX: nr_free_pages() is equivalent */ -	free_pages = max_mapnr - reserved_pages; - -	/* -	 * For the purpose of display below, split the "reserve mem" -	 * kernel code/data is already shown explicitly, -	 * Show any other reservations (mem_map[ ] et al) -	 */ -	reserved_pages -= (((unsigned int)_end - CONFIG_LINUX_LINK_BASE) >> -								PAGE_SHIFT); - -	codesize = _etext - _text; -	datasize = _end - _etext; -	initsize = __init_end - __init_begin; - -	pr_info("Memory Available: %dM / %ldM (%dK code, %dK data, %dK init, %dK reserv)\n", -		PAGES_TO_MB(free_pages), -		TO_MB(arc_mem_sz), -		TO_KB(codesize), TO_KB(datasize), TO_KB(initsize), -		PAGES_TO_KB(reserved_pages)); +	free_all_bootmem(); +	mem_init_print_info(NULL);  }  /* @@ -146,13 +116,13 @@ void __init mem_init(void)   */  void __init_refok free_initmem(void)  { -	free_initmem_default(0); +	free_initmem_default(-1);  }  #ifdef CONFIG_BLK_DEV_INITRD  void __init free_initrd_mem(unsigned long start, unsigned long end)  { -	free_reserved_area(start, end, 0, "initrd"); +	free_reserved_area((void *)start, (void *)end, -1, "initrd");  }  #endif diff --git a/arch/arc/mm/tlb.c b/arch/arc/mm/tlb.c index fe1c5a073af..7957dc4e4d4 100644 --- a/arch/arc/mm/tlb.c +++ b/arch/arc/mm/tlb.c @@ -55,7 +55,7 @@  #include <asm/arcregs.h>  #include <asm/setup.h>  #include <asm/mmu_context.h> -#include <asm/tlb.h> +#include <asm/mmu.h>  /*			Need for ARC MMU v2   * @@ -97,6 +97,7 @@   * J-TLB entry got evicted/replaced.   */ +  /* A copy of the ASID from the PID reg is kept in asid_cache */  int asid_cache = FIRST_ASID; @@ -432,9 +433,14 @@ void update_mmu_cache(struct vm_area_struct *vma, unsigned long vaddr_unaligned,  {  	unsigned long vaddr = vaddr_unaligned & PAGE_MASK;  	unsigned long paddr = pte_val(*ptep) & PAGE_MASK; +	struct page *page = pfn_to_page(pte_pfn(*ptep));  	create_tlb(vma, vaddr, ptep); +	if (page == ZERO_PAGE(0)) { +		return; +	} +  	/*  	 * Exec page : Independent of aliasing/page-color considerations,  	 *	       since icache doesn't snoop dcache on ARC, any dirty @@ -446,9 +452,8 @@ void update_mmu_cache(struct vm_area_struct *vma, unsigned long vaddr_unaligned,  	 */  	if ((vma->vm_flags & VM_EXEC) ||  	     addr_not_cache_congruent(paddr, vaddr)) { -		struct page *page = pfn_to_page(pte_pfn(*ptep)); -		int dirty = test_and_clear_bit(PG_arch_1, &page->flags); +		int dirty = !test_and_set_bit(PG_dc_clean, &page->flags);  		if (dirty) {  			/* wback + inv dcache lines */  			__flush_dcache_page(paddr, paddr); @@ -464,12 +469,27 @@ void update_mmu_cache(struct vm_area_struct *vma, unsigned long vaddr_unaligned,   * the cpuinfo structure for later use.   * No Validation is done here, simply read/convert the BCRs   */ -void __cpuinit read_decode_mmu_bcr(void) +void read_decode_mmu_bcr(void)  { -	unsigned int tmp; -	struct bcr_mmu_1_2 *mmu2;	/* encoded MMU2 attr */ -	struct bcr_mmu_3 *mmu3;		/* encoded MMU3 attr */  	struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; +	unsigned int tmp; +	struct bcr_mmu_1_2 { +#ifdef CONFIG_CPU_BIG_ENDIAN +		unsigned int ver:8, ways:4, sets:4, u_itlb:8, u_dtlb:8; +#else +		unsigned int u_dtlb:8, u_itlb:8, sets:4, ways:4, ver:8; +#endif +	} *mmu2; + +	struct bcr_mmu_3 { +#ifdef CONFIG_CPU_BIG_ENDIAN +	unsigned int ver:8, ways:4, sets:4, osm:1, reserv:3, pg_sz:4, +		     u_itlb:4, u_dtlb:4; +#else +	unsigned int u_dtlb:4, u_itlb:4, pg_sz:4, reserv:3, osm:1, sets:4, +		     ways:4, ver:8; +#endif +	} *mmu3;  	tmp = read_aux_reg(ARC_REG_MMU_BCR);  	mmu->ver = (tmp >> 24); @@ -505,12 +525,12 @@ char *arc_mmu_mumbojumbo(int cpu_id, char *buf, int len)  		       "J-TLB %d (%dx%d), uDTLB %d, uITLB %d, %s\n",  		       p_mmu->num_tlb, p_mmu->sets, p_mmu->ways,  		       p_mmu->u_dtlb, p_mmu->u_itlb, -		       __CONFIG_ARC_MMU_SASID_VAL ? "SASID" : ""); +		       IS_ENABLED(CONFIG_ARC_MMU_SASID) ? "SASID" : "");  	return buf;  } -void __cpuinit arc_mmu_init(void) +void arc_mmu_init(void)  {  	char str[256];  	struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu; diff --git a/arch/arc/mm/tlbex.S b/arch/arc/mm/tlbex.S index 3357d26ffe5..5c5bb23001b 100644 --- a/arch/arc/mm/tlbex.S +++ b/arch/arc/mm/tlbex.S @@ -39,7 +39,7 @@  #include <linux/linkage.h>  #include <asm/entry.h> -#include <asm/tlb.h> +#include <asm/mmu.h>  #include <asm/pgtable.h>  #include <asm/arcregs.h>  #include <asm/cache.h> @@ -147,9 +147,9 @@ ex_saved_reg1:  #ifdef CONFIG_ARC_DBG_TLB_MISS_COUNT  	and.f 0, r0, _PAGE_PRESENT  	bz   1f -	ld   r2, [num_pte_not_present] -	add  r2, r2, 1 -	st   r2, [num_pte_not_present] +	ld   r3, [num_pte_not_present] +	add  r3, r3, 1 +	st   r3, [num_pte_not_present]  1:  #endif @@ -271,22 +271,22 @@ ARC_ENTRY EV_TLBMissI  #endif  	;---------------------------------------------------------------- -	; Get the PTE corresponding to V-addr accessed +	; Get the PTE corresponding to V-addr accessed, r2 is setup with EFA  	LOAD_FAULT_PTE  	;----------------------------------------------------------------  	; VERIFY_PTE: Check if PTE permissions approp for executing code  	cmp_s   r2, VMALLOC_START -	mov.lo  r2, (_PAGE_PRESENT | _PAGE_U_READ | _PAGE_U_EXECUTE) -	mov.hs  r2, (_PAGE_PRESENT | _PAGE_K_READ | _PAGE_K_EXECUTE) +	mov.lo  r2, (_PAGE_PRESENT | _PAGE_U_EXECUTE) +	mov.hs  r2, (_PAGE_PRESENT | _PAGE_K_EXECUTE)  	and     r3, r0, r2  ; Mask out NON Flag bits from PTE  	xor.f   r3, r3, r2  ; check ( ( pte & flags_test ) == flags_test )  	bnz     do_slow_path_pf  	; Let Linux VM know that the page was accessed -	or      r0, r0, (_PAGE_PRESENT | _PAGE_ACCESSED)  ; set Accessed Bit -	st_s    r0, [r1]                                  ; Write back PTE +	or      r0, r0, _PAGE_ACCESSED  ; set Accessed Bit +	st_s    r0, [r1]                ; Write back PTE  	CONV_PTE_TO_TLB  	COMMIT_ENTRY_TO_MMU @@ -311,7 +311,7 @@ ARC_ENTRY EV_TLBMissD  	;----------------------------------------------------------------  	; Get the PTE corresponding to V-addr accessed -	; If PTE exists, it will setup, r0 = PTE, r1 = Ptr to PTE +	; If PTE exists, it will setup, r0 = PTE, r1 = Ptr to PTE, r2 = EFA  	LOAD_FAULT_PTE  	;---------------------------------------------------------------- @@ -345,7 +345,7 @@ ARC_ENTRY EV_TLBMissD  	;----------------------------------------------------------------  	; UPDATE_PTE: Let Linux VM know that page was accessed/dirty  	lr      r3, [ecr] -	or      r0, r0, (_PAGE_PRESENT | _PAGE_ACCESSED) ; Accessed bit always +	or      r0, r0, _PAGE_ACCESSED        ; Accessed bit always  	btst_s  r3,  ECR_C_BIT_DTLB_ST_MISS   ; See if it was a Write Access ?  	or.nz   r0, r0, _PAGE_MODIFIED        ; if Write, set Dirty bit as well  	st_s    r0, [r1]                      ; Write back PTE @@ -381,18 +381,7 @@ do_slow_path_pf:  	; ------- setup args for Linux Page fault Hanlder ---------  	mov_s r0, sp -	lr  r2, [efa] -	lr  r3, [ecr] - -	; Both st and ex imply WRITE access of some sort, hence do_page_fault( ) -	; invoked with write=1 for DTLB-st/ex Miss and write=0 for ITLB miss or -	; DTLB-ld Miss -	; DTLB Miss Cause code is ld = 0x01 , st = 0x02, ex = 0x03 -	; Following code uses that fact that st/ex have one bit in common - -	btst_s r3,  ECR_C_BIT_DTLB_ST_MISS -	mov.z  r1, 0 -	mov.nz r1, 1 +	lr  r1, [efa]  	; We don't want exceptions to be disabled while the fault is handled.  	; Now that we have saved the context we return from exception hence | 
