⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 flatload.c

📁 xen虚拟机源代码安装包
💻 C
📖 第 1 页 / 共 2 页
字号:
/****************************************************************************//* *  QEMU bFLT binary loader.  Based on linux/fs/binfmt_flat.c * *  This program is free software; you can redistribute it and/or modify *  it under the terms of the GNU General Public License as published by *  the Free Software Foundation; either version 2 of the License, or *  (at your option) any later version. * *  This program is distributed in the hope that it will be useful, *  but WITHOUT ANY WARRANTY; without even the implied warranty of *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the *  GNU General Public License for more details. * *  You should have received a copy of the GNU General Public License *  along with this program; if not, write to the Free Software *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *      Copyright (C) 2006 CodeSourcery. *	Copyright (C) 2000-2003 David McCullough <davidm@snapgear.com> *	Copyright (C) 2002 Greg Ungerer <gerg@snapgear.com> *	Copyright (C) 2002 SnapGear, by Paul Dale <pauli@snapgear.com> *	Copyright (C) 2000, 2001 Lineo, by David McCullough <davidm@lineo.com> *  based heavily on: * *  linux/fs/binfmt_aout.c: *      Copyright (C) 1991, 1992, 1996  Linus Torvalds *  linux/fs/binfmt_flat.c for 2.0 kernel *	    Copyright (C) 1998  Kenneth Albanowski <kjahds@kjahds.com> *	JAN/99 -- coded full program relocation (gerg@snapgear.com) *//* ??? ZFLAT and shared library support is currently disabled.  *//****************************************************************************/#include <stdio.h>#include <stdlib.h>#include <errno.h>#include <sys/mman.h>#include <unistd.h>#include "qemu.h"#include "flat.h"//#define DEBUG#ifdef DEBUG#define	DBG_FLT(a...)	printf(a)#else#define	DBG_FLT(a...)#endif#define flat_reloc_valid(reloc, size)             ((reloc) <= (size))#define flat_old_ram_flag(flag)                   (flag)#ifdef TARGET_WORDS_BIGENDIAN#define flat_get_relocate_addr(relval)            (relval)#else#define flat_get_relocate_addr(relval)            bswap32(relval)#endif#define RELOC_FAILED 0xff00ff01		/* Relocation incorrect somewhere */#define UNLOADED_LIB 0x7ff000ff		/* Placeholder for unused library */struct lib_info {    abi_ulong start_code;       /* Start of text segment */    abi_ulong start_data;       /* Start of data segment */    abi_ulong end_data;         /* Start of bss section */    abi_ulong start_brk;        /* End of data segment */    abi_ulong text_len;	        /* Length of text segment */    abi_ulong entry;	        /* Start address for this module */    abi_ulong build_date;       /* When this one was compiled */    short loaded;		/* Has this library been loaded? */};#ifdef CONFIG_BINFMT_SHARED_FLATstatic int load_flat_shared_library(int id, struct lib_info *p);#endifstruct linux_binprm;#define ntohl(x) be32_to_cpu(x)/****************************************************************************//* * create_flat_tables() parses the env- and arg-strings in new user * memory and creates the pointer tables from them, and puts their * addresses on the "stack", returning the new stack pointer value. *//* Push a block of strings onto the guest stack.  */static abi_ulong copy_strings(abi_ulong p, int n, char **s){    int len;    while (n-- > 0) {        len = strlen(s[n]) + 1;        p -= len;        memcpy_to_target(p, s[n], len);    }    return p;}int target_pread(int fd, abi_ulong ptr, abi_ulong len,                 abi_ulong offset){    void *buf;    int ret;    buf = lock_user(VERIFY_WRITE, ptr, len, 0);    ret = pread(fd, buf, len, offset);    unlock_user(buf, ptr, len);    return ret;}/****************************************************************************/#ifdef CONFIG_BINFMT_ZFLAT#include <linux/zlib.h>#define LBUFSIZE	4000/* gzip flag byte */#define ASCII_FLAG   0x01 /* bit 0 set: file probably ASCII text */#define CONTINUATION 0x02 /* bit 1 set: continuation of multi-part gzip file */#define EXTRA_FIELD  0x04 /* bit 2 set: extra field present */#define ORIG_NAME    0x08 /* bit 3 set: original file name present */#define COMMENT      0x10 /* bit 4 set: file comment present */#define ENCRYPTED    0x20 /* bit 5 set: file is encrypted */#define RESERVED     0xC0 /* bit 6,7:   reserved */static int decompress_exec(	struct linux_binprm *bprm,	unsigned long offset,	char *dst,	long len,	int fd){	unsigned char *buf;	z_stream strm;	loff_t fpos;	int ret, retval;	DBG_FLT("decompress_exec(offset=%x,buf=%x,len=%x)\n",(int)offset, (int)dst, (int)len);	memset(&strm, 0, sizeof(strm));	strm.workspace = kmalloc(zlib_inflate_workspacesize(), GFP_KERNEL);	if (strm.workspace == NULL) {		DBG_FLT("binfmt_flat: no memory for decompress workspace\n");		return -ENOMEM;	}	buf = kmalloc(LBUFSIZE, GFP_KERNEL);	if (buf == NULL) {		DBG_FLT("binfmt_flat: no memory for read buffer\n");		retval = -ENOMEM;		goto out_free;	}	/* Read in first chunk of data and parse gzip header. */	fpos = offset;	ret = bprm->file->f_op->read(bprm->file, buf, LBUFSIZE, &fpos);	strm.next_in = buf;	strm.avail_in = ret;	strm.total_in = 0;	retval = -ENOEXEC;	/* Check minimum size -- gzip header */	if (ret < 10) {		DBG_FLT("binfmt_flat: file too small?\n");		goto out_free_buf;	}	/* Check gzip magic number */	if ((buf[0] != 037) || ((buf[1] != 0213) && (buf[1] != 0236))) {		DBG_FLT("binfmt_flat: unknown compression magic?\n");		goto out_free_buf;	}	/* Check gzip method */	if (buf[2] != 8) {		DBG_FLT("binfmt_flat: unknown compression method?\n");		goto out_free_buf;	}	/* Check gzip flags */	if ((buf[3] & ENCRYPTED) || (buf[3] & CONTINUATION) ||	    (buf[3] & RESERVED)) {		DBG_FLT("binfmt_flat: unknown flags?\n");		goto out_free_buf;	}	ret = 10;	if (buf[3] & EXTRA_FIELD) {		ret += 2 + buf[10] + (buf[11] << 8);		if (unlikely(LBUFSIZE == ret)) {			DBG_FLT("binfmt_flat: buffer overflow (EXTRA)?\n");			goto out_free_buf;		}	}	if (buf[3] & ORIG_NAME) {		for (; ret < LBUFSIZE && (buf[ret] != 0); ret++)			;		if (unlikely(LBUFSIZE == ret)) {			DBG_FLT("binfmt_flat: buffer overflow (ORIG_NAME)?\n");			goto out_free_buf;		}	}	if (buf[3] & COMMENT) {		for (;  ret < LBUFSIZE && (buf[ret] != 0); ret++)			;		if (unlikely(LBUFSIZE == ret)) {			DBG_FLT("binfmt_flat: buffer overflow (COMMENT)?\n");			goto out_free_buf;		}	}	strm.next_in += ret;	strm.avail_in -= ret;	strm.next_out = dst;	strm.avail_out = len;	strm.total_out = 0;	if (zlib_inflateInit2(&strm, -MAX_WBITS) != Z_OK) {		DBG_FLT("binfmt_flat: zlib init failed?\n");		goto out_free_buf;	}	while ((ret = zlib_inflate(&strm, Z_NO_FLUSH)) == Z_OK) {		ret = bprm->file->f_op->read(bprm->file, buf, LBUFSIZE, &fpos);		if (ret <= 0)			break;		if (ret >= (unsigned long) -4096)			break;		len -= ret;		strm.next_in = buf;		strm.avail_in = ret;		strm.total_in = 0;	}	if (ret < 0) {		DBG_FLT("binfmt_flat: decompression failed (%d), %s\n",			ret, strm.msg);		goto out_zlib;	}	retval = 0;out_zlib:	zlib_inflateEnd(&strm);out_free_buf:	kfree(buf);out_free:	kfree(strm.workspace);out:	return retval;}#endif /* CONFIG_BINFMT_ZFLAT *//****************************************************************************/static abi_ulongcalc_reloc(abi_ulong r, struct lib_info *p, int curid, int internalp){    abi_ulong addr;    int id;    abi_ulong start_brk;    abi_ulong start_data;    abi_ulong text_len;    abi_ulong start_code;#ifdef CONFIG_BINFMT_SHARED_FLAT#error needs checking    if (r == 0)        id = curid;	/* Relocs of 0 are always self referring */    else {        id = (r >> 24) & 0xff;	/* Find ID for this reloc */        r &= 0x00ffffff;	/* Trim ID off here */    }    if (id >= MAX_SHARED_LIBS) {        fprintf(stderr, "BINFMT_FLAT: reference 0x%x to shared library %d\n",                (unsigned) r, id);        goto failed;    }    if (curid != id) {        if (internalp) {            fprintf(stderr, "BINFMT_FLAT: reloc address 0x%x not "                    "in same module (%d != %d)\n",                    (unsigned) r, curid, id);            goto failed;        } else if ( ! p[id].loaded &&                    load_flat_shared_library(id, p) > (unsigned long) -4096) {            fprintf(stderr, "BINFMT_FLAT: failed to load library %d\n", id);            goto failed;        }        /* Check versioning information (i.e. time stamps) */        if (p[id].build_date && p[curid].build_date            && p[curid].build_date < p[id].build_date) {            fprintf(stderr, "BINFMT_FLAT: library %d is younger than %d\n",                    id, curid);            goto failed;        }    }#else    id = 0;#endif    start_brk = p[id].start_brk;    start_data = p[id].start_data;    start_code = p[id].start_code;    text_len = p[id].text_len;    if (!flat_reloc_valid(r, start_brk - start_data + text_len)) {        fprintf(stderr, "BINFMT_FLAT: reloc outside program 0x%x "                "(0 - 0x%x/0x%x)\n",               (int) r,(int)(start_brk-start_code),(int)text_len);        goto failed;    }    if (r < text_len)			/* In text segment */        addr = r + start_code;    else					/* In data segment */        addr = r - text_len + start_data;    /* Range checked already above so doing the range tests is redundant...*/    return(addr);failed:    abort();    return RELOC_FAILED;}/****************************************************************************//* ??? This does not handle endianness correctly.  */void old_reloc(struct lib_info *libinfo, uint32_t rl){#ifdef DEBUG	char *segment[] = { "TEXT", "DATA", "BSS", "*UNKNOWN*" };#endif	uint32_t *ptr;        uint32_t offset;        int reloc_type;        offset = rl & 0x3fffffff;        reloc_type = rl >> 30;        /* ??? How to handle this?  */#if defined(CONFIG_COLDFIRE)	ptr = (uint32_t *) (libinfo->start_code + offset);#else	ptr = (uint32_t *) (libinfo->start_data + offset);#endif#ifdef DEBUG	fprintf(stderr, "Relocation of variable at DATASEG+%x "		"(address %p, currently %x) into segment %s\n",		offset, ptr, (int)*ptr, segment[reloc_type]);#endif	switch (reloc_type) {	case OLD_FLAT_RELOC_TYPE_TEXT:		*ptr += libinfo->start_code;		break;	case OLD_FLAT_RELOC_TYPE_DATA:		*ptr += libinfo->start_data;		break;	case OLD_FLAT_RELOC_TYPE_BSS:		*ptr += libinfo->end_data;		break;	default:		fprintf(stderr, "BINFMT_FLAT: Unknown relocation type=%x\n",                        reloc_type);		break;	}	DBG_FLT("Relocation became %x\n", (int)*ptr);}/****************************************************************************/static int load_flat_file(struct linux_binprm * bprm,		struct lib_info *libinfo, int id, abi_ulong *extra_stack){    struct flat_hdr * hdr;    abi_ulong textpos = 0, datapos = 0, result;    abi_ulong realdatastart = 0;    abi_ulong text_len, data_len, bss_len, stack_len, flags;    abi_ulong memp = 0; /* for finding the brk area */    abi_ulong extra;    abi_ulong reloc = 0, rp;    int i, rev, relocs = 0;    abi_ulong fpos;    abi_ulong start_code, end_code;    abi_ulong indx_len;    hdr = ((struct flat_hdr *) bprm->buf);		/* exec-header */    text_len  = ntohl(hdr->data_start);    data_len  = ntohl(hdr->data_end) - ntohl(hdr->data_start);    bss_len   = ntohl(hdr->bss_end) - ntohl(hdr->data_end);    stack_len = ntohl(hdr->stack_size);    if (extra_stack) {        stack_len += *extra_stack;        *extra_stack = stack_len;    }    relocs    = ntohl(hdr->reloc_count);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -