提交 3c178e72 编写于 作者: G Gerd Hoffmann 提交者: Anthony Liguori

rom loader: fix sparc -kernel boot.

Changes:
(1) register pstrcpy_targphys() in rom list, it is used for kernel
command lines by a number of architectures.
(2) add rom_ptr() function to get a pointer for applying changes
to loaded images.  Needed for example to tell the linux kernel
where it finds the initrd image by updating the header.
(3) make sparc use rom_ptr for initrd setup.

booting sparc-test works now, and 'info roms' shows this:

(qemu) info roms
addr=0000000000000000 size=0x2a3828 mem=ram name="phdr #0: vmlinux-2.6.11+tcx"
addr=00000000007ff000 size=0x00000e mem=ram name="cmdline"
addr=0000000000800000 size=0x400000 mem=ram name="/root/qemu-test/sparc-test/linux.img"
addr=0000000070000000 size=0x0e4000 mem=rom name="phdr #0: /home/kraxel/projects/qemu/build-zfull/pc-bios/openbios-sparc32"

reboot via 'system_reset' works too.

Patchworks-ID: 35262
Signed-off-by: NGerd Hoffmann <kraxel@redhat.com>
Signed-off-by: NAnthony Liguori <aliguori@us.ibm.com>
上级 d549db5a
...@@ -365,7 +365,7 @@ void axisdev88_init (ram_addr_t ram_size, ...@@ -365,7 +365,7 @@ void axisdev88_init (ram_addr_t ram_size,
/* Let the kernel know we are modifying the cmdline. */ /* Let the kernel know we are modifying the cmdline. */
env->regs[10] = 0x87109563; env->regs[10] = 0x87109563;
env->regs[11] = 0x40000000; env->regs[11] = 0x40000000;
pstrcpy_targphys(env->regs[11], 256, kernel_cmdline); pstrcpy_targphys("cmdline", env->regs[11], 256, kernel_cmdline);
} }
} }
env->pc = bootstrap_pc; env->pc = bootstrap_pc;
......
...@@ -157,7 +157,7 @@ void bareetraxfs_init (ram_addr_t ram_size, ...@@ -157,7 +157,7 @@ void bareetraxfs_init (ram_addr_t ram_size,
/* Let the kernel know we are modifying the cmdline. */ /* Let the kernel know we are modifying the cmdline. */
env->regs[10] = 0x87109563; env->regs[10] = 0x87109563;
env->regs[11] = 0x40000000; env->regs[11] = 0x40000000;
pstrcpy_targphys(env->regs[11], 256, kernel_cmdline); pstrcpy_targphys("cmdline", env->regs[11], 256, kernel_cmdline);
} }
} }
env->pc = bootstrap_pc; env->pc = bootstrap_pc;
......
...@@ -108,20 +108,20 @@ int load_image_targphys(const char *filename, ...@@ -108,20 +108,20 @@ int load_image_targphys(const char *filename,
return size; return size;
} }
void pstrcpy_targphys(target_phys_addr_t dest, int buf_size, void pstrcpy_targphys(const char *name, target_phys_addr_t dest, int buf_size,
const char *source) const char *source)
{ {
static const uint8_t nul_byte = 0;
const char *nulp; const char *nulp;
char *ptr;
if (buf_size <= 0) return; if (buf_size <= 0) return;
nulp = memchr(source, 0, buf_size); nulp = memchr(source, 0, buf_size);
if (nulp) { if (nulp) {
cpu_physical_memory_write_rom(dest, (uint8_t *)source, rom_add_blob_fixed(name, source, (nulp - source) + 1, dest);
(nulp - source) + 1);
} else { } else {
cpu_physical_memory_write_rom(dest, (uint8_t *)source, buf_size - 1); rom_add_blob_fixed(name, source, buf_size, dest);
cpu_physical_memory_write_rom(dest, &nul_byte, 1); ptr = rom_ptr(dest + buf_size - 1);
*ptr = 0;
} }
} }
...@@ -672,6 +672,32 @@ int rom_load_all(void) ...@@ -672,6 +672,32 @@ int rom_load_all(void)
return 0; return 0;
} }
static Rom *find_rom(target_phys_addr_t addr)
{
Rom *rom;
QTAILQ_FOREACH(rom, &roms, next) {
if (rom->max)
continue;
if (rom->min > addr)
continue;
if (rom->min + rom->romsize < addr)
continue;
return rom;
}
return NULL;
}
void *rom_ptr(target_phys_addr_t addr)
{
Rom *rom;
rom = find_rom(addr);
if (!rom || !rom->data)
return NULL;
return rom->data + (addr - rom->min);
}
void do_info_roms(Monitor *mon) void do_info_roms(Monitor *mon)
{ {
Rom *rom; Rom *rom;
......
...@@ -15,7 +15,8 @@ int load_uimage(const char *filename, target_phys_addr_t *ep, ...@@ -15,7 +15,8 @@ int load_uimage(const char *filename, target_phys_addr_t *ep,
int read_targphys(const char *name, int read_targphys(const char *name,
int fd, target_phys_addr_t dst_addr, size_t nbytes); int fd, target_phys_addr_t dst_addr, size_t nbytes);
void pstrcpy_targphys(target_phys_addr_t dest, int buf_size, void pstrcpy_targphys(const char *name,
target_phys_addr_t dest, int buf_size,
const char *source); const char *source);
int rom_add_file(const char *file, int rom_add_file(const char *file,
...@@ -23,6 +24,7 @@ int rom_add_file(const char *file, ...@@ -23,6 +24,7 @@ int rom_add_file(const char *file,
int rom_add_blob(const char *name, const void *blob, size_t len, int rom_add_blob(const char *name, const void *blob, size_t len,
target_phys_addr_t min, target_phys_addr_t max, int align); target_phys_addr_t min, target_phys_addr_t max, int align);
int rom_load_all(void); int rom_load_all(void);
void *rom_ptr(target_phys_addr_t addr);
void do_info_roms(Monitor *mon); void do_info_roms(Monitor *mon);
#define rom_add_file_fixed(_f, _a) \ #define rom_add_file_fixed(_f, _a) \
......
...@@ -679,7 +679,7 @@ static void prom_set(int index, const char *string, ...) ...@@ -679,7 +679,7 @@ static void prom_set(int index, const char *string, ...)
va_start(ap, string); va_start(ap, string);
vsnprintf(buf, ENVP_ENTRY_SIZE, string, ap); vsnprintf(buf, ENVP_ENTRY_SIZE, string, ap);
va_end(ap); va_end(ap);
pstrcpy_targphys(table_addr + VIRT_TO_PHYS_ADDEND, ENVP_ENTRY_SIZE, buf); pstrcpy_targphys("prom", table_addr + VIRT_TO_PHYS_ADDEND, ENVP_ENTRY_SIZE, buf);
} }
/* Kernel */ /* Kernel */
......
...@@ -130,7 +130,7 @@ static void load_kernel (CPUState *env) ...@@ -130,7 +130,7 @@ static void load_kernel (CPUState *env)
} else { } else {
ret = 0; ret = 0;
} }
pstrcpy_targphys((16 << 20) - 256 + ret, 256, pstrcpy_targphys("cmdline", (16 << 20) - 256 + ret, 256,
loaderparams.kernel_cmdline); loaderparams.kernel_cmdline);
stl_phys((16 << 20) - 260, 0x12345678); stl_phys((16 << 20) - 260, 0x12345678);
......
...@@ -865,7 +865,7 @@ static void load_linux(void *fw_cfg, ...@@ -865,7 +865,7 @@ static void load_linux(void *fw_cfg,
initrd_max = max_ram_size-ACPI_DATA_SIZE-1; initrd_max = max_ram_size-ACPI_DATA_SIZE-1;
/* kernel command line */ /* kernel command line */
rom_add_blob_fixed("linux-cmdline", kernel_cmdline, rom_add_blob_fixed("cmdline", kernel_cmdline,
strlen(kernel_cmdline)+1, cmdline_addr); strlen(kernel_cmdline)+1, cmdline_addr);
if (protocol >= 0x202) { if (protocol >= 0x202) {
......
...@@ -176,7 +176,7 @@ petalogix_s3adsp1800_init(ram_addr_t ram_size, ...@@ -176,7 +176,7 @@ petalogix_s3adsp1800_init(ram_addr_t ram_size,
env->regs[5] = ddr_base + kernel_size; env->regs[5] = ddr_base + kernel_size;
if (kernel_cmdline && (kcmdline_len = strlen(kernel_cmdline))) { if (kernel_cmdline && (kcmdline_len = strlen(kernel_cmdline))) {
pstrcpy_targphys(env->regs[5], 256, kernel_cmdline); pstrcpy_targphys("cmdline", env->regs[5], 256, kernel_cmdline);
} }
env->regs[6] = 0; env->regs[6] = 0;
/* Provide a device-tree. */ /* Provide a device-tree. */
......
...@@ -1256,7 +1256,7 @@ int PPC_NVRAM_set_params (nvram_t *nvram, uint16_t NVRAM_size, ...@@ -1256,7 +1256,7 @@ int PPC_NVRAM_set_params (nvram_t *nvram, uint16_t NVRAM_size,
NVRAM_set_lword(nvram, 0x3C, kernel_size); NVRAM_set_lword(nvram, 0x3C, kernel_size);
if (cmdline) { if (cmdline) {
/* XXX: put the cmdline in NVRAM too ? */ /* XXX: put the cmdline in NVRAM too ? */
pstrcpy_targphys(CMDLINE_ADDR, RAM_size - CMDLINE_ADDR, cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, RAM_size - CMDLINE_ADDR, cmdline);
NVRAM_set_lword(nvram, 0x40, CMDLINE_ADDR); NVRAM_set_lword(nvram, 0x40, CMDLINE_ADDR);
NVRAM_set_lword(nvram, 0x44, strlen(cmdline)); NVRAM_set_lword(nvram, 0x44, strlen(cmdline));
} else { } else {
......
...@@ -364,7 +364,7 @@ static void ppc_core99_init (ram_addr_t ram_size, ...@@ -364,7 +364,7 @@ static void ppc_core99_init (ram_addr_t ram_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
......
...@@ -384,7 +384,7 @@ static void ppc_heathrow_init (ram_addr_t ram_size, ...@@ -384,7 +384,7 @@ static void ppc_heathrow_init (ram_addr_t ram_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
......
...@@ -250,7 +250,7 @@ static void r2d_init(ram_addr_t ram_size, ...@@ -250,7 +250,7 @@ static void r2d_init(ram_addr_t ram_size,
SDRAM_BASE + LINUX_LOAD_OFFSET, SDRAM_BASE + LINUX_LOAD_OFFSET,
SDRAM_SIZE - LINUX_LOAD_OFFSET); SDRAM_SIZE - LINUX_LOAD_OFFSET);
env->pc = (SDRAM_BASE + LINUX_LOAD_OFFSET) | 0xa0000000; env->pc = (SDRAM_BASE + LINUX_LOAD_OFFSET) | 0xa0000000;
pstrcpy_targphys(SDRAM_BASE + 0x10100, 256, kernel_cmdline); pstrcpy_targphys("cmdline", SDRAM_BASE + 0x10100, 256, kernel_cmdline);
} else { } else {
kernel_size = load_image_targphys(kernel_filename, SDRAM_BASE, SDRAM_SIZE); kernel_size = load_image_targphys(kernel_filename, SDRAM_BASE, SDRAM_SIZE);
env->pc = SDRAM_BASE | 0xa0000000; /* Start from P2 area */ env->pc = SDRAM_BASE | 0xa0000000; /* Start from P2 area */
......
...@@ -299,6 +299,7 @@ static unsigned long sun4m_load_kernel(const char *kernel_filename, ...@@ -299,6 +299,7 @@ static unsigned long sun4m_load_kernel(const char *kernel_filename,
int linux_boot; int linux_boot;
unsigned int i; unsigned int i;
long initrd_size, kernel_size; long initrd_size, kernel_size;
uint8_t *ptr;
linux_boot = (kernel_filename != NULL); linux_boot = (kernel_filename != NULL);
...@@ -341,9 +342,10 @@ static unsigned long sun4m_load_kernel(const char *kernel_filename, ...@@ -341,9 +342,10 @@ static unsigned long sun4m_load_kernel(const char *kernel_filename,
} }
if (initrd_size > 0) { if (initrd_size > 0) {
for (i = 0; i < 64 * TARGET_PAGE_SIZE; i += TARGET_PAGE_SIZE) { for (i = 0; i < 64 * TARGET_PAGE_SIZE; i += TARGET_PAGE_SIZE) {
if (ldl_phys(KERNEL_LOAD_ADDR + i) == 0x48647253) { // HdrS ptr = rom_ptr(KERNEL_LOAD_ADDR + i);
stl_phys(KERNEL_LOAD_ADDR + i + 16, INITRD_LOAD_ADDR); if (ldl_p(ptr) == 0x48647253) { // HdrS
stl_phys(KERNEL_LOAD_ADDR + i + 20, initrd_size); stl_p(ptr + 16, INITRD_LOAD_ADDR);
stl_p(ptr + 20, initrd_size);
break; break;
} }
} }
...@@ -875,7 +877,7 @@ static void sun4m_hw_init(const struct sun4m_hwdef *hwdef, ram_addr_t RAM_size, ...@@ -875,7 +877,7 @@ static void sun4m_hw_init(const struct sun4m_hwdef *hwdef, ram_addr_t RAM_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
...@@ -1457,7 +1459,7 @@ static void sun4d_hw_init(const struct sun4d_hwdef *hwdef, ram_addr_t RAM_size, ...@@ -1457,7 +1459,7 @@ static void sun4d_hw_init(const struct sun4d_hwdef *hwdef, ram_addr_t RAM_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
...@@ -1645,7 +1647,7 @@ static void sun4c_hw_init(const struct sun4c_hwdef *hwdef, ram_addr_t RAM_size, ...@@ -1645,7 +1647,7 @@ static void sun4c_hw_init(const struct sun4c_hwdef *hwdef, ram_addr_t RAM_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
......
...@@ -651,7 +651,7 @@ static void sun4uv_init(ram_addr_t RAM_size, ...@@ -651,7 +651,7 @@ static void sun4uv_init(ram_addr_t RAM_size,
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size);
if (kernel_cmdline) { if (kernel_cmdline) {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, CMDLINE_ADDR);
pstrcpy_targphys(CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline); pstrcpy_targphys("cmdline", CMDLINE_ADDR, TARGET_PAGE_SIZE, kernel_cmdline);
} else { } else {
fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0); fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_CMDLINE, 0);
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册