| /* | 
 |  * arch/sh/mm/tlb-urb.c | 
 |  * | 
 |  * TLB entry wiring helpers for URB-equipped parts. | 
 |  * | 
 |  * Copyright (C) 2010  Matt Fleming | 
 |  * | 
 |  * This file is subject to the terms and conditions of the GNU General Public | 
 |  * License.  See the file "COPYING" in the main directory of this archive | 
 |  * for more details. | 
 |  */ | 
 | #include <linux/mm.h> | 
 | #include <linux/io.h> | 
 | #include <asm/tlb.h> | 
 | #include <asm/mmu_context.h> | 
 |  | 
 | /* | 
 |  * Load the entry for 'addr' into the TLB and wire the entry. | 
 |  */ | 
 | void tlb_wire_entry(struct vm_area_struct *vma, unsigned long addr, pte_t pte) | 
 | { | 
 | 	unsigned long status, flags; | 
 | 	int urb; | 
 |  | 
 | 	local_irq_save(flags); | 
 |  | 
 | 	status = __raw_readl(MMUCR); | 
 | 	urb = (status & MMUCR_URB) >> MMUCR_URB_SHIFT; | 
 | 	status &= ~MMUCR_URC; | 
 |  | 
 | 	/* | 
 | 	 * Make sure we're not trying to wire the last TLB entry slot. | 
 | 	 */ | 
 | 	BUG_ON(!--urb); | 
 |  | 
 | 	urb = urb % MMUCR_URB_NENTRIES; | 
 |  | 
 | 	/* | 
 | 	 * Insert this entry into the highest non-wired TLB slot (via | 
 | 	 * the URC field). | 
 | 	 */ | 
 | 	status |= (urb << MMUCR_URC_SHIFT); | 
 | 	__raw_writel(status, MMUCR); | 
 | 	ctrl_barrier(); | 
 |  | 
 | 	/* Load the entry into the TLB */ | 
 | 	__update_tlb(vma, addr, pte); | 
 |  | 
 | 	/* ... and wire it up. */ | 
 | 	status = __raw_readl(MMUCR); | 
 |  | 
 | 	status &= ~MMUCR_URB; | 
 | 	status |= (urb << MMUCR_URB_SHIFT); | 
 |  | 
 | 	__raw_writel(status, MMUCR); | 
 | 	ctrl_barrier(); | 
 |  | 
 | 	local_irq_restore(flags); | 
 | } | 
 |  | 
 | /* | 
 |  * Unwire the last wired TLB entry. | 
 |  * | 
 |  * It should also be noted that it is not possible to wire and unwire | 
 |  * TLB entries in an arbitrary order. If you wire TLB entry N, followed | 
 |  * by entry N+1, you must unwire entry N+1 first, then entry N. In this | 
 |  * respect, it works like a stack or LIFO queue. | 
 |  */ | 
 | void tlb_unwire_entry(void) | 
 | { | 
 | 	unsigned long status, flags; | 
 | 	int urb; | 
 |  | 
 | 	local_irq_save(flags); | 
 |  | 
 | 	status = __raw_readl(MMUCR); | 
 | 	urb = (status & MMUCR_URB) >> MMUCR_URB_SHIFT; | 
 | 	status &= ~MMUCR_URB; | 
 |  | 
 | 	/* | 
 | 	 * Make sure we're not trying to unwire a TLB entry when none | 
 | 	 * have been wired. | 
 | 	 */ | 
 | 	BUG_ON(urb++ == MMUCR_URB_NENTRIES); | 
 |  | 
 | 	urb = urb % MMUCR_URB_NENTRIES; | 
 |  | 
 | 	status |= (urb << MMUCR_URB_SHIFT); | 
 | 	__raw_writel(status, MMUCR); | 
 | 	ctrl_barrier(); | 
 |  | 
 | 	local_irq_restore(flags); | 
 | } |