diff options
Diffstat (limited to 'arch/sparc64/mm/tsb.c')
-rw-r--r-- | arch/sparc64/mm/tsb.c | 109 |
1 files changed, 95 insertions, 14 deletions
diff --git a/arch/sparc64/mm/tsb.c b/arch/sparc64/mm/tsb.c index 2f84cef6c1b5..dfe7144fcdf6 100644 --- a/arch/sparc64/mm/tsb.c +++ b/arch/sparc64/mm/tsb.c | |||
@@ -9,13 +9,7 @@ | |||
9 | #include <asm/tlbflush.h> | 9 | #include <asm/tlbflush.h> |
10 | #include <asm/tlb.h> | 10 | #include <asm/tlb.h> |
11 | #include <asm/mmu_context.h> | 11 | #include <asm/mmu_context.h> |
12 | 12 | #include <asm/pgtable.h> | |
13 | #define TSB_ENTRY_ALIGNMENT 16 | ||
14 | |||
15 | struct tsb { | ||
16 | unsigned long tag; | ||
17 | unsigned long pte; | ||
18 | } __attribute__((aligned(TSB_ENTRY_ALIGNMENT))); | ||
19 | 13 | ||
20 | /* We use an 8K TSB for the whole kernel, this allows to | 14 | /* We use an 8K TSB for the whole kernel, this allows to |
21 | * handle about 4MB of modules and vmalloc mappings without | 15 | * handle about 4MB of modules and vmalloc mappings without |
@@ -27,10 +21,10 @@ struct tsb { | |||
27 | 21 | ||
28 | extern struct tsb swapper_tsb[KERNEL_TSB_NENTRIES]; | 22 | extern struct tsb swapper_tsb[KERNEL_TSB_NENTRIES]; |
29 | 23 | ||
30 | static inline unsigned long tsb_hash(unsigned long vaddr) | 24 | static inline unsigned long tsb_hash(unsigned long vaddr, unsigned long nentries) |
31 | { | 25 | { |
32 | vaddr >>= PAGE_SHIFT; | 26 | vaddr >>= PAGE_SHIFT; |
33 | return vaddr & (KERNEL_TSB_NENTRIES - 1); | 27 | return vaddr & (nentries - 1); |
34 | } | 28 | } |
35 | 29 | ||
36 | static inline int tag_compare(struct tsb *entry, unsigned long vaddr, unsigned long context) | 30 | static inline int tag_compare(struct tsb *entry, unsigned long vaddr, unsigned long context) |
@@ -51,7 +45,8 @@ void flush_tsb_kernel_range(unsigned long start, unsigned long end) | |||
51 | unsigned long v; | 45 | unsigned long v; |
52 | 46 | ||
53 | for (v = start; v < end; v += PAGE_SIZE) { | 47 | for (v = start; v < end; v += PAGE_SIZE) { |
54 | struct tsb *ent = &swapper_tsb[tsb_hash(v)]; | 48 | unsigned long hash = tsb_hash(v, KERNEL_TSB_NENTRIES); |
49 | struct tsb *ent = &swapper_tsb[hash]; | ||
55 | 50 | ||
56 | if (tag_compare(ent, v, 0)) { | 51 | if (tag_compare(ent, v, 0)) { |
57 | ent->tag = 0UL; | 52 | ent->tag = 0UL; |
@@ -63,8 +58,9 @@ void flush_tsb_kernel_range(unsigned long start, unsigned long end) | |||
63 | void flush_tsb_user(struct mmu_gather *mp) | 58 | void flush_tsb_user(struct mmu_gather *mp) |
64 | { | 59 | { |
65 | struct mm_struct *mm = mp->mm; | 60 | struct mm_struct *mm = mp->mm; |
66 | struct tsb *tsb = (struct tsb *) mm->context.sparc64_tsb; | 61 | struct tsb *tsb = mm->context.tsb; |
67 | unsigned long ctx = ~0UL; | 62 | unsigned long ctx = ~0UL; |
63 | unsigned long nentries = mm->context.tsb_nentries; | ||
68 | int i; | 64 | int i; |
69 | 65 | ||
70 | if (CTX_VALID(mm->context)) | 66 | if (CTX_VALID(mm->context)) |
@@ -76,7 +72,7 @@ void flush_tsb_user(struct mmu_gather *mp) | |||
76 | 72 | ||
77 | v &= ~0x1UL; | 73 | v &= ~0x1UL; |
78 | 74 | ||
79 | ent = &tsb[tsb_hash(v)]; | 75 | ent = &tsb[tsb_hash(v, nentries)]; |
80 | if (tag_compare(ent, v, ctx)) { | 76 | if (tag_compare(ent, v, ctx)) { |
81 | ent->tag = 0UL; | 77 | ent->tag = 0UL; |
82 | membar_storeload_storestore(); | 78 | membar_storeload_storestore(); |
@@ -84,6 +80,83 @@ void flush_tsb_user(struct mmu_gather *mp) | |||
84 | } | 80 | } |
85 | } | 81 | } |
86 | 82 | ||
83 | static void setup_tsb_params(struct mm_struct *mm, unsigned long tsb_bytes) | ||
84 | { | ||
85 | unsigned long tsb_reg, base, tsb_paddr; | ||
86 | unsigned long page_sz, tte; | ||
87 | |||
88 | mm->context.tsb_nentries = tsb_bytes / sizeof(struct tsb); | ||
89 | |||
90 | base = TSBMAP_BASE; | ||
91 | tte = (_PAGE_VALID | _PAGE_L | _PAGE_CP | | ||
92 | _PAGE_CV | _PAGE_P | _PAGE_W); | ||
93 | tsb_paddr = __pa(mm->context.tsb); | ||
94 | |||
95 | /* Use the smallest page size that can map the whole TSB | ||
96 | * in one TLB entry. | ||
97 | */ | ||
98 | switch (tsb_bytes) { | ||
99 | case 8192 << 0: | ||
100 | tsb_reg = 0x0UL; | ||
101 | #ifdef DCACHE_ALIASING_POSSIBLE | ||
102 | base += (tsb_paddr & 8192); | ||
103 | #endif | ||
104 | tte |= _PAGE_SZ8K; | ||
105 | page_sz = 8192; | ||
106 | break; | ||
107 | |||
108 | case 8192 << 1: | ||
109 | tsb_reg = 0x1UL; | ||
110 | tte |= _PAGE_SZ64K; | ||
111 | page_sz = 64 * 1024; | ||
112 | break; | ||
113 | |||
114 | case 8192 << 2: | ||
115 | tsb_reg = 0x2UL; | ||
116 | tte |= _PAGE_SZ64K; | ||
117 | page_sz = 64 * 1024; | ||
118 | break; | ||
119 | |||
120 | case 8192 << 3: | ||
121 | tsb_reg = 0x3UL; | ||
122 | tte |= _PAGE_SZ64K; | ||
123 | page_sz = 64 * 1024; | ||
124 | break; | ||
125 | |||
126 | case 8192 << 4: | ||
127 | tsb_reg = 0x4UL; | ||
128 | tte |= _PAGE_SZ512K; | ||
129 | page_sz = 512 * 1024; | ||
130 | break; | ||
131 | |||
132 | case 8192 << 5: | ||
133 | tsb_reg = 0x5UL; | ||
134 | tte |= _PAGE_SZ512K; | ||
135 | page_sz = 512 * 1024; | ||
136 | break; | ||
137 | |||
138 | case 8192 << 6: | ||
139 | tsb_reg = 0x6UL; | ||
140 | tte |= _PAGE_SZ512K; | ||
141 | page_sz = 512 * 1024; | ||
142 | break; | ||
143 | |||
144 | case 8192 << 7: | ||
145 | tsb_reg = 0x7UL; | ||
146 | tte |= _PAGE_SZ4MB; | ||
147 | page_sz = 4 * 1024 * 1024; | ||
148 | break; | ||
149 | }; | ||
150 | |||
151 | tsb_reg |= base; | ||
152 | tsb_reg |= (tsb_paddr & (page_sz - 1UL)); | ||
153 | tte |= (tsb_paddr & ~(page_sz - 1UL)); | ||
154 | |||
155 | mm->context.tsb_reg_val = tsb_reg; | ||
156 | mm->context.tsb_map_vaddr = base; | ||
157 | mm->context.tsb_map_pte = tte; | ||
158 | } | ||
159 | |||
87 | int init_new_context(struct task_struct *tsk, struct mm_struct *mm) | 160 | int init_new_context(struct task_struct *tsk, struct mm_struct *mm) |
88 | { | 161 | { |
89 | unsigned long page = get_zeroed_page(GFP_KERNEL); | 162 | unsigned long page = get_zeroed_page(GFP_KERNEL); |
@@ -92,14 +165,22 @@ int init_new_context(struct task_struct *tsk, struct mm_struct *mm) | |||
92 | if (unlikely(!page)) | 165 | if (unlikely(!page)) |
93 | return -ENOMEM; | 166 | return -ENOMEM; |
94 | 167 | ||
95 | mm->context.sparc64_tsb = (unsigned long *) page; | 168 | mm->context.tsb = (struct tsb *) page; |
169 | setup_tsb_params(mm, PAGE_SIZE); | ||
96 | 170 | ||
97 | return 0; | 171 | return 0; |
98 | } | 172 | } |
99 | 173 | ||
100 | void destroy_context(struct mm_struct *mm) | 174 | void destroy_context(struct mm_struct *mm) |
101 | { | 175 | { |
102 | free_page((unsigned long) mm->context.sparc64_tsb); | 176 | free_page((unsigned long) mm->context.tsb); |
177 | |||
178 | /* We can remove these later, but for now it's useful | ||
179 | * to catch any bogus post-destroy_context() references | ||
180 | * to the TSB. | ||
181 | */ | ||
182 | mm->context.tsb = NULL; | ||
183 | mm->context.tsb_reg_val = 0UL; | ||
103 | 184 | ||
104 | spin_lock(&ctx_alloc_lock); | 185 | spin_lock(&ctx_alloc_lock); |
105 | 186 | ||