Skip to content
Snippets Groups Projects
c-r4k.c 38.7 KiB
Newer Older
  • Learn to ignore specific revisions
  • Linus Torvalds's avatar
    Linus Torvalds committed
    /*
     * 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.
     *
    
     * Copyright (C) 1996 David S. Miller (davem@davemloft.net)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
     * Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002 Ralf Baechle (ralf@gnu.org)
     * Copyright (C) 1999, 2000 Silicon Graphics, Inc.
     */
    
    #include <linux/hardirq.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/init.h>
    
    #include <linux/highmem.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/kernel.h>
    
    #include <linux/preempt.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/sched.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/mm.h>
    
    #include <linux/module.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/bitops.h>
    
    #include <asm/bcache.h>
    #include <asm/bootinfo.h>
    
    #include <asm/cache.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <asm/cacheops.h>
    #include <asm/cpu.h>
    #include <asm/cpu-features.h>
    
    #include <asm/cpu-type.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <asm/io.h>
    #include <asm/page.h>
    #include <asm/pgtable.h>
    #include <asm/r4kcache.h>
    
    #include <asm/sections.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <asm/mmu_context.h>
    #include <asm/war.h>
    
    #include <asm/cacheflush.h> /* for run_uncached() */
    
    #include <asm/dma-coherence.h>
    
    
    /*
     * Special Variant of smp_call_function for use by cache functions:
     *
     *  o No return value
     *  o collapses to normal function call on UP kernels
     *  o collapses to normal function call on systems with a single shared
     *    primary cache.
    
     *  o doesn't disable interrupts on the local CPU
    
    static inline void r4k_on_each_cpu(void (*func) (void *info), void *info)
    
    {
    	preempt_disable();
    
    #if !defined(CONFIG_MIPS_MT_SMP) && !defined(CONFIG_MIPS_MT_SMTC)
    
    	smp_call_function(func, info, 1);
    
    #endif
    	func(info);
    	preempt_enable();
    }
    
    
    #if defined(CONFIG_MIPS_CMP)
    #define cpu_has_safe_index_cacheops 0
    #else
    #define cpu_has_safe_index_cacheops 1
    #endif
    
    
    /*
     * Must die.
     */
    static unsigned long icache_size __read_mostly;
    static unsigned long dcache_size __read_mostly;
    static unsigned long scache_size __read_mostly;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    /*
     * Dummy cache handling routines for machines without boardcaches
     */
    
    static void cache_noop(void) {}
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    static struct bcache_ops no_sc_ops = {
    
    	.bc_enable = (void *)cache_noop,
    	.bc_disable = (void *)cache_noop,
    	.bc_wback_inv = (void *)cache_noop,
    	.bc_inv = (void *)cache_noop
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    };
    
    struct bcache_ops *bcops = &no_sc_ops;
    
    
    #define cpu_is_r4600_v1_x()	((read_c0_prid() & 0xfffffff0) == 0x00002010)
    #define cpu_is_r4600_v2_x()	((read_c0_prid() & 0xfffffff0) == 0x00002020)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    #define R4600_HIT_CACHEOP_WAR_IMPL					\
    do {									\
    	if (R4600_V2_HIT_CACHEOP_WAR && cpu_is_r4600_v2_x())		\
    		*(volatile unsigned long *)CKSEG1;			\
    	if (R4600_V1_HIT_CACHEOP_WAR)					\
    		__asm__ __volatile__("nop;nop;nop;nop");		\
    } while (0)
    
    static void (*r4k_blast_dcache_page)(unsigned long addr);
    
    static inline void r4k_blast_dcache_page_dc32(unsigned long addr)
    {
    	R4600_HIT_CACHEOP_WAR_IMPL;
    	blast_dcache32_page(addr);
    }
    
    
    static inline void r4k_blast_dcache_page_dc64(unsigned long addr)
    {
    	R4600_HIT_CACHEOP_WAR_IMPL;
    	blast_dcache64_page(addr);
    }
    
    
    static void r4k_blast_dcache_page_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long  dc_lsize = cpu_dcache_line_size();
    
    
    	if (dc_lsize == 0)
    		r4k_blast_dcache_page = (void *)cache_noop;
    	else if (dc_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_dcache_page = blast_dcache16_page;
    	else if (dc_lsize == 32)
    		r4k_blast_dcache_page = r4k_blast_dcache_page_dc32;
    
    	else if (dc_lsize == 64)
    		r4k_blast_dcache_page = r4k_blast_dcache_page_dc64;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void (* r4k_blast_dcache_page_indexed)(unsigned long addr);
    
    
    static void r4k_blast_dcache_page_indexed_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long dc_lsize = cpu_dcache_line_size();
    
    
    	if (dc_lsize == 0)
    		r4k_blast_dcache_page_indexed = (void *)cache_noop;
    	else if (dc_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_dcache_page_indexed = blast_dcache16_page_indexed;
    	else if (dc_lsize == 32)
    		r4k_blast_dcache_page_indexed = blast_dcache32_page_indexed;
    
    	else if (dc_lsize == 64)
    		r4k_blast_dcache_page_indexed = blast_dcache64_page_indexed;
    
    void (* r4k_blast_dcache)(void);
    EXPORT_SYMBOL(r4k_blast_dcache);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    static void r4k_blast_dcache_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long dc_lsize = cpu_dcache_line_size();
    
    
    	if (dc_lsize == 0)
    		r4k_blast_dcache = (void *)cache_noop;
    	else if (dc_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_dcache = blast_dcache16;
    	else if (dc_lsize == 32)
    		r4k_blast_dcache = blast_dcache32;
    
    	else if (dc_lsize == 64)
    		r4k_blast_dcache = blast_dcache64;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    /* force code alignment (used for TX49XX_ICACHE_INDEX_INV_WAR) */
    #define JUMP_TO_ALIGN(order) \
    	__asm__ __volatile__( \
    		"b\t1f\n\t" \
    		".align\t" #order "\n\t" \
    		"1:\n\t" \
    		)
    #define CACHE32_UNROLL32_ALIGN	JUMP_TO_ALIGN(10) /* 32 * 32 = 1024 */
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    #define CACHE32_UNROLL32_ALIGN2 JUMP_TO_ALIGN(11)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    static inline void blast_r4600_v1_icache32(void)
    {
    	unsigned long flags;
    
    	local_irq_save(flags);
    	blast_icache32();
    	local_irq_restore(flags);
    }
    
    static inline void tx49_blast_icache32(void)
    {
    	unsigned long start = INDEX_BASE;
    	unsigned long end = start + current_cpu_data.icache.waysize;
    	unsigned long ws_inc = 1UL << current_cpu_data.icache.waybit;
    	unsigned long ws_end = current_cpu_data.icache.ways <<
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    			       current_cpu_data.icache.waybit;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	unsigned long ws, addr;
    
    	CACHE32_UNROLL32_ALIGN2;
    	/* I'm in even chunk.  blast odd chunks */
    
    	for (ws = 0; ws < ws_end; ws += ws_inc)
    		for (addr = start + 0x400; addr < end; addr += 0x400 * 2)
    
    			cache32_unroll32(addr|ws, Index_Invalidate_I);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	CACHE32_UNROLL32_ALIGN;
    	/* I'm in odd chunk.  blast even chunks */
    
    	for (ws = 0; ws < ws_end; ws += ws_inc)
    		for (addr = start; addr < end; addr += 0x400 * 2)
    
    			cache32_unroll32(addr|ws, Index_Invalidate_I);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static inline void blast_icache32_r4600_v1_page_indexed(unsigned long page)
    {
    	unsigned long flags;
    
    	local_irq_save(flags);
    	blast_icache32_page_indexed(page);
    	local_irq_restore(flags);
    }
    
    static inline void tx49_blast_icache32_page_indexed(unsigned long page)
    {
    
    	unsigned long indexmask = current_cpu_data.icache.waysize - 1;
    	unsigned long start = INDEX_BASE + (page & indexmask);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	unsigned long end = start + PAGE_SIZE;
    	unsigned long ws_inc = 1UL << current_cpu_data.icache.waybit;
    	unsigned long ws_end = current_cpu_data.icache.ways <<
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    			       current_cpu_data.icache.waybit;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	unsigned long ws, addr;
    
    	CACHE32_UNROLL32_ALIGN2;
    	/* I'm in even chunk.  blast odd chunks */
    
    	for (ws = 0; ws < ws_end; ws += ws_inc)
    		for (addr = start + 0x400; addr < end; addr += 0x400 * 2)
    
    			cache32_unroll32(addr|ws, Index_Invalidate_I);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	CACHE32_UNROLL32_ALIGN;
    	/* I'm in odd chunk.  blast even chunks */
    
    	for (ws = 0; ws < ws_end; ws += ws_inc)
    		for (addr = start; addr < end; addr += 0x400 * 2)
    
    			cache32_unroll32(addr|ws, Index_Invalidate_I);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void (* r4k_blast_icache_page)(unsigned long addr);
    
    
    static void r4k_blast_icache_page_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long ic_lsize = cpu_icache_line_size();
    
    
    	if (ic_lsize == 0)
    		r4k_blast_icache_page = (void *)cache_noop;
    	else if (ic_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_icache_page = blast_icache16_page;
    
    	else if (ic_lsize == 32 && current_cpu_type() == CPU_LOONGSON2)
    		r4k_blast_icache_page = loongson2_blast_icache32_page;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	else if (ic_lsize == 32)
    		r4k_blast_icache_page = blast_icache32_page;
    	else if (ic_lsize == 64)
    		r4k_blast_icache_page = blast_icache64_page;
    }
    
    
    static void (* r4k_blast_icache_page_indexed)(unsigned long addr);
    
    
    static void r4k_blast_icache_page_indexed_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long ic_lsize = cpu_icache_line_size();
    
    
    	if (ic_lsize == 0)
    		r4k_blast_icache_page_indexed = (void *)cache_noop;
    	else if (ic_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_icache_page_indexed = blast_icache16_page_indexed;
    	else if (ic_lsize == 32) {
    
    Thiemo Seufer's avatar
    Thiemo Seufer committed
    		if (R4600_V1_INDEX_ICACHEOP_WAR && cpu_is_r4600_v1_x())
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			r4k_blast_icache_page_indexed =
    				blast_icache32_r4600_v1_page_indexed;
    
    Thiemo Seufer's avatar
    Thiemo Seufer committed
    		else if (TX49XX_ICACHE_INDEX_INV_WAR)
    			r4k_blast_icache_page_indexed =
    				tx49_blast_icache32_page_indexed;
    
    		else if (current_cpu_type() == CPU_LOONGSON2)
    			r4k_blast_icache_page_indexed =
    				loongson2_blast_icache32_page_indexed;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		else
    			r4k_blast_icache_page_indexed =
    				blast_icache32_page_indexed;
    	} else if (ic_lsize == 64)
    		r4k_blast_icache_page_indexed = blast_icache64_page_indexed;
    }
    
    
    void (* r4k_blast_icache)(void);
    EXPORT_SYMBOL(r4k_blast_icache);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    static void r4k_blast_icache_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long ic_lsize = cpu_icache_line_size();
    
    
    	if (ic_lsize == 0)
    		r4k_blast_icache = (void *)cache_noop;
    	else if (ic_lsize == 16)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_icache = blast_icache16;
    	else if (ic_lsize == 32) {
    		if (R4600_V1_INDEX_ICACHEOP_WAR && cpu_is_r4600_v1_x())
    			r4k_blast_icache = blast_r4600_v1_icache32;
    		else if (TX49XX_ICACHE_INDEX_INV_WAR)
    			r4k_blast_icache = tx49_blast_icache32;
    
    		else if (current_cpu_type() == CPU_LOONGSON2)
    			r4k_blast_icache = loongson2_blast_icache32;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		else
    			r4k_blast_icache = blast_icache32;
    	} else if (ic_lsize == 64)
    		r4k_blast_icache = blast_icache64;
    }
    
    static void (* r4k_blast_scache_page)(unsigned long addr);
    
    
    static void r4k_blast_scache_page_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long sc_lsize = cpu_scache_line_size();
    
    
    		r4k_blast_scache_page = (void *)cache_noop;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_scache_page = blast_scache16_page;
    	else if (sc_lsize == 32)
    		r4k_blast_scache_page = blast_scache32_page;
    	else if (sc_lsize == 64)
    		r4k_blast_scache_page = blast_scache64_page;
    	else if (sc_lsize == 128)
    		r4k_blast_scache_page = blast_scache128_page;
    }
    
    static void (* r4k_blast_scache_page_indexed)(unsigned long addr);
    
    
    static void r4k_blast_scache_page_indexed_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long sc_lsize = cpu_scache_line_size();
    
    
    		r4k_blast_scache_page_indexed = (void *)cache_noop;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_scache_page_indexed = blast_scache16_page_indexed;
    	else if (sc_lsize == 32)
    		r4k_blast_scache_page_indexed = blast_scache32_page_indexed;
    	else if (sc_lsize == 64)
    		r4k_blast_scache_page_indexed = blast_scache64_page_indexed;
    	else if (sc_lsize == 128)
    		r4k_blast_scache_page_indexed = blast_scache128_page_indexed;
    }
    
    static void (* r4k_blast_scache)(void);
    
    
    static void r4k_blast_scache_setup(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	unsigned long sc_lsize = cpu_scache_line_size();
    
    
    		r4k_blast_scache = (void *)cache_noop;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_scache = blast_scache16;
    	else if (sc_lsize == 32)
    		r4k_blast_scache = blast_scache32;
    	else if (sc_lsize == 64)
    		r4k_blast_scache = blast_scache64;
    	else if (sc_lsize == 128)
    		r4k_blast_scache = blast_scache128;
    }
    
    static inline void local_r4k___flush_cache_all(void * args)
    {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	case CPU_R4000SC:
    	case CPU_R4000MC:
    	case CPU_R4400SC:
    	case CPU_R4400MC:
    	case CPU_R10000:
    	case CPU_R12000:
    
    Kumba's avatar
    Kumba committed
    	case CPU_R14000:
    
    		/*
    		 * These caches are inclusive caches, that is, if something
    		 * is not cached in the S-cache, we know it also won't be
    		 * in one of the primary caches.
    		 */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_scache();
    
    		break;
    
    	default:
    		r4k_blast_dcache();
    		r4k_blast_icache();
    		break;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    }
    
    static void r4k___flush_cache_all(void)
    {
    
    	r4k_on_each_cpu(local_r4k___flush_cache_all, NULL);
    
    static inline int has_valid_asid(const struct mm_struct *mm)
    {
    #if defined(CONFIG_MIPS_MT_SMP) || defined(CONFIG_MIPS_MT_SMTC)
    	int i;
    
    	for_each_online_cpu(i)
    		if (cpu_context(i, mm))
    			return 1;
    
    	return 0;
    #else
    	return cpu_context(smp_processor_id(), mm);
    #endif
    }
    
    
    static void r4k__flush_cache_vmap(void)
    {
    	r4k_blast_dcache();
    }
    
    static void r4k__flush_cache_vunmap(void)
    {
    	r4k_blast_dcache();
    }
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    static inline void local_r4k_flush_cache_range(void * args)
    {
    	struct vm_area_struct *vma = args;
    
    	int exec = vma->vm_flags & VM_EXEC;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    	if (!(has_valid_asid(vma->vm_mm)))
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		return;
    
    
    	r4k_blast_dcache();
    
    	if (exec)
    		r4k_blast_icache();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void r4k_flush_cache_range(struct vm_area_struct *vma,
    	unsigned long start, unsigned long end)
    {
    
    	int exec = vma->vm_flags & VM_EXEC;
    
    	if (cpu_has_dc_aliases || (exec && !cpu_has_ic_fills_f_dc))
    
    		r4k_on_each_cpu(local_r4k_flush_cache_range, vma);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static inline void local_r4k_flush_cache_mm(void * args)
    {
    	struct mm_struct *mm = args;
    
    
    	if (!has_valid_asid(mm))
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		return;
    
    	/*
    	 * Kludge alert.  For obscure reasons R4000SC and R4400SC go nuts if we
    	 * only flush the primary caches but R10000 and R12000 behave sane ...
    
    	 * R4000SC and R4400SC indexed S-cache ops also invalidate primary
    	 * caches, so we can bail out early.
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	 */
    
    	if (current_cpu_type() == CPU_R4000SC ||
    	    current_cpu_type() == CPU_R4000MC ||
    	    current_cpu_type() == CPU_R4400SC ||
    	    current_cpu_type() == CPU_R4400MC) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_scache();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void r4k_flush_cache_mm(struct mm_struct *mm)
    {
    	if (!cpu_has_dc_aliases)
    		return;
    
    
    	r4k_on_each_cpu(local_r4k_flush_cache_mm, mm);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    struct flush_cache_page_args {
    	struct vm_area_struct *vma;
    
    	unsigned long pfn;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    };
    
    static inline void local_r4k_flush_cache_page(void *args)
    {
    	struct flush_cache_page_args *fcp_args = args;
    	struct vm_area_struct *vma = fcp_args->vma;
    
    	unsigned long addr = fcp_args->addr;
    
    	struct page *page = pfn_to_page(fcp_args->pfn);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	int exec = vma->vm_flags & VM_EXEC;
    	struct mm_struct *mm = vma->vm_mm;
    
    	int map_coherent = 0;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	pgd_t *pgdp;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	pmd_t *pmdp;
    	pte_t *ptep;
    
    	void *vaddr;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    	/*
    	 * If ownes no valid ASID yet, cannot possibly have gotten
    	 * this page into the cache.
    	 */
    
    	if (!has_valid_asid(mm))
    
    	addr &= PAGE_MASK;
    	pgdp = pgd_offset(mm, addr);
    	pudp = pud_offset(pgdp, addr);
    	pmdp = pmd_offset(pudp, addr);
    	ptep = pte_offset(pmdp, addr);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    	/*
    	 * If the page isn't marked valid, the page cannot possibly be
    	 * in the cache.
    	 */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		return;
    
    
    	if ((mm == current->active_mm) && (pte_val(*ptep) & _PAGE_VALID))
    		vaddr = NULL;
    	else {
    		/*
    		 * Use kmap_coherent or kmap_atomic to do flushes for
    		 * another ASID than the current one.
    		 */
    
    		map_coherent = (cpu_has_dc_aliases &&
    				page_mapped(page) && !Page_dcache_dirty(page));
    		if (map_coherent)
    
    			vaddr = kmap_coherent(page, addr);
    		else
    
    			vaddr = kmap_atomic(page);
    
    		addr = (unsigned long)vaddr;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    
    	if (cpu_has_dc_aliases || (exec && !cpu_has_ic_fills_f_dc)) {
    
    		r4k_blast_dcache_page(addr);
    
    		if (exec && !cpu_icache_snoops_remote_store)
    			r4k_blast_scache_page(addr);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    	if (exec) {
    
    		if (vaddr && cpu_has_vtag_icache && mm == current->active_mm) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			int cpu = smp_processor_id();
    
    
    			if (cpu_context(cpu, mm) != 0)
    				drop_mmu_context(mm, cpu);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		} else
    
    			r4k_blast_icache_page(addr);
    	}
    
    	if (vaddr) {
    
    		if (map_coherent)
    
    			kunmap_coherent();
    		else
    
    			kunmap_atomic(vaddr);
    
    static void r4k_flush_cache_page(struct vm_area_struct *vma,
    	unsigned long addr, unsigned long pfn)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	struct flush_cache_page_args args;
    
    	args.vma = vma;
    
    	args.pfn = pfn;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    	r4k_on_each_cpu(local_r4k_flush_cache_page, &args);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static inline void local_r4k_flush_data_cache_page(void * addr)
    {
    	r4k_blast_dcache_page((unsigned long) addr);
    }
    
    static void r4k_flush_data_cache_page(unsigned long addr)
    {
    
    	if (in_atomic())
    		local_r4k_flush_data_cache_page((void *)addr);
    	else
    
    		r4k_on_each_cpu(local_r4k_flush_data_cache_page, (void *) addr);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    struct flush_icache_range_args {
    
    	unsigned long start;
    	unsigned long end;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    };
    
    
    static inline void local_r4k_flush_icache_range(unsigned long start, unsigned long end)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	if (!cpu_has_ic_fills_f_dc) {
    
    		if (end - start >= dcache_size) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			r4k_blast_dcache();
    		} else {
    
    			R4600_HIT_CACHEOP_WAR_IMPL;
    
    			protected_blast_dcache_range(start, end);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		}
    	}
    
    	if (end - start > icache_size)
    		r4k_blast_icache();
    
    	else {
    		switch (boot_cpu_type()) {
    		case CPU_LOONGSON2:
    
    			protected_loongson2_blast_icache_range(start, end);
    
    			protected_blast_icache_range(start, end);
    
    static inline void local_r4k_flush_icache_range_ipi(void *args)
    {
    	struct flush_icache_range_args *fir_args = args;
    	unsigned long start = fir_args->start;
    	unsigned long end = fir_args->end;
    
    	local_r4k_flush_icache_range(start, end);
    }
    
    
    static void r4k_flush_icache_range(unsigned long start, unsigned long end)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	struct flush_icache_range_args args;
    
    	args.start = start;
    	args.end = end;
    
    
    	r4k_on_each_cpu(local_r4k_flush_icache_range_ipi, &args);
    
    	instruction_hazard();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    #ifdef CONFIG_DMA_NONCOHERENT
    
    static void r4k_dma_cache_wback_inv(unsigned long addr, unsigned long size)
    {
    	/* Catch bad driver code */
    	BUG_ON(size == 0);
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			r4k_blast_scache();
    
    		else
    			blast_scache_range(addr, addr + size);
    
    		__sync();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		return;
    	}
    
    	/*
    	 * Either no secondary cache or the available caches don't have the
    	 * subset property so we have to flush the primary caches
    	 * explicitly
    	 */
    
    	if (cpu_has_safe_index_cacheops && size >= dcache_size) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_dcache();
    	} else {
    		R4600_HIT_CACHEOP_WAR_IMPL;
    
    		blast_dcache_range(addr, addr + size);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    	bc_wback_inv(addr, size);
    
    	__sync();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void r4k_dma_cache_inv(unsigned long addr, unsigned long size)
    {
    	/* Catch bad driver code */
    	BUG_ON(size == 0);
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			r4k_blast_scache();
    
    		else {
    			/*
    			 * There is no clearly documented alignment requirement
    			 * for the cache instruction on MIPS processors and
    			 * some processors, among them the RM5200 and RM7000
    			 * QED processors will throw an address error for cache
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    			 * hit ops with insufficient alignment.	 Solved by
    
    			 * aligning the address to cache line size.
    			 */
    
    			blast_inv_scache_range(addr, addr + size);
    
    		__sync();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		return;
    	}
    
    
    	if (cpu_has_safe_index_cacheops && size >= dcache_size) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		r4k_blast_dcache();
    	} else {
    		R4600_HIT_CACHEOP_WAR_IMPL;
    
    		blast_inv_dcache_range(addr, addr + size);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    	bc_inv(addr, size);
    
    	__sync();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    #endif /* CONFIG_DMA_NONCOHERENT */
    
    /*
     * While we're protected against bad userland addresses we don't care
     * very much about what happens in that case.  Usually a segmentation
     * fault will dump the process later on anyway ...
     */
    static void local_r4k_flush_cache_sigtramp(void * arg)
    {
    
    Thiemo Seufer's avatar
    Thiemo Seufer committed
    	unsigned long ic_lsize = cpu_icache_line_size();
    	unsigned long dc_lsize = cpu_dcache_line_size();
    	unsigned long sc_lsize = cpu_scache_line_size();
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	unsigned long addr = (unsigned long) arg;
    
    	R4600_HIT_CACHEOP_WAR_IMPL;
    
    	if (dc_lsize)
    		protected_writeback_dcache_line(addr & ~(dc_lsize - 1));
    
    	if (!cpu_icache_snoops_remote_store && scache_size)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		protected_writeback_scache_line(addr & ~(sc_lsize - 1));
    
    	if (ic_lsize)
    		protected_flush_icache_line(addr & ~(ic_lsize - 1));
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	if (MIPS4K_ICACHE_REFILL_WAR) {
    		__asm__ __volatile__ (
    			".set push\n\t"
    			".set noat\n\t"
    			".set mips3\n\t"
    
    #ifdef CONFIG_32BIT
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			"la	$at,1f\n\t"
    #endif
    
    #ifdef CONFIG_64BIT
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			"dla	$at,1f\n\t"
    #endif
    			"cache	%0,($at)\n\t"
    			"nop; nop; nop\n"
    			"1:\n\t"
    			".set pop"
    			:
    			: "i" (Hit_Invalidate_I));
    	}
    	if (MIPS_CACHE_SYNC_WAR)
    		__asm__ __volatile__ ("sync");
    }
    
    static void r4k_flush_cache_sigtramp(unsigned long addr)
    {
    
    	r4k_on_each_cpu(local_r4k_flush_cache_sigtramp, (void *) addr);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    static void r4k_flush_icache_all(void)
    {
    	if (cpu_has_vtag_icache)
    		r4k_blast_icache();
    }
    
    
    struct flush_kernel_vmap_range_args {
    	unsigned long	vaddr;
    	int		size;
    };
    
    static inline void local_r4k_flush_kernel_vmap_range(void *args)
    {
    	struct flush_kernel_vmap_range_args *vmra = args;
    	unsigned long vaddr = vmra->vaddr;
    	int size = vmra->size;
    
    	/*
    	 * Aliases only affect the primary caches so don't bother with
    	 * S-caches or T-caches.
    	 */
    	if (cpu_has_safe_index_cacheops && size >= dcache_size)
    		r4k_blast_dcache();
    	else {
    		R4600_HIT_CACHEOP_WAR_IMPL;
    		blast_dcache_range(vaddr, vaddr + size);
    	}
    }
    
    static void r4k_flush_kernel_vmap_range(unsigned long vaddr, int size)
    {
    	struct flush_kernel_vmap_range_args args;
    
    	args.vaddr = (unsigned long) vaddr;
    	args.size = size;
    
    	r4k_on_each_cpu(local_r4k_flush_kernel_vmap_range, &args);
    }
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    static inline void rm7k_erratum31(void)
    {
    	const unsigned long ic_lsize = 32;
    	unsigned long addr;
    
    	/* RM7000 erratum #31. The icache is screwed at startup. */
    	write_c0_taglo(0);
    	write_c0_taghi(0);
    
    	for (addr = INDEX_BASE; addr <= INDEX_BASE + 4096; addr += ic_lsize) {
    		__asm__ __volatile__ (
    
    Thiemo Seufer's avatar
    Thiemo Seufer committed
    			".set push\n\t"
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			".set noreorder\n\t"
    			".set mips3\n\t"
    			"cache\t%1, 0(%0)\n\t"
    			"cache\t%1, 0x1000(%0)\n\t"
    			"cache\t%1, 0x2000(%0)\n\t"
    			"cache\t%1, 0x3000(%0)\n\t"
    			"cache\t%2, 0(%0)\n\t"
    			"cache\t%2, 0x1000(%0)\n\t"
    			"cache\t%2, 0x2000(%0)\n\t"
    			"cache\t%2, 0x3000(%0)\n\t"
    			"cache\t%1, 0(%0)\n\t"
    			"cache\t%1, 0x1000(%0)\n\t"
    			"cache\t%1, 0x2000(%0)\n\t"
    			"cache\t%1, 0x3000(%0)\n\t"
    
    Thiemo Seufer's avatar
    Thiemo Seufer committed
    			".set pop\n"
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			:
    			: "r" (addr), "i" (Index_Store_Tag_I), "i" (Fill));
    	}
    }
    
    
    static inline void alias_74k_erratum(struct cpuinfo_mips *c)
    {
    
    	unsigned int imp = c->processor_id & PRID_IMP_MASK;
    	unsigned int rev = c->processor_id & PRID_REV_MASK;
    
    
    	/*
    	 * Early versions of the 74K do not update the cache tags on a
    	 * vtag miss/ptag hit which can occur in the case of KSEG0/KUSEG
    	 * aliases. In this case it is better to treat the cache as always
    	 * having aliases.
    	 */
    
    	switch (imp) {
    	case PRID_IMP_74K:
    		if (rev <= PRID_REV_ENCODE_332(2, 4, 0))
    			c->dcache.flags |= MIPS_CACHE_VTAG;
    		if (rev == PRID_REV_ENCODE_332(2, 4, 0))
    			write_c0_config6(read_c0_config6() | MIPS_CONF6_SYND);
    		break;
    	case PRID_IMP_1074K:
    		if (rev <= PRID_REV_ENCODE_332(1, 1, 0)) {
    			c->dcache.flags |= MIPS_CACHE_VTAG;
    			write_c0_config6(read_c0_config6() | MIPS_CONF6_SYND);
    		}
    		break;
    	default:
    		BUG();
    
    static char *way_string[] = { NULL, "direct mapped", "2-way",
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	"3-way", "4-way", "5-way", "6-way", "7-way", "8-way"
    };
    
    
    static void probe_pcache(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	struct cpuinfo_mips *c = &current_cpu_data;
    	unsigned int config = read_c0_config();
    	unsigned int prid = read_c0_prid();
    	unsigned long config1;
    	unsigned int lsize;
    
    
    	switch (current_cpu_type()) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	case CPU_R4600:			/* QED style two way caches? */
    	case CPU_R4700:
    	case CPU_R5000:
    	case CPU_NEVADA:
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 2;
    
    		c->icache.waybit = __ffs(icache_size/2);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		dcache_size = 1 << (12 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 2;
    
    		c->dcache.waybit= __ffs(dcache_size/2);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P;
    		break;
    
    	case CPU_R5432:
    	case CPU_R5500:
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 2;
    		c->icache.waybit= 0;
    
    		dcache_size = 1 << (12 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 2;
    		c->dcache.waybit = 0;
    
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P | MIPS_CPU_PREFETCH;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		break;
    
    	case CPU_TX49XX:
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 4;
    		c->icache.waybit= 0;
    
    		dcache_size = 1 << (12 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 4;
    		c->dcache.waybit = 0;
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P;
    
    		c->options |= MIPS_CPU_PREFETCH;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		break;
    
    	case CPU_R4000PC:
    	case CPU_R4000SC:
    	case CPU_R4000MC:
    	case CPU_R4400PC:
    	case CPU_R4400SC:
    	case CPU_R4400MC:
    	case CPU_R4300:
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 1;
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    		c->icache.waybit = 0;	/* doesn't matter */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		dcache_size = 1 << (12 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 1;
    		c->dcache.waybit = 0;	/* does not matter */
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P;
    		break;
    
    	case CPU_R10000:
    	case CPU_R12000:
    
    Kumba's avatar
    Kumba committed
    	case CPU_R14000:
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		icache_size = 1 << (12 + ((config & R10K_CONF_IC) >> 29));
    		c->icache.linesz = 64;
    		c->icache.ways = 2;
    		c->icache.waybit = 0;
    
    		dcache_size = 1 << (12 + ((config & R10K_CONF_DC) >> 26));
    		c->dcache.linesz = 32;
    		c->dcache.ways = 2;
    		c->dcache.waybit = 0;
    
    		c->options |= MIPS_CPU_PREFETCH;
    		break;
    
    	case CPU_VR4133:
    
    		write_c0_config(config & ~VR41_CONF_P4K);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	case CPU_VR4131:
    		/* Workaround for cache instruction bug of VR4131 */
    		if (c->processor_id == 0x0c80U || c->processor_id == 0x0c81U ||
    		    c->processor_id == 0x0c82U) {
    
    			config |= 0x00400000U;
    			if (c->processor_id == 0x0c80U)
    				config |= VR41_CONF_BP;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			write_c0_config(config);
    
    		} else
    			c->options |= MIPS_CPU_CACHE_CDEX_P;
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		icache_size = 1 << (10 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 2;
    
    		c->icache.waybit = __ffs(icache_size/2);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		dcache_size = 1 << (10 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 2;
    
    		c->dcache.waybit = __ffs(dcache_size/2);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		break;
    
    	case CPU_VR41XX:
    	case CPU_VR4111:
    	case CPU_VR4121:
    	case CPU_VR4122:
    	case CPU_VR4181:
    	case CPU_VR4181A:
    		icache_size = 1 << (10 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 1;
    
    Ralf Baechle's avatar
    Ralf Baechle committed
    		c->icache.waybit = 0;	/* doesn't matter */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		dcache_size = 1 << (10 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 1;
    		c->dcache.waybit = 0;	/* does not matter */
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P;
    		break;
    
    	case CPU_RM7000:
    		rm7k_erratum31();
    
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		c->icache.ways = 4;
    
    		c->icache.waybit = __ffs(icache_size / c->icache.ways);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		dcache_size = 1 << (12 + ((config & CONF_DC) >> 6));
    		c->dcache.linesz = 16 << ((config & CONF_DB) >> 4);
    		c->dcache.ways = 4;
    
    		c->dcache.waybit = __ffs(dcache_size / c->dcache.ways);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    		c->options |= MIPS_CPU_CACHE_CDEX_P;
    		c->options |= MIPS_CPU_PREFETCH;
    		break;
    
    
    	case CPU_LOONGSON2:
    		icache_size = 1 << (12 + ((config & CONF_IC) >> 9));
    		c->icache.linesz = 16 << ((config & CONF_IB) >> 5);
    		if (prid & 0x3)
    			c->icache.ways = 4;
    		else