diff options
Diffstat (limited to 'arch/sh/kernel')
-rw-r--r-- | arch/sh/kernel/setup.c | 78 |
1 files changed, 58 insertions, 20 deletions
diff --git a/arch/sh/kernel/setup.c b/arch/sh/kernel/setup.c index 14735d5ede52..9c7f7811af70 100644 --- a/arch/sh/kernel/setup.c +++ b/arch/sh/kernel/setup.c | |||
@@ -188,6 +188,63 @@ static inline void __init reserve_crashkernel(void) | |||
188 | {} | 188 | {} |
189 | #endif | 189 | #endif |
190 | 190 | ||
191 | static void __init check_for_initrd(void) | ||
192 | { | ||
193 | #ifdef CONFIG_BLK_DEV_INITRD | ||
194 | unsigned long start, end; | ||
195 | |||
196 | /* | ||
197 | * Check for the rare cases where boot loaders adhere to the boot | ||
198 | * ABI. | ||
199 | */ | ||
200 | if (!LOADER_TYPE || !INITRD_START || !INITRD_SIZE) | ||
201 | goto disable; | ||
202 | |||
203 | start = INITRD_START + __MEMORY_START; | ||
204 | end = start + INITRD_SIZE; | ||
205 | |||
206 | if (unlikely(end <= start)) | ||
207 | goto disable; | ||
208 | if (unlikely(start & ~PAGE_MASK)) { | ||
209 | pr_err("initrd must be page aligned\n"); | ||
210 | goto disable; | ||
211 | } | ||
212 | |||
213 | if (unlikely(start < PAGE_OFFSET)) { | ||
214 | pr_err("initrd start < PAGE_OFFSET\n"); | ||
215 | goto disable; | ||
216 | } | ||
217 | |||
218 | if (unlikely(end > lmb_end_of_DRAM())) { | ||
219 | pr_err("initrd extends beyond end of memory " | ||
220 | "(0x%08lx > 0x%08lx)\ndisabling initrd\n", | ||
221 | end, (unsigned long)lmb_end_of_DRAM()); | ||
222 | goto disable; | ||
223 | } | ||
224 | |||
225 | /* | ||
226 | * If we got this far inspite of the boot loader's best efforts | ||
227 | * to the contrary, assume we actually have a valid initrd and | ||
228 | * fix up the root dev. | ||
229 | */ | ||
230 | ROOT_DEV = Root_RAM0; | ||
231 | |||
232 | /* | ||
233 | * Address sanitization | ||
234 | */ | ||
235 | initrd_start = (unsigned long)__va(__pa(start)); | ||
236 | initrd_end = initrd_start + INITRD_SIZE; | ||
237 | |||
238 | reserve_bootmem(__pa(initrd_start), INITRD_SIZE, BOOTMEM_DEFAULT); | ||
239 | |||
240 | return; | ||
241 | |||
242 | disable: | ||
243 | pr_info("initrd disabled\n"); | ||
244 | initrd_start = initrd_end = 0; | ||
245 | #endif | ||
246 | } | ||
247 | |||
191 | void __cpuinit calibrate_delay(void) | 248 | void __cpuinit calibrate_delay(void) |
192 | { | 249 | { |
193 | struct clk *clk = clk_get(NULL, "cpu_clk"); | 250 | struct clk *clk = clk_get(NULL, "cpu_clk"); |
@@ -277,26 +334,7 @@ void __init setup_bootmem_allocator(unsigned long free_pfn) | |||
277 | 334 | ||
278 | sparse_memory_present_with_active_regions(0); | 335 | sparse_memory_present_with_active_regions(0); |
279 | 336 | ||
280 | #ifdef CONFIG_BLK_DEV_INITRD | 337 | check_for_initrd(); |
281 | ROOT_DEV = Root_RAM0; | ||
282 | |||
283 | if (LOADER_TYPE && INITRD_START) { | ||
284 | unsigned long initrd_start_phys = INITRD_START + __MEMORY_START; | ||
285 | |||
286 | if (initrd_start_phys + INITRD_SIZE <= PFN_PHYS(max_low_pfn)) { | ||
287 | reserve_bootmem(initrd_start_phys, INITRD_SIZE, | ||
288 | BOOTMEM_DEFAULT); | ||
289 | initrd_start = (unsigned long)__va(initrd_start_phys); | ||
290 | initrd_end = initrd_start + INITRD_SIZE; | ||
291 | } else { | ||
292 | printk("initrd extends beyond end of memory " | ||
293 | "(0x%08lx > 0x%08lx)\ndisabling initrd\n", | ||
294 | initrd_start_phys + INITRD_SIZE, | ||
295 | (unsigned long)PFN_PHYS(max_low_pfn)); | ||
296 | initrd_start = 0; | ||
297 | } | ||
298 | } | ||
299 | #endif | ||
300 | 338 | ||
301 | reserve_crashkernel(); | 339 | reserve_crashkernel(); |
302 | } | 340 | } |