EMILE/second/main.c

244 lines
5.9 KiB
C
Raw Normal View History

2004-02-15 20:46:45 +00:00
/*
*
* (c) 2004 Laurent Vivier <LaurentVivier@wanadoo.fr>
*
*/
#include <stdio.h>
#include <malloc.h>
2004-02-21 01:43:51 +00:00
#include <string.h>
2004-02-15 20:46:45 +00:00
#include "lowmem.h"
#include "bank.h"
2004-02-15 20:46:45 +00:00
#include "memory.h"
#include "uncompress.h"
#include "bootinfo.h"
2004-02-21 01:43:51 +00:00
#include "console.h"
2004-02-26 22:51:37 +00:00
#include "arch.h"
#include "misc.h"
#include "glue.h"
#include "load.h"
2004-02-15 20:46:45 +00:00
2004-02-24 21:27:48 +00:00
typedef void (*entry_t) (unsigned long , unsigned long , unsigned long );
typedef void (*disable_cache_t) (void);
2004-02-24 21:27:48 +00:00
extern void enter_kernel030(unsigned long addr, unsigned long size, unsigned long dest);
2004-03-06 01:10:31 +00:00
extern char end_enter_kernel030;
extern void MMU030_disable_cache(void);
2004-03-06 01:10:31 +00:00
extern void enter_kernel040(unsigned long addr, unsigned long size, unsigned long dest);
extern char end_enter_kernel040;
extern void MMU040_disable_cache(void);
2004-02-15 20:46:45 +00:00
2004-02-24 21:27:48 +00:00
#define PAGE_SHIFT 12
#define PAGE_SIZE (1UL << PAGE_SHIFT)
#define PAGE_MASK (~(PAGE_SIZE-1))
2004-02-15 20:46:45 +00:00
#define BI_ALLOC_SIZE (4096L) // Allocate 4K for bootinfo
2004-02-24 21:27:48 +00:00
#define KERNEL_ALIGN (256L * 1024L) // Kernel alignment, on 256K boundary
2004-02-15 20:46:45 +00:00
struct first_level_info {
unsigned long kernel_image_offset;
unsigned long kernel_image_size;
unsigned long kernel_size;
unsigned long ramdisk_offset;
unsigned long ramdisk_size;
char command_line[CL_SIZE];
};
int start(struct first_level_info* info)
2004-02-15 20:46:45 +00:00
{
2004-02-26 22:51:37 +00:00
#ifdef TARGET_M68K
2004-02-15 20:46:45 +00:00
char * kernel;
unsigned long physImage;
2004-02-21 01:43:51 +00:00
entry_t entry;
disable_cache_t disable_cache;
2004-02-21 01:43:51 +00:00
int ret;
unsigned long start_mem;
unsigned long aligned_size;
unsigned long aligned_addr;
2004-03-06 01:10:31 +00:00
unsigned long enter_kernel;
unsigned long end_enter_kernel;
unsigned long kernel_image_start;
unsigned long ramdisk_start;
int uncompressed_size;
2004-02-26 22:51:37 +00:00
#endif
2004-02-15 20:46:45 +00:00
printf("Early Macintosh Image LoadEr\n");
printf("EMILE v"VERSION" (c) 2004 Laurent Vivier\n");
printf("This is free software, redistribute it under GPL\n");
2004-05-26 18:32:49 +00:00
arch_init();
init_memory_map();
#ifdef BANK_DUMP
bank_dump();
#endif
/* load kernel */
printf("vmlinux %s\n", info->command_line);
printf("Loading kernel...\n");
kernel_image_start = (unsigned long)load_image(
(unsigned long)info->kernel_image_offset,
info->kernel_image_size);
printf("Kernel image loaded at 0x%lx\n", kernel_image_start);
printf("Kernel image size is %ld Bytes\n", info->kernel_image_size);
2004-02-26 22:51:37 +00:00
#ifdef TARGET_PPC
if (arch_type == gestalt68k)
{
error("You're trying to boot a powerPC kernel on 680x0 Machine\n");
}
/* FIXME: add some stuff to start 3rd level (powerPC) */
2004-02-26 22:51:37 +00:00
while(1);
return 0;
#elif defined(TARGET_M68K)
if (arch_type == gestaltPowerPC)
{
error("You're trying to boot a m68k kernel on powerPC Machine\n");
}
/* where is mapped my boot function ? */
if (mmu_type == gestalt68040MMU)
{
enter_kernel = (unsigned long)enter_kernel040;
end_enter_kernel = (unsigned long)&end_enter_kernel040;
disable_cache = MMU040_disable_cache;
}
else
{
enter_kernel = (unsigned long)enter_kernel030;
end_enter_kernel = (unsigned long)&end_enter_kernel030;
disable_cache = MMU030_disable_cache;
}
2004-02-26 22:51:37 +00:00
printf("Available Memory: %ld kB\n", bank_mem_avail() / 1024);
2004-02-19 13:09:41 +00:00
if (info->kernel_image_size != 0)
2004-02-15 20:46:45 +00:00
{
if (info->kernel_size == 0)
info->kernel_size = info->kernel_image_size * 3;
2004-02-15 20:46:45 +00:00
/* add KERNEL_ALIGN if we have to align
* and BI_ALLOC_SIZE for bootinfo
*/
printf("Allocating %ld bytes for kernel\n", info->kernel_size);
kernel = (char*)malloc(info->kernel_size + 4 + BI_ALLOC_SIZE +
end_enter_kernel - enter_kernel);
2004-02-15 20:46:45 +00:00
if (kernel == 0)
{
printf("cannot allocate %ld bytes\n", info->kernel_size);
2004-02-26 22:51:37 +00:00
while(1);
2004-02-15 20:46:45 +00:00
}
/* align kernel address to a 4 byte word */
kernel = (unsigned char*)(((unsigned long)kernel + 3) & 0xFFFFFFFC);
uncompressed_size = uncompress(kernel, (char*)kernel_image_start);
printf("\n");
/* copy enter_kernel at end of kernel */
memcpy((char*)kernel_image_start + uncompressed_size,
(char*)enter_kernel, end_enter_kernel - enter_kernel);
end_enter_kernel = kernel_image_start + uncompressed_size +
(end_enter_kernel - enter_kernel);
enter_kernel = kernel_image_start + uncompressed_size;
2004-02-15 20:46:45 +00:00
}
else
{
2004-02-26 22:51:37 +00:00
error("Kernel is missing !!!!\n");
2004-02-15 20:46:45 +00:00
}
/* free kernel image */
free((void*)kernel_image_start);
/* load ramdisk if needed */
if (info->ramdisk_size != 0)
{
printf("Loading RAMDISK...\n");
ramdisk_start = (unsigned long)load_image(
(unsigned long)info->ramdisk_offset,
info->ramdisk_size);
printf("RAMDISK loaded at 0x%lx\n", ramdisk_start);
printf("RAMDISK size is %ld Bytes\n", info->ramdisk_size);
}
else
{
ramdisk_start = 0;
printf("no RAMDISK\n");
}
ret = logical2physical((unsigned long)kernel, &physImage);
/* initialize bootinfo structure */
bootinfo_init(info->command_line,
(char*)ramdisk_start, info->ramdisk_size);
2004-02-24 21:27:48 +00:00
/* compute final address of kernel */
aligned_size = boot_info.memory[0].addr & (KERNEL_ALIGN - 1);
if ( aligned_size > 0 )
{
aligned_size = KERNEL_ALIGN - aligned_size;
aligned_addr = boot_info.memory[0].addr + aligned_size;
aligned_size = boot_info.memory[0].size - aligned_size;
boot_info.memory[0].addr = aligned_addr;
boot_info.memory[0].size = aligned_size;
}
/* set bootinfo at end of kernel image */
2004-02-15 20:46:45 +00:00
set_kernel_bootinfo(kernel + uncompressed_size);
2004-02-23 00:20:11 +00:00
/* disable interrupt */
asm("ori.w #0x0700,%sr");
2004-03-06 01:10:31 +00:00
ret = logical2physical(enter_kernel, (unsigned long*)&entry);
2004-02-23 00:20:11 +00:00
2004-03-06 01:10:31 +00:00
if ( (ret == 0) && (enter_kernel != (unsigned long)entry) )
2004-02-21 01:43:51 +00:00
{
unsigned long logi;
2004-03-06 01:10:31 +00:00
unsigned long size = end_enter_kernel - enter_kernel;
2004-02-21 01:43:51 +00:00
logi = console_get_video();
ret = logical2physical(logi, (unsigned long*)&entry);
2004-02-23 00:20:11 +00:00
2004-03-06 01:10:31 +00:00
memcpy((char*)logi, (char*)enter_kernel, size);
memcpy((char*)entry, (char*)enter_kernel, size);
2004-02-21 01:43:51 +00:00
}
2004-02-24 21:27:48 +00:00
start_mem = boot_info.memory[0].addr + PAGE_SIZE;
printf("\n");
printf("Physical address of kernel will be 0x%08lx\n", start_mem);
printf("Ok, booting the kernel.\n");
/* disable and flush cache */
disable_cache();
/* kick off */
entry(physImage, uncompressed_size + BI_ALLOC_SIZE, start_mem);
2004-02-15 20:46:45 +00:00
return 0;
2004-02-26 22:51:37 +00:00
#endif /* TARGET_M68K */
2004-02-15 20:46:45 +00:00
}