diff options
author | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2009-02-10 21:37:44 -0500 |
---|---|---|
committer | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2009-02-10 21:37:44 -0500 |
commit | edbc29d76ddbc9bd56e1cbc772188f70c616ffe1 (patch) | |
tree | 71b994ba493c46e10189c090fd17fe2da80281ec /arch/powerpc/mm/fsl_booke_mmu.c | |
parent | 1db8508cf483dc1ecf66141f90a7c03659d69512 (diff) | |
parent | a0e8618c71b9b685977c1407dee928d86c5bdc2c (diff) |
Merge commit 'kumar/next' into next
Diffstat (limited to 'arch/powerpc/mm/fsl_booke_mmu.c')
-rw-r--r-- | arch/powerpc/mm/fsl_booke_mmu.c | 93 |
1 files changed, 46 insertions, 47 deletions
diff --git a/arch/powerpc/mm/fsl_booke_mmu.c b/arch/powerpc/mm/fsl_booke_mmu.c index ea6e41e39d9f..3d8cf01582f5 100644 --- a/arch/powerpc/mm/fsl_booke_mmu.c +++ b/arch/powerpc/mm/fsl_booke_mmu.c | |||
@@ -56,10 +56,14 @@ | |||
56 | 56 | ||
57 | extern void loadcam_entry(unsigned int index); | 57 | extern void loadcam_entry(unsigned int index); |
58 | unsigned int tlbcam_index; | 58 | unsigned int tlbcam_index; |
59 | static unsigned long __cam0, __cam1, __cam2; | 59 | static unsigned long cam[CONFIG_LOWMEM_CAM_NUM]; |
60 | 60 | ||
61 | #define NUM_TLBCAMS (16) | 61 | #define NUM_TLBCAMS (16) |
62 | 62 | ||
63 | #if defined(CONFIG_LOWMEM_CAM_NUM_BOOL) && (CONFIG_LOWMEM_CAM_NUM >= NUM_TLBCAMS) | ||
64 | #error "LOWMEM_CAM_NUM must be less than NUM_TLBCAMS" | ||
65 | #endif | ||
66 | |||
63 | struct tlbcam TLBCAM[NUM_TLBCAMS]; | 67 | struct tlbcam TLBCAM[NUM_TLBCAMS]; |
64 | 68 | ||
65 | struct tlbcamrange { | 69 | struct tlbcamrange { |
@@ -152,19 +156,19 @@ void invalidate_tlbcam_entry(int index) | |||
152 | loadcam_entry(index); | 156 | loadcam_entry(index); |
153 | } | 157 | } |
154 | 158 | ||
155 | void __init cam_mapin_ram(unsigned long cam0, unsigned long cam1, | 159 | unsigned long __init mmu_mapin_ram(void) |
156 | unsigned long cam2) | ||
157 | { | 160 | { |
158 | settlbcam(0, PAGE_OFFSET, memstart_addr, cam0, _PAGE_KERNEL, 0); | 161 | unsigned long virt = PAGE_OFFSET; |
159 | tlbcam_index++; | 162 | phys_addr_t phys = memstart_addr; |
160 | if (cam1) { | 163 | |
161 | tlbcam_index++; | 164 | while (cam[tlbcam_index] && tlbcam_index < ARRAY_SIZE(cam)) { |
162 | settlbcam(1, PAGE_OFFSET+cam0, memstart_addr+cam0, cam1, _PAGE_KERNEL, 0); | 165 | settlbcam(tlbcam_index, virt, phys, cam[tlbcam_index], _PAGE_KERNEL, 0); |
163 | } | 166 | virt += cam[tlbcam_index]; |
164 | if (cam2) { | 167 | phys += cam[tlbcam_index]; |
165 | tlbcam_index++; | 168 | tlbcam_index++; |
166 | settlbcam(2, PAGE_OFFSET+cam0+cam1, memstart_addr+cam0+cam1, cam2, _PAGE_KERNEL, 0); | ||
167 | } | 169 | } |
170 | |||
171 | return virt - PAGE_OFFSET; | ||
168 | } | 172 | } |
169 | 173 | ||
170 | /* | 174 | /* |
@@ -175,51 +179,46 @@ void __init MMU_init_hw(void) | |||
175 | flush_instruction_cache(); | 179 | flush_instruction_cache(); |
176 | } | 180 | } |
177 | 181 | ||
178 | unsigned long __init mmu_mapin_ram(void) | ||
179 | { | ||
180 | cam_mapin_ram(__cam0, __cam1, __cam2); | ||
181 | |||
182 | return __cam0 + __cam1 + __cam2; | ||
183 | } | ||
184 | |||
185 | |||
186 | void __init | 182 | void __init |
187 | adjust_total_lowmem(void) | 183 | adjust_total_lowmem(void) |
188 | { | 184 | { |
189 | phys_addr_t max_lowmem_size = __max_low_memory; | ||
190 | phys_addr_t cam_max_size = 0x10000000; | ||
191 | phys_addr_t ram; | 185 | phys_addr_t ram; |
186 | unsigned int max_cam = (mfspr(SPRN_TLB1CFG) >> 16) & 0xff; | ||
187 | char buf[ARRAY_SIZE(cam) * 5 + 1], *p = buf; | ||
188 | int i; | ||
189 | unsigned long virt = PAGE_OFFSET & 0xffffffffUL; | ||
190 | unsigned long phys = memstart_addr & 0xffffffffUL; | ||
192 | 191 | ||
193 | /* adjust CAM size to max_lowmem_size */ | 192 | /* Convert (4^max) kB to (2^max) bytes */ |
194 | if (max_lowmem_size < cam_max_size) | 193 | max_cam = max_cam * 2 + 10; |
195 | cam_max_size = max_lowmem_size; | ||
196 | 194 | ||
197 | /* adjust lowmem size to max_lowmem_size */ | 195 | /* adjust lowmem size to __max_low_memory */ |
198 | ram = min(max_lowmem_size, total_lowmem); | 196 | ram = min((phys_addr_t)__max_low_memory, (phys_addr_t)total_lowmem); |
199 | 197 | ||
200 | /* Calculate CAM values */ | 198 | /* Calculate CAM values */ |
201 | __cam0 = 1UL << 2 * (__ilog2(ram) / 2); | 199 | __max_low_memory = 0; |
202 | if (__cam0 > cam_max_size) | 200 | for (i = 0; ram && i < ARRAY_SIZE(cam); i++) { |
203 | __cam0 = cam_max_size; | 201 | unsigned int camsize = __ilog2(ram) & ~1U; |
204 | ram -= __cam0; | 202 | unsigned int align = __ffs(virt | phys) & ~1U; |
205 | if (ram) { | 203 | |
206 | __cam1 = 1UL << 2 * (__ilog2(ram) / 2); | 204 | if (camsize > align) |
207 | if (__cam1 > cam_max_size) | 205 | camsize = align; |
208 | __cam1 = cam_max_size; | 206 | if (camsize > max_cam) |
209 | ram -= __cam1; | 207 | camsize = max_cam; |
210 | } | 208 | |
211 | if (ram) { | 209 | cam[i] = 1UL << camsize; |
212 | __cam2 = 1UL << 2 * (__ilog2(ram) / 2); | 210 | ram -= cam[i]; |
213 | if (__cam2 > cam_max_size) | 211 | __max_low_memory += cam[i]; |
214 | __cam2 = cam_max_size; | 212 | virt += cam[i]; |
215 | ram -= __cam2; | 213 | phys += cam[i]; |
214 | |||
215 | p += sprintf(p, "%lu/", cam[i] >> 20); | ||
216 | } | 216 | } |
217 | for (; i < ARRAY_SIZE(cam); i++) | ||
218 | p += sprintf(p, "0/"); | ||
219 | p[-1] = '\0'; | ||
217 | 220 | ||
218 | printk(KERN_INFO "Memory CAM mapping: CAM0=%ldMb, CAM1=%ldMb," | 221 | pr_info("Memory CAM mapping: %s Mb, residual: %ldMb\n", buf, |
219 | " CAM2=%ldMb residual: %ldMb\n", | 222 | (total_lowmem - __max_low_memory) >> 20); |
220 | __cam0 >> 20, __cam1 >> 20, __cam2 >> 20, | ||
221 | (long int)((total_lowmem - __cam0 - __cam1 - __cam2) | ||
222 | >> 20)); | ||
223 | __max_low_memory = __cam0 + __cam1 + __cam2; | ||
224 | __initial_memory_limit_addr = memstart_addr + __max_low_memory; | 223 | __initial_memory_limit_addr = memstart_addr + __max_low_memory; |
225 | } | 224 | } |