Save memory allocation in the elf loader
The current elf loader uses too much memory. For example, I have a executable with a bss section of 400 MB and I set the ram size to 512 MB. Qemu uses about 780MB of RAM (which is fine), but there's a peak at 1.6 GB during initialization (this is not fine). This patch fixes two things: 1) do not allocate each elf program twice. 2) do not allocate memory for areas that are only zeros. For this we need a new field in Rom: "datasize" which is the size of the allocated data. If datasize is less than romsize, it means that the area from datasize to romsize is filled with zeros. Signed-off-by: Fabien Chouteau <chouteau@adacore.com> Signed-off-by: Alexander Graf <agraf@suse.de>
This commit is contained in:
		
							parent
							
								
									6bbd5dde9a
								
							
						
					
					
						commit
						d60fa42e8b
					
				
							
								
								
									
										19
									
								
								hw/elf_ops.h
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								hw/elf_ops.h
									
									
									
									
									
								
							@ -197,7 +197,7 @@ static int glue(load_elf, SZ)(const char *name, int fd,
 | 
				
			|||||||
    struct elfhdr ehdr;
 | 
					    struct elfhdr ehdr;
 | 
				
			||||||
    struct elf_phdr *phdr = NULL, *ph;
 | 
					    struct elf_phdr *phdr = NULL, *ph;
 | 
				
			||||||
    int size, i, total_size;
 | 
					    int size, i, total_size;
 | 
				
			||||||
    elf_word mem_size;
 | 
					    elf_word mem_size, file_size;
 | 
				
			||||||
    uint64_t addr, low = (uint64_t)-1, high = 0;
 | 
					    uint64_t addr, low = (uint64_t)-1, high = 0;
 | 
				
			||||||
    uint8_t *data = NULL;
 | 
					    uint8_t *data = NULL;
 | 
				
			||||||
    char label[128];
 | 
					    char label[128];
 | 
				
			||||||
@ -252,14 +252,16 @@ static int glue(load_elf, SZ)(const char *name, int fd,
 | 
				
			|||||||
    for(i = 0; i < ehdr.e_phnum; i++) {
 | 
					    for(i = 0; i < ehdr.e_phnum; i++) {
 | 
				
			||||||
        ph = &phdr[i];
 | 
					        ph = &phdr[i];
 | 
				
			||||||
        if (ph->p_type == PT_LOAD) {
 | 
					        if (ph->p_type == PT_LOAD) {
 | 
				
			||||||
            mem_size = ph->p_memsz;
 | 
					            mem_size = ph->p_memsz; /* Size of the ROM */
 | 
				
			||||||
            /* XXX: avoid allocating */
 | 
					            file_size = ph->p_filesz; /* Size of the allocated data */
 | 
				
			||||||
            data = g_malloc0(mem_size);
 | 
					            data = g_malloc0(file_size);
 | 
				
			||||||
            if (ph->p_filesz > 0) {
 | 
					            if (ph->p_filesz > 0) {
 | 
				
			||||||
                if (lseek(fd, ph->p_offset, SEEK_SET) < 0)
 | 
					                if (lseek(fd, ph->p_offset, SEEK_SET) < 0) {
 | 
				
			||||||
                    goto fail;
 | 
					                    goto fail;
 | 
				
			||||||
                if (read(fd, data, ph->p_filesz) != ph->p_filesz)
 | 
					                }
 | 
				
			||||||
 | 
					                if (read(fd, data, file_size) != file_size) {
 | 
				
			||||||
                    goto fail;
 | 
					                    goto fail;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            /* address_offset is hack for kernel images that are
 | 
					            /* address_offset is hack for kernel images that are
 | 
				
			||||||
               linked at the wrong physical address.  */
 | 
					               linked at the wrong physical address.  */
 | 
				
			||||||
@ -281,7 +283,9 @@ static int glue(load_elf, SZ)(const char *name, int fd,
 | 
				
			|||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            snprintf(label, sizeof(label), "phdr #%d: %s", i, name);
 | 
					            snprintf(label, sizeof(label), "phdr #%d: %s", i, name);
 | 
				
			||||||
            rom_add_blob_fixed(label, data, mem_size, addr);
 | 
					
 | 
				
			||||||
 | 
					            /* rom_add_elf_program() seize the ownership of 'data' */
 | 
				
			||||||
 | 
					            rom_add_elf_program(label, data, file_size, mem_size, addr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            total_size += mem_size;
 | 
					            total_size += mem_size;
 | 
				
			||||||
            if (addr < low)
 | 
					            if (addr < low)
 | 
				
			||||||
@ -289,7 +293,6 @@ static int glue(load_elf, SZ)(const char *name, int fd,
 | 
				
			|||||||
            if ((addr + mem_size) > high)
 | 
					            if ((addr + mem_size) > high)
 | 
				
			||||||
                high = addr + mem_size;
 | 
					                high = addr + mem_size;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            g_free(data);
 | 
					 | 
				
			||||||
            data = NULL;
 | 
					            data = NULL;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										75
									
								
								hw/loader.c
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								hw/loader.c
									
									
									
									
									
								
							@ -533,7 +533,14 @@ typedef struct Rom Rom;
 | 
				
			|||||||
struct Rom {
 | 
					struct Rom {
 | 
				
			||||||
    char *name;
 | 
					    char *name;
 | 
				
			||||||
    char *path;
 | 
					    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 romsize;
 | 
				
			||||||
 | 
					    size_t datasize;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    uint8_t *data;
 | 
					    uint8_t *data;
 | 
				
			||||||
    int isrom;
 | 
					    int isrom;
 | 
				
			||||||
    char *fw_dir;
 | 
					    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_dir  = g_strdup(fw_dir);
 | 
				
			||||||
        rom->fw_file = g_strdup(file);
 | 
					        rom->fw_file = g_strdup(file);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    rom->addr    = addr;
 | 
					    rom->addr     = addr;
 | 
				
			||||||
    rom->romsize = lseek(fd, 0, SEEK_END);
 | 
					    rom->romsize  = lseek(fd, 0, SEEK_END);
 | 
				
			||||||
    rom->data    = g_malloc0(rom->romsize);
 | 
					    rom->datasize = rom->romsize;
 | 
				
			||||||
 | 
					    rom->data     = g_malloc0(rom->datasize);
 | 
				
			||||||
    lseek(fd, 0, SEEK_SET);
 | 
					    lseek(fd, 0, SEEK_SET);
 | 
				
			||||||
    rc = read(fd, rom->data, rom->romsize);
 | 
					    rc = read(fd, rom->data, rom->datasize);
 | 
				
			||||||
    if (rc != rom->romsize) {
 | 
					    if (rc != rom->datasize) {
 | 
				
			||||||
        fprintf(stderr, "rom: file %-20s: read error: rc=%d (expected %zd)\n",
 | 
					        fprintf(stderr, "rom: file %-20s: read error: rc=%d (expected %zd)\n",
 | 
				
			||||||
                rom->name, rc, rom->romsize);
 | 
					                rom->name, rc, rom->datasize);
 | 
				
			||||||
        goto err;
 | 
					        goto err;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    close(fd);
 | 
					    close(fd);
 | 
				
			||||||
@ -637,16 +645,37 @@ int rom_add_blob(const char *name, const void *blob, size_t len,
 | 
				
			|||||||
{
 | 
					{
 | 
				
			||||||
    Rom *rom;
 | 
					    Rom *rom;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    rom = g_malloc0(sizeof(*rom));
 | 
					    rom           = g_malloc0(sizeof(*rom));
 | 
				
			||||||
    rom->name    = g_strdup(name);
 | 
					    rom->name     = g_strdup(name);
 | 
				
			||||||
    rom->addr    = addr;
 | 
					    rom->addr     = addr;
 | 
				
			||||||
    rom->romsize = len;
 | 
					    rom->romsize  = len;
 | 
				
			||||||
    rom->data    = g_malloc0(rom->romsize);
 | 
					    rom->datasize = len;
 | 
				
			||||||
 | 
					    rom->data     = g_malloc0(rom->datasize);
 | 
				
			||||||
    memcpy(rom->data, blob, len);
 | 
					    memcpy(rom->data, blob, len);
 | 
				
			||||||
    rom_insert(rom);
 | 
					    rom_insert(rom);
 | 
				
			||||||
    return 0;
 | 
					    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)
 | 
					int rom_add_vga(const char *file)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    return rom_add_file(file, "vgaroms", 0, -1);
 | 
					    return rom_add_file(file, "vgaroms", 0, -1);
 | 
				
			||||||
@ -668,7 +697,7 @@ static void rom_reset(void *unused)
 | 
				
			|||||||
        if (rom->data == NULL) {
 | 
					        if (rom->data == NULL) {
 | 
				
			||||||
            continue;
 | 
					            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) {
 | 
					        if (rom->isrom) {
 | 
				
			||||||
            /* rom needs to be written only once */
 | 
					            /* rom needs to be written only once */
 | 
				
			||||||
            g_free(rom->data);
 | 
					            g_free(rom->data);
 | 
				
			||||||
@ -756,13 +785,33 @@ int rom_copy(uint8_t *dest, hwaddr addr, size_t size)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        d = dest + (rom->addr - addr);
 | 
					        d = dest + (rom->addr - addr);
 | 
				
			||||||
        s = rom->data;
 | 
					        s = rom->data;
 | 
				
			||||||
        l = rom->romsize;
 | 
					        l = rom->datasize;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if ((d + l) > (dest + size)) {
 | 
					        if ((d + l) > (dest + size)) {
 | 
				
			||||||
            l = dest - d;
 | 
					            l = dest - d;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        memcpy(d, s, l);
 | 
					        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;
 | 
					    return (d + l) - dest;
 | 
				
			||||||
 | 
				
			|||||||
@ -27,6 +27,8 @@ int rom_add_file(const char *file, const char *fw_dir,
 | 
				
			|||||||
                 hwaddr addr, int32_t bootindex);
 | 
					                 hwaddr addr, int32_t bootindex);
 | 
				
			||||||
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,
 | 
				
			||||||
                 hwaddr addr);
 | 
					                 hwaddr addr);
 | 
				
			||||||
 | 
					int rom_add_elf_program(const char *name, void *data, size_t datasize,
 | 
				
			||||||
 | 
					                        size_t romsize, hwaddr addr);
 | 
				
			||||||
int rom_load_all(void);
 | 
					int rom_load_all(void);
 | 
				
			||||||
void rom_set_fw(void *f);
 | 
					void rom_set_fw(void *f);
 | 
				
			||||||
int rom_copy(uint8_t *dest, hwaddr addr, size_t size);
 | 
					int rom_copy(uint8_t *dest, hwaddr addr, size_t size);
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user