qemu-riscv
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: [PATCH] hw/riscv: Respect firmware ELF entry point


From: Alistair Francis
Subject: Re: [PATCH] hw/riscv: Respect firmware ELF entry point
Date: Mon, 9 Sep 2024 12:04:07 +1000

On Sat, Aug 17, 2024 at 10:56 AM Samuel Holland
<samuel.holland@sifive.com> wrote:
>
> When riscv_load_firmware() loads an ELF, the ELF segment addresses are
> used, not the passed-in firmware_load_addr. The machine models assume
> the firmware entry point is what they provided for firmware_load_addr,
> and use that address to generate the boot ROM, so if the ELF is linked
> at any other address, the boot ROM will jump to empty memory.
>
> Pass back the ELF entry point to use when generating the boot ROM, so
> the boot ROM can jump to firmware loaded anywhere in RAM. For example,
> on the virt machine, this allows using an OpenSBI fw_dynamic.elf built
> with FW_TEXT_START values other than 0x80000000.
>
> Signed-off-by: Samuel Holland <samuel.holland@sifive.com>

Reviewed-by: Alistair Francis <alistair.francis@wdc.com>

Alistair

> ---
>
>  hw/riscv/boot.c            | 11 ++++++-----
>  hw/riscv/microchip_pfsoc.c |  2 +-
>  hw/riscv/opentitan.c       |  3 ++-
>  hw/riscv/shakti_c.c        | 13 ++++++-------
>  hw/riscv/sifive_u.c        |  4 ++--
>  hw/riscv/spike.c           |  5 +++--
>  hw/riscv/virt.c            |  4 ++--
>  include/hw/riscv/boot.h    |  4 ++--
>  8 files changed, 24 insertions(+), 22 deletions(-)
>
> diff --git a/hw/riscv/boot.c b/hw/riscv/boot.c
> index 47281ca853..9115ecd91f 100644
> --- a/hw/riscv/boot.c
> +++ b/hw/riscv/boot.c
> @@ -128,11 +128,11 @@ char *riscv_find_firmware(const char *firmware_filename,
>
>  target_ulong riscv_find_and_load_firmware(MachineState *machine,
>                                            const char 
> *default_machine_firmware,
> -                                          hwaddr firmware_load_addr,
> +                                          hwaddr *firmware_load_addr,
>                                            symbol_fn_t sym_cb)
>  {
>      char *firmware_filename;
> -    target_ulong firmware_end_addr = firmware_load_addr;
> +    target_ulong firmware_end_addr = *firmware_load_addr;
>
>      firmware_filename = riscv_find_firmware(machine->firmware,
>                                              default_machine_firmware);
> @@ -148,7 +148,7 @@ target_ulong riscv_find_and_load_firmware(MachineState 
> *machine,
>  }
>
>  target_ulong riscv_load_firmware(const char *firmware_filename,
> -                                 hwaddr firmware_load_addr,
> +                                 hwaddr *firmware_load_addr,
>                                   symbol_fn_t sym_cb)
>  {
>      uint64_t firmware_entry, firmware_end;
> @@ -159,15 +159,16 @@ target_ulong riscv_load_firmware(const char 
> *firmware_filename,
>      if (load_elf_ram_sym(firmware_filename, NULL, NULL, NULL,
>                           &firmware_entry, NULL, &firmware_end, NULL,
>                           0, EM_RISCV, 1, 0, NULL, true, sym_cb) > 0) {
> +        *firmware_load_addr = firmware_entry;
>          return firmware_end;
>      }
>
>      firmware_size = load_image_targphys_as(firmware_filename,
> -                                           firmware_load_addr,
> +                                           *firmware_load_addr,
>                                             current_machine->ram_size, NULL);
>
>      if (firmware_size > 0) {
> -        return firmware_load_addr + firmware_size;
> +        return *firmware_load_addr + firmware_size;
>      }
>
>      error_report("could not load firmware '%s'", firmware_filename);
> diff --git a/hw/riscv/microchip_pfsoc.c b/hw/riscv/microchip_pfsoc.c
> index 7725dfbde5..f9a3b43d2e 100644
> --- a/hw/riscv/microchip_pfsoc.c
> +++ b/hw/riscv/microchip_pfsoc.c
> @@ -613,7 +613,7 @@ static void 
> microchip_icicle_kit_machine_init(MachineState *machine)
>
>      /* Load the firmware */
>      firmware_end_addr = riscv_find_and_load_firmware(machine, firmware_name,
> -                                                     firmware_load_addr, 
> NULL);
> +                                                     &firmware_load_addr, 
> NULL);
>
>      if (kernel_as_payload) {
>          kernel_start_addr = riscv_calc_kernel_start_addr(&s->soc.u_cpus,
> diff --git a/hw/riscv/opentitan.c b/hw/riscv/opentitan.c
> index 436503f1ba..e2830e9dc2 100644
> --- a/hw/riscv/opentitan.c
> +++ b/hw/riscv/opentitan.c
> @@ -98,7 +98,8 @@ static void opentitan_machine_init(MachineState *machine)
>          memmap[IBEX_DEV_RAM].base, machine->ram);
>
>      if (machine->firmware) {
> -        riscv_load_firmware(machine->firmware, memmap[IBEX_DEV_RAM].base, 
> NULL);
> +        hwaddr firmware_load_addr = memmap[IBEX_DEV_RAM].base;
> +        riscv_load_firmware(machine->firmware, &firmware_load_addr, NULL);
>      }
>
>      if (machine->kernel_filename) {
> diff --git a/hw/riscv/shakti_c.c b/hw/riscv/shakti_c.c
> index 3888034c2b..2dccc1eff2 100644
> --- a/hw/riscv/shakti_c.c
> +++ b/hw/riscv/shakti_c.c
> @@ -45,6 +45,7 @@ static void shakti_c_machine_state_init(MachineState 
> *mstate)
>  {
>      ShaktiCMachineState *sms = RISCV_SHAKTI_MACHINE(mstate);
>      MemoryRegion *system_memory = get_system_memory();
> +    hwaddr firmware_load_addr = shakti_c_memmap[SHAKTI_C_RAM].base;
>
>      /* Initialize SoC */
>      object_initialize_child(OBJECT(mstate), "soc", &sms->soc,
> @@ -56,16 +57,14 @@ static void shakti_c_machine_state_init(MachineState 
> *mstate)
>                                  shakti_c_memmap[SHAKTI_C_RAM].base,
>                                  mstate->ram);
>
> +    if (mstate->firmware) {
> +        riscv_load_firmware(mstate->firmware, &firmware_load_addr, NULL);
> +    }
> +
>      /* ROM reset vector */
> -    riscv_setup_rom_reset_vec(mstate, &sms->soc.cpus,
> -                              shakti_c_memmap[SHAKTI_C_RAM].base,
> +    riscv_setup_rom_reset_vec(mstate, &sms->soc.cpus, firmware_load_addr,
>                                shakti_c_memmap[SHAKTI_C_ROM].base,
>                                shakti_c_memmap[SHAKTI_C_ROM].size, 0, 0);
> -    if (mstate->firmware) {
> -        riscv_load_firmware(mstate->firmware,
> -                            shakti_c_memmap[SHAKTI_C_RAM].base,
> -                            NULL);
> -    }
>  }
>
>  static void shakti_c_machine_instance_init(Object *obj)
> diff --git a/hw/riscv/sifive_u.c b/hw/riscv/sifive_u.c
> index af5f923f54..35a689309d 100644
> --- a/hw/riscv/sifive_u.c
> +++ b/hw/riscv/sifive_u.c
> @@ -515,7 +515,7 @@ static void sifive_u_machine_init(MachineState *machine)
>      SiFiveUState *s = RISCV_U_MACHINE(machine);
>      MemoryRegion *system_memory = get_system_memory();
>      MemoryRegion *flash0 = g_new(MemoryRegion, 1);
> -    target_ulong start_addr = memmap[SIFIVE_U_DEV_DRAM].base;
> +    hwaddr start_addr = memmap[SIFIVE_U_DEV_DRAM].base;
>      target_ulong firmware_end_addr, kernel_start_addr;
>      const char *firmware_name;
>      uint32_t start_addr_hi32 = 0x00000000;
> @@ -589,7 +589,7 @@ static void sifive_u_machine_init(MachineState *machine)
>
>      firmware_name = riscv_default_firmware_name(&s->soc.u_cpus);
>      firmware_end_addr = riscv_find_and_load_firmware(machine, firmware_name,
> -                                                     start_addr, NULL);
> +                                                     &start_addr, NULL);
>
>      if (machine->kernel_filename) {
>          kernel_start_addr = riscv_calc_kernel_start_addr(&s->soc.u_cpus,
> diff --git a/hw/riscv/spike.c b/hw/riscv/spike.c
> index 64074395bc..fceb91d946 100644
> --- a/hw/riscv/spike.c
> +++ b/hw/riscv/spike.c
> @@ -198,6 +198,7 @@ static void spike_board_init(MachineState *machine)
>      MemoryRegion *system_memory = get_system_memory();
>      MemoryRegion *mask_rom = g_new(MemoryRegion, 1);
>      target_ulong firmware_end_addr = memmap[SPIKE_DRAM].base;
> +    hwaddr firmware_load_addr = memmap[SPIKE_DRAM].base;
>      target_ulong kernel_start_addr;
>      char *firmware_name;
>      uint32_t fdt_load_addr;
> @@ -290,7 +291,7 @@ static void spike_board_init(MachineState *machine)
>      /* Load firmware */
>      if (firmware_name) {
>          firmware_end_addr = riscv_load_firmware(firmware_name,
> -                                                memmap[SPIKE_DRAM].base,
> +                                                &firmware_load_addr,
>                                                  htif_symbol_callback);
>          g_free(firmware_name);
>      }
> @@ -320,7 +321,7 @@ static void spike_board_init(MachineState *machine)
>      riscv_load_fdt(fdt_load_addr, machine->fdt);
>
>      /* load the reset vector */
> -    riscv_setup_rom_reset_vec(machine, &s->soc[0], memmap[SPIKE_DRAM].base,
> +    riscv_setup_rom_reset_vec(machine, &s->soc[0], firmware_load_addr,
>                                memmap[SPIKE_MROM].base,
>                                memmap[SPIKE_MROM].size, kernel_entry,
>                                fdt_load_addr);
> diff --git a/hw/riscv/virt.c b/hw/riscv/virt.c
> index 9981e0f6c9..aef5e284a7 100644
> --- a/hw/riscv/virt.c
> +++ b/hw/riscv/virt.c
> @@ -1336,7 +1336,7 @@ static void virt_machine_done(Notifier *notifier, void 
> *data)
>                                       machine_done);
>      const MemMapEntry *memmap = virt_memmap;
>      MachineState *machine = MACHINE(s);
> -    target_ulong start_addr = memmap[VIRT_DRAM].base;
> +    hwaddr start_addr = memmap[VIRT_DRAM].base;
>      target_ulong firmware_end_addr, kernel_start_addr;
>      const char *firmware_name = riscv_default_firmware_name(&s->soc[0]);
>      uint64_t fdt_load_addr;
> @@ -1368,7 +1368,7 @@ static void virt_machine_done(Notifier *notifier, void 
> *data)
>      }
>
>      firmware_end_addr = riscv_find_and_load_firmware(machine, firmware_name,
> -                                                     start_addr, NULL);
> +                                                     &start_addr, NULL);
>
>      pflash_blk0 = pflash_cfi01_get_blk(s->flash[0]);
>      if (pflash_blk0) {
> diff --git a/include/hw/riscv/boot.h b/include/hw/riscv/boot.h
> index a2e4ae9cb0..18bfe9f7bf 100644
> --- a/include/hw/riscv/boot.h
> +++ b/include/hw/riscv/boot.h
> @@ -35,13 +35,13 @@ target_ulong 
> riscv_calc_kernel_start_addr(RISCVHartArrayState *harts,
>                                            target_ulong firmware_end_addr);
>  target_ulong riscv_find_and_load_firmware(MachineState *machine,
>                                            const char 
> *default_machine_firmware,
> -                                          hwaddr firmware_load_addr,
> +                                          hwaddr *firmware_load_addr,
>                                            symbol_fn_t sym_cb);
>  const char *riscv_default_firmware_name(RISCVHartArrayState *harts);
>  char *riscv_find_firmware(const char *firmware_filename,
>                            const char *default_machine_firmware);
>  target_ulong riscv_load_firmware(const char *firmware_filename,
> -                                 hwaddr firmware_load_addr,
> +                                 hwaddr *firmware_load_addr,
>                                   symbol_fn_t sym_cb);
>  target_ulong riscv_load_kernel(MachineState *machine,
>                                 RISCVHartArrayState *harts,
> --
> 2.45.1
>
>



reply via email to

[Prev in Thread] Current Thread [Next in Thread]