mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-03 15:53:54 -06:00
Merge remote-tracking branch 'origin/master' into staging
* origin/master: (75 commits) tcg: Don't make exitreq flag a local temporary Makefile: Add subdir dependency on config-devices-all.mak make_device_config.sh: Emit dependency file to directory where included Revert "make_device_config.sh: Fix target path in generated dependency file" s390/virtio-ccw: remove redundant call to blockdev_mark_auto_del s390/css: Fix subchannel detection Allow virtio-net features for legacy s390 virtio bus s390: virtio-ccw maintainer s390: simplify kvm cpu init pseries: Add compatible property to root of device tree target-ppc: Move CPU aliases out of translate_init.c target-ppc: Report CPU aliases for QMP target-ppc: List alias names alongside CPU models target-ppc: Make host CPU a subclass of the host's CPU model PPC: xnu kernel expects FLUSH to be cleared on STOP PPC: Fix dma interrupt target-ppc: Fix PPC_DUMP_SPR_ACCESS build target-ppc: Synchronize FPU state with KVM target-ppc: Add mechanism for synchronizing SPRs with KVM Save memory allocation in the elf loader ...
This commit is contained in:
commit
fa3889162f
25 changed files with 4153 additions and 3836 deletions
75
hw/loader.c
75
hw/loader.c
|
@ -533,7 +533,14 @@ typedef struct Rom Rom;
|
|||
struct Rom {
|
||||
char *name;
|
||||
char *path;
|
||||
|
||||
/* datasize is the amount of memory allocated in "data". If datasize is less
|
||||
* than romsize, it means that the area from datasize to romsize is filled
|
||||
* with zeros.
|
||||
*/
|
||||
size_t romsize;
|
||||
size_t datasize;
|
||||
|
||||
uint8_t *data;
|
||||
int isrom;
|
||||
char *fw_dir;
|
||||
|
@ -589,14 +596,15 @@ int rom_add_file(const char *file, const char *fw_dir,
|
|||
rom->fw_dir = g_strdup(fw_dir);
|
||||
rom->fw_file = g_strdup(file);
|
||||
}
|
||||
rom->addr = addr;
|
||||
rom->romsize = lseek(fd, 0, SEEK_END);
|
||||
rom->data = g_malloc0(rom->romsize);
|
||||
rom->addr = addr;
|
||||
rom->romsize = lseek(fd, 0, SEEK_END);
|
||||
rom->datasize = rom->romsize;
|
||||
rom->data = g_malloc0(rom->datasize);
|
||||
lseek(fd, 0, SEEK_SET);
|
||||
rc = read(fd, rom->data, rom->romsize);
|
||||
if (rc != rom->romsize) {
|
||||
rc = read(fd, rom->data, rom->datasize);
|
||||
if (rc != rom->datasize) {
|
||||
fprintf(stderr, "rom: file %-20s: read error: rc=%d (expected %zd)\n",
|
||||
rom->name, rc, rom->romsize);
|
||||
rom->name, rc, rom->datasize);
|
||||
goto err;
|
||||
}
|
||||
close(fd);
|
||||
|
@ -637,16 +645,37 @@ int rom_add_blob(const char *name, const void *blob, size_t len,
|
|||
{
|
||||
Rom *rom;
|
||||
|
||||
rom = g_malloc0(sizeof(*rom));
|
||||
rom->name = g_strdup(name);
|
||||
rom->addr = addr;
|
||||
rom->romsize = len;
|
||||
rom->data = g_malloc0(rom->romsize);
|
||||
rom = g_malloc0(sizeof(*rom));
|
||||
rom->name = g_strdup(name);
|
||||
rom->addr = addr;
|
||||
rom->romsize = len;
|
||||
rom->datasize = len;
|
||||
rom->data = g_malloc0(rom->datasize);
|
||||
memcpy(rom->data, blob, len);
|
||||
rom_insert(rom);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* This function is specific for elf program because we don't need to allocate
|
||||
* all the rom. We just allocate the first part and the rest is just zeros. This
|
||||
* is why romsize and datasize are different. Also, this function seize the
|
||||
* memory ownership of "data", so we don't have to allocate and copy the buffer.
|
||||
*/
|
||||
int rom_add_elf_program(const char *name, void *data, size_t datasize,
|
||||
size_t romsize, hwaddr addr)
|
||||
{
|
||||
Rom *rom;
|
||||
|
||||
rom = g_malloc0(sizeof(*rom));
|
||||
rom->name = g_strdup(name);
|
||||
rom->addr = addr;
|
||||
rom->datasize = datasize;
|
||||
rom->romsize = romsize;
|
||||
rom->data = data;
|
||||
rom_insert(rom);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rom_add_vga(const char *file)
|
||||
{
|
||||
return rom_add_file(file, "vgaroms", 0, -1);
|
||||
|
@ -668,7 +697,7 @@ static void rom_reset(void *unused)
|
|||
if (rom->data == NULL) {
|
||||
continue;
|
||||
}
|
||||
cpu_physical_memory_write_rom(rom->addr, rom->data, rom->romsize);
|
||||
cpu_physical_memory_write_rom(rom->addr, rom->data, rom->datasize);
|
||||
if (rom->isrom) {
|
||||
/* rom needs to be written only once */
|
||||
g_free(rom->data);
|
||||
|
@ -756,13 +785,33 @@ int rom_copy(uint8_t *dest, hwaddr addr, size_t size)
|
|||
|
||||
d = dest + (rom->addr - addr);
|
||||
s = rom->data;
|
||||
l = rom->romsize;
|
||||
l = rom->datasize;
|
||||
|
||||
if ((d + l) > (dest + size)) {
|
||||
l = dest - d;
|
||||
}
|
||||
|
||||
memcpy(d, s, l);
|
||||
|
||||
if (rom->romsize > rom->datasize) {
|
||||
/* If datasize is less than romsize, it means that we didn't
|
||||
* allocate all the ROM because the trailing data are only zeros.
|
||||
*/
|
||||
|
||||
d += l;
|
||||
l = rom->romsize - rom->datasize;
|
||||
|
||||
if ((d + l) > (dest + size)) {
|
||||
/* Rom size doesn't fit in the destination area. Adjust to avoid
|
||||
* overflow.
|
||||
*/
|
||||
l = dest - d;
|
||||
}
|
||||
|
||||
if (l > 0) {
|
||||
memset(d, 0x0, l);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return (d + l) - dest;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue