diff options
-rw-r--r-- | arch/sh/Kconfig | 1 | ||||
-rw-r--r-- | arch/sh/Kconfig.cpu | 3 | ||||
-rw-r--r-- | arch/sh/include/asm/cpu-features.h | 1 | ||||
-rw-r--r-- | arch/sh/include/asm/mmu_context.h | 15 | ||||
-rw-r--r-- | arch/sh/include/asm/mmu_context_32.h | 12 | ||||
-rw-r--r-- | arch/sh/include/cpu-sh4/cpu/mmu_context.h | 35 | ||||
-rw-r--r-- | arch/sh/kernel/cpu/sh4/probe.c | 2 | ||||
-rw-r--r-- | arch/sh/kernel/setup.c | 2 | ||||
-rw-r--r-- | arch/sh/mm/Makefile_32 | 6 | ||||
-rw-r--r-- | arch/sh/mm/tlb-pteaex.c | 99 |
10 files changed, 148 insertions, 28 deletions
diff --git a/arch/sh/Kconfig b/arch/sh/Kconfig index a0c879d17fd6..6c56495fd158 100644 --- a/arch/sh/Kconfig +++ b/arch/sh/Kconfig | |||
@@ -365,6 +365,7 @@ config CPU_SUBTYPE_SH7786 | |||
365 | bool "Support SH7786 processor" | 365 | bool "Support SH7786 processor" |
366 | select CPU_SH4A | 366 | select CPU_SH4A |
367 | select CPU_SHX3 | 367 | select CPU_SHX3 |
368 | select CPU_HAS_PTEAEX | ||
368 | select ARCH_SPARSEMEM_ENABLE | 369 | select ARCH_SPARSEMEM_ENABLE |
369 | select SYS_SUPPORTS_NUMA | 370 | select SYS_SUPPORTS_NUMA |
370 | 371 | ||
diff --git a/arch/sh/Kconfig.cpu b/arch/sh/Kconfig.cpu index 0e27fe3b182b..c7d704381a6d 100644 --- a/arch/sh/Kconfig.cpu +++ b/arch/sh/Kconfig.cpu | |||
@@ -104,6 +104,9 @@ config CPU_HAS_SR_RB | |||
104 | config CPU_HAS_PTEA | 104 | config CPU_HAS_PTEA |
105 | bool | 105 | bool |
106 | 106 | ||
107 | config CPU_HAS_PTEAEX | ||
108 | bool | ||
109 | |||
107 | config CPU_HAS_DSP | 110 | config CPU_HAS_DSP |
108 | bool | 111 | bool |
109 | 112 | ||
diff --git a/arch/sh/include/asm/cpu-features.h b/arch/sh/include/asm/cpu-features.h index 86308aa39731..694abe490edb 100644 --- a/arch/sh/include/asm/cpu-features.h +++ b/arch/sh/include/asm/cpu-features.h | |||
@@ -21,5 +21,6 @@ | |||
21 | #define CPU_HAS_LLSC 0x0040 /* movli.l/movco.l */ | 21 | #define CPU_HAS_LLSC 0x0040 /* movli.l/movco.l */ |
22 | #define CPU_HAS_L2_CACHE 0x0080 /* Secondary cache / URAM */ | 22 | #define CPU_HAS_L2_CACHE 0x0080 /* Secondary cache / URAM */ |
23 | #define CPU_HAS_OP32 0x0100 /* 32-bit instruction support */ | 23 | #define CPU_HAS_OP32 0x0100 /* 32-bit instruction support */ |
24 | #define CPU_HAS_PTEAEX 0x0200 /* PTE ASID Extension support */ | ||
24 | 25 | ||
25 | #endif /* __ASM_SH_CPU_FEATURES_H */ | 26 | #endif /* __ASM_SH_CPU_FEATURES_H */ |
diff --git a/arch/sh/include/asm/mmu_context.h b/arch/sh/include/asm/mmu_context.h index 5d9157bd474d..2a9c55f1a83f 100644 --- a/arch/sh/include/asm/mmu_context.h +++ b/arch/sh/include/asm/mmu_context.h | |||
@@ -19,13 +19,18 @@ | |||
19 | * (a) TLB cache version (or round, cycle whatever expression you like) | 19 | * (a) TLB cache version (or round, cycle whatever expression you like) |
20 | * (b) ASID (Address Space IDentifier) | 20 | * (b) ASID (Address Space IDentifier) |
21 | */ | 21 | */ |
22 | #ifdef CONFIG_CPU_HAS_PTEAEX | ||
23 | #define MMU_CONTEXT_ASID_MASK 0x0000ffff | ||
24 | #else | ||
22 | #define MMU_CONTEXT_ASID_MASK 0x000000ff | 25 | #define MMU_CONTEXT_ASID_MASK 0x000000ff |
23 | #define MMU_CONTEXT_VERSION_MASK 0xffffff00 | 26 | #endif |
24 | #define MMU_CONTEXT_FIRST_VERSION 0x00000100 | ||
25 | #define NO_CONTEXT 0UL | ||
26 | 27 | ||
27 | /* ASID is 8-bit value, so it can't be 0x100 */ | 28 | #define MMU_CONTEXT_VERSION_MASK (~0UL & ~MMU_CONTEXT_ASID_MASK) |
28 | #define MMU_NO_ASID 0x100 | 29 | #define MMU_CONTEXT_FIRST_VERSION (MMU_CONTEXT_ASID_MASK + 1) |
30 | |||
31 | /* Impossible ASID value, to differentiate from NO_CONTEXT. */ | ||
32 | #define MMU_NO_ASID MMU_CONTEXT_FIRST_VERSION | ||
33 | #define NO_CONTEXT 0UL | ||
29 | 34 | ||
30 | #define asid_cache(cpu) (cpu_data[cpu].asid_cache) | 35 | #define asid_cache(cpu) (cpu_data[cpu].asid_cache) |
31 | 36 | ||
diff --git a/arch/sh/include/asm/mmu_context_32.h b/arch/sh/include/asm/mmu_context_32.h index f4f9aebd68b7..8ef800c549ab 100644 --- a/arch/sh/include/asm/mmu_context_32.h +++ b/arch/sh/include/asm/mmu_context_32.h | |||
@@ -10,6 +10,17 @@ static inline void destroy_context(struct mm_struct *mm) | |||
10 | /* Do nothing */ | 10 | /* Do nothing */ |
11 | } | 11 | } |
12 | 12 | ||
13 | #ifdef CONFIG_CPU_HAS_PTEAEX | ||
14 | static inline void set_asid(unsigned long asid) | ||
15 | { | ||
16 | __raw_writel(asid, MMU_PTEAEX); | ||
17 | } | ||
18 | |||
19 | static inline unsigned long get_asid(void) | ||
20 | { | ||
21 | return __raw_readl(MMU_PTEAEX) & MMU_CONTEXT_ASID_MASK; | ||
22 | } | ||
23 | #else | ||
13 | static inline void set_asid(unsigned long asid) | 24 | static inline void set_asid(unsigned long asid) |
14 | { | 25 | { |
15 | unsigned long __dummy; | 26 | unsigned long __dummy; |
@@ -33,6 +44,7 @@ static inline unsigned long get_asid(void) | |||
33 | asid &= MMU_CONTEXT_ASID_MASK; | 44 | asid &= MMU_CONTEXT_ASID_MASK; |
34 | return asid; | 45 | return asid; |
35 | } | 46 | } |
47 | #endif /* CONFIG_CPU_HAS_PTEAEX */ | ||
36 | 48 | ||
37 | /* MMU_TTB is used for optimizing the fault handling. */ | 49 | /* MMU_TTB is used for optimizing the fault handling. */ |
38 | static inline void set_TTB(pgd_t *pgd) | 50 | static inline void set_TTB(pgd_t *pgd) |
diff --git a/arch/sh/include/cpu-sh4/cpu/mmu_context.h b/arch/sh/include/cpu-sh4/cpu/mmu_context.h index 9ea8eb27b18e..3ce7ef6c2978 100644 --- a/arch/sh/include/cpu-sh4/cpu/mmu_context.h +++ b/arch/sh/include/cpu-sh4/cpu/mmu_context.h | |||
@@ -14,28 +14,35 @@ | |||
14 | #define MMU_PTEL 0xFF000004 /* Page table entry register LOW */ | 14 | #define MMU_PTEL 0xFF000004 /* Page table entry register LOW */ |
15 | #define MMU_TTB 0xFF000008 /* Translation table base register */ | 15 | #define MMU_TTB 0xFF000008 /* Translation table base register */ |
16 | #define MMU_TEA 0xFF00000C /* TLB Exception Address */ | 16 | #define MMU_TEA 0xFF00000C /* TLB Exception Address */ |
17 | #define MMU_PTEA 0xFF000034 /* Page table entry assistance register */ | 17 | #define MMU_PTEA 0xFF000034 /* PTE assistance register */ |
18 | #define MMU_PTEAEX 0xFF00007C /* PTE ASID extension register */ | ||
18 | 19 | ||
19 | #define MMUCR 0xFF000010 /* MMU Control Register */ | 20 | #define MMUCR 0xFF000010 /* MMU Control Register */ |
20 | 21 | ||
21 | #define MMU_ITLB_ADDRESS_ARRAY 0xF2000000 | ||
22 | #define MMU_UTLB_ADDRESS_ARRAY 0xF6000000 | 22 | #define MMU_UTLB_ADDRESS_ARRAY 0xF6000000 |
23 | #define MMU_UTLB_ADDRESS_ARRAY2 0xF6800000 | ||
23 | #define MMU_PAGE_ASSOC_BIT 0x80 | 24 | #define MMU_PAGE_ASSOC_BIT 0x80 |
24 | 25 | ||
25 | #define MMUCR_TI (1<<2) | 26 | #define MMUCR_TI (1<<2) |
26 | 27 | ||
27 | #ifdef CONFIG_X2TLB | ||
28 | #define MMUCR_ME (1 << 7) | ||
29 | #else | ||
30 | #define MMUCR_ME (0) | ||
31 | #endif | ||
32 | |||
33 | #if defined(CONFIG_32BIT) && defined(CONFIG_CPU_SUBTYPE_ST40) | 28 | #if defined(CONFIG_32BIT) && defined(CONFIG_CPU_SUBTYPE_ST40) |
34 | #define MMUCR_SE (1 << 4) | 29 | #define MMUCR_SE (1 << 4) |
35 | #else | 30 | #else |
36 | #define MMUCR_SE (0) | 31 | #define MMUCR_SE (0) |
37 | #endif | 32 | #endif |
38 | 33 | ||
34 | #ifdef CONFIG_CPU_HAS_PTEAEX | ||
35 | #define MMUCR_AEX (1 << 6) | ||
36 | #else | ||
37 | #define MMUCR_AEX (0) | ||
38 | #endif | ||
39 | |||
40 | #ifdef CONFIG_X2TLB | ||
41 | #define MMUCR_ME (1 << 7) | ||
42 | #else | ||
43 | #define MMUCR_ME (0) | ||
44 | #endif | ||
45 | |||
39 | #ifdef CONFIG_SH_STORE_QUEUES | 46 | #ifdef CONFIG_SH_STORE_QUEUES |
40 | #define MMUCR_SQMD (1 << 9) | 47 | #define MMUCR_SQMD (1 << 9) |
41 | #else | 48 | #else |
@@ -43,17 +50,7 @@ | |||
43 | #endif | 50 | #endif |
44 | 51 | ||
45 | #define MMU_NTLB_ENTRIES 64 | 52 | #define MMU_NTLB_ENTRIES 64 |
46 | #define MMU_CONTROL_INIT (0x05|MMUCR_SQMD|MMUCR_ME|MMUCR_SE) | 53 | #define MMU_CONTROL_INIT (0x05|MMUCR_SQMD|MMUCR_ME|MMUCR_SE|MMUCR_AEX) |
47 | |||
48 | #define MMU_ITLB_DATA_ARRAY 0xF3000000 | ||
49 | #define MMU_UTLB_DATA_ARRAY 0xF7000000 | ||
50 | |||
51 | #define MMU_UTLB_ENTRIES 64 | ||
52 | #define MMU_U_ENTRY_SHIFT 8 | ||
53 | #define MMU_UTLB_VALID 0x100 | ||
54 | #define MMU_ITLB_ENTRIES 4 | ||
55 | #define MMU_I_ENTRY_SHIFT 8 | ||
56 | #define MMU_ITLB_VALID 0x100 | ||
57 | 54 | ||
58 | #define TRA 0xff000020 | 55 | #define TRA 0xff000020 |
59 | #define EXPEVT 0xff000024 | 56 | #define EXPEVT 0xff000024 |
diff --git a/arch/sh/kernel/cpu/sh4/probe.c b/arch/sh/kernel/cpu/sh4/probe.c index 2bd0ec962639..3d3a3c4425a9 100644 --- a/arch/sh/kernel/cpu/sh4/probe.c +++ b/arch/sh/kernel/cpu/sh4/probe.c | |||
@@ -134,7 +134,7 @@ int __init detect_cpu_and_cache_system(void) | |||
134 | boot_cpu_data.icache.ways = 4; | 134 | boot_cpu_data.icache.ways = 4; |
135 | boot_cpu_data.dcache.ways = 4; | 135 | boot_cpu_data.dcache.ways = 4; |
136 | boot_cpu_data.flags |= CPU_HAS_FPU | CPU_HAS_PERF_COUNTER | | 136 | boot_cpu_data.flags |= CPU_HAS_FPU | CPU_HAS_PERF_COUNTER | |
137 | CPU_HAS_LLSC; | 137 | CPU_HAS_LLSC | CPU_HAS_PTEAEX; |
138 | break; | 138 | break; |
139 | case 0x3008: | 139 | case 0x3008: |
140 | boot_cpu_data.icache.ways = 4; | 140 | boot_cpu_data.icache.ways = 4; |
diff --git a/arch/sh/kernel/setup.c b/arch/sh/kernel/setup.c index 61ab2a7f8647..24c60251f680 100644 --- a/arch/sh/kernel/setup.c +++ b/arch/sh/kernel/setup.c | |||
@@ -449,7 +449,7 @@ EXPORT_SYMBOL(get_cpu_subtype); | |||
449 | /* Symbolic CPU flags, keep in sync with asm/cpu-features.h */ | 449 | /* Symbolic CPU flags, keep in sync with asm/cpu-features.h */ |
450 | static const char *cpu_flags[] = { | 450 | static const char *cpu_flags[] = { |
451 | "none", "fpu", "p2flush", "mmuassoc", "dsp", "perfctr", | 451 | "none", "fpu", "p2flush", "mmuassoc", "dsp", "perfctr", |
452 | "ptea", "llsc", "l2", "op32", NULL | 452 | "ptea", "llsc", "l2", "op32", "pteaex", NULL |
453 | }; | 453 | }; |
454 | 454 | ||
455 | static void show_cpuflags(struct seq_file *m, struct sh_cpuinfo *c) | 455 | static void show_cpuflags(struct seq_file *m, struct sh_cpuinfo *c) |
diff --git a/arch/sh/mm/Makefile_32 b/arch/sh/mm/Makefile_32 index 469ff1672451..986a1e055834 100644 --- a/arch/sh/mm/Makefile_32 +++ b/arch/sh/mm/Makefile_32 | |||
@@ -25,8 +25,10 @@ obj-$(CONFIG_CPU_SH4) += cache-debugfs.o | |||
25 | endif | 25 | endif |
26 | 26 | ||
27 | ifdef CONFIG_MMU | 27 | ifdef CONFIG_MMU |
28 | obj-$(CONFIG_CPU_SH3) += tlb-sh3.o | 28 | tlb-$(CONFIG_CPU_SH3) := tlb-sh3.o |
29 | obj-$(CONFIG_CPU_SH4) += tlb-sh4.o | 29 | tlb-$(CONFIG_CPU_SH4) := tlb-sh4.o |
30 | tlb-$(CONFIG_CPU_HAS_PTEAEX) := tlb-pteaex.o | ||
31 | obj-y += $(tlb-y) | ||
30 | ifndef CONFIG_CACHE_OFF | 32 | ifndef CONFIG_CACHE_OFF |
31 | obj-$(CONFIG_CPU_SH4) += pg-sh4.o | 33 | obj-$(CONFIG_CPU_SH4) += pg-sh4.o |
32 | obj-$(CONFIG_SH7705_CACHE_32KB) += pg-sh7705.o | 34 | obj-$(CONFIG_SH7705_CACHE_32KB) += pg-sh7705.o |
diff --git a/arch/sh/mm/tlb-pteaex.c b/arch/sh/mm/tlb-pteaex.c new file mode 100644 index 000000000000..5c9b2d781e08 --- /dev/null +++ b/arch/sh/mm/tlb-pteaex.c | |||
@@ -0,0 +1,99 @@ | |||
1 | /* | ||
2 | * arch/sh/mm/tlb-pteaex.c | ||
3 | * | ||
4 | * TLB operations for SH-X3 CPUs featuring PTE ASID Extensions. | ||
5 | * | ||
6 | * Copyright (C) 2009 Paul Mundt | ||
7 | * | ||
8 | * This file is subject to the terms and conditions of the GNU General Public | ||
9 | * License. See the file "COPYING" in the main directory of this archive | ||
10 | * for more details. | ||
11 | */ | ||
12 | #include <linux/kernel.h> | ||
13 | #include <linux/mm.h> | ||
14 | #include <linux/io.h> | ||
15 | #include <asm/system.h> | ||
16 | #include <asm/mmu_context.h> | ||
17 | #include <asm/cacheflush.h> | ||
18 | |||
19 | void update_mmu_cache(struct vm_area_struct * vma, | ||
20 | unsigned long address, pte_t pte) | ||
21 | { | ||
22 | unsigned long flags; | ||
23 | unsigned long pteval; | ||
24 | unsigned long vpn; | ||
25 | |||
26 | /* Ptrace may call this routine. */ | ||
27 | if (vma && current->active_mm != vma->vm_mm) | ||
28 | return; | ||
29 | |||
30 | #ifndef CONFIG_CACHE_OFF | ||
31 | { | ||
32 | unsigned long pfn = pte_pfn(pte); | ||
33 | |||
34 | if (pfn_valid(pfn)) { | ||
35 | struct page *page = pfn_to_page(pfn); | ||
36 | |||
37 | if (!test_bit(PG_mapped, &page->flags)) { | ||
38 | unsigned long phys = pte_val(pte) & PTE_PHYS_MASK; | ||
39 | __flush_wback_region((void *)P1SEGADDR(phys), | ||
40 | PAGE_SIZE); | ||
41 | __set_bit(PG_mapped, &page->flags); | ||
42 | } | ||
43 | } | ||
44 | } | ||
45 | #endif | ||
46 | |||
47 | local_irq_save(flags); | ||
48 | |||
49 | /* Set PTEH register */ | ||
50 | vpn = address & MMU_VPN_MASK; | ||
51 | __raw_writel(vpn, MMU_PTEH); | ||
52 | |||
53 | /* Set PTEAEX */ | ||
54 | __raw_writel(get_asid(), MMU_PTEAEX); | ||
55 | |||
56 | pteval = pte.pte_low; | ||
57 | |||
58 | /* Set PTEA register */ | ||
59 | #ifdef CONFIG_X2TLB | ||
60 | /* | ||
61 | * For the extended mode TLB this is trivial, only the ESZ and | ||
62 | * EPR bits need to be written out to PTEA, with the remainder of | ||
63 | * the protection bits (with the exception of the compat-mode SZ | ||
64 | * and PR bits, which are cleared) being written out in PTEL. | ||
65 | */ | ||
66 | __raw_writel(pte.pte_high, MMU_PTEA); | ||
67 | #else | ||
68 | /* TODO: make this look less hacky */ | ||
69 | __raw_writel(((pteval >> 28) & 0xe) | (pteval & 0x1), MMU_PTEA); | ||
70 | #endif | ||
71 | |||
72 | /* Set PTEL register */ | ||
73 | pteval &= _PAGE_FLAGS_HARDWARE_MASK; /* drop software flags */ | ||
74 | #ifdef CONFIG_CACHE_WRITETHROUGH | ||
75 | pteval |= _PAGE_WT; | ||
76 | #endif | ||
77 | /* conveniently, we want all the software flags to be 0 anyway */ | ||
78 | __raw_writel(pteval, MMU_PTEL); | ||
79 | |||
80 | /* Load the TLB */ | ||
81 | asm volatile("ldtlb": /* no output */ : /* no input */ : "memory"); | ||
82 | local_irq_restore(flags); | ||
83 | } | ||
84 | |||
85 | /* | ||
86 | * While SH-X2 extended TLB mode splits out the memory-mapped I/UTLB | ||
87 | * data arrays, SH-X3 cores with PTEAEX split out the memory-mapped | ||
88 | * address arrays. In compat mode the second array is inaccessible, while | ||
89 | * in extended mode, the legacy 8-bit ASID field in address array 1 has | ||
90 | * undefined behaviour. | ||
91 | */ | ||
92 | void __uses_jump_to_uncached local_flush_tlb_one(unsigned long asid, | ||
93 | unsigned long page) | ||
94 | { | ||
95 | jump_to_uncached(); | ||
96 | __raw_writel(page, MMU_UTLB_ADDRESS_ARRAY | MMU_PAGE_ASSOC_BIT); | ||
97 | __raw_writel(asid, MMU_UTLB_ADDRESS_ARRAY2 | MMU_PAGE_ASSOC_BIT); | ||
98 | back_to_cached(); | ||
99 | } | ||