/*      $OpenBSD: elf2olf.c,v 1.7 2003/06/10 22:20:46 deraadt Exp $	*/
/*
 * Copyright (c) 1996 Erik Theisen.  All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. The name of the author may not be used to endorse or promote products
 *    derived from this software without specific prior written permission
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
#ifndef lint
static char copyright[] = 
"@(#) Copyright (c) 1996 Erik Theisen.  All rights reserved.\n";
#endif /* not lint */

#ifndef lint
static char rcsid[] = "@(#) $Id: elf2olf.c,v 1.7 2003/06/10 22:20:46 deraadt Exp $";
#endif

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
#include <unistd.h>

#include <sys/types.h>
#include <sys/uio.h>
#include <sys/stat.h>
#include <sys/param.h>

#include <olf_abi.h>

int   retval  = 0;
int   olf2elf;
char *progname;
int   verbose;
int   opsys = OS_ID;

char *os_namev[] = ONAMEV;

/* Handle endianess */
#define word(x,y)((y == ELFDATA2LSB) ? ntohl(htonl(x)) : ntohl(x))
#define half(x,y)((y == ELFDATA2LSB) ? ntohs(htons(x)) : ntohs(x))

void usage(void);
void pwarn(char *, char *, int);

int
main(int argc, char*argv[])
{
    extern char *optarg;
    extern int optind;
    int ch, i, okay;
    char *opstring;

    int fd;
    struct stat st;
    Elf32_Ehdr ehdr;
    Elf32_Shdr shdr;
    int e;
    
    if ((progname = strrchr(*argv, '/')))
	++progname;
    else
	progname = *argv;

    if (strstr(progname, "olf2elf"))
	olf2elf = 1;
    
    /* 
     * Process cmdline
     */
    opstring = olf2elf ? "v" : "vo:";
    while((ch = getopt(argc, argv, opstring)) != -1)
	switch(ch) {
	case 'v':
	    verbose = 1;
	    break;
	case 'o':
	    for (i = 1; i <= OOS_NUM; i++) {
	        if (os_namev[i] == NULL) {
		    fprintf(stderr, 
			    "%s: illegal -o argument -- %s\n", 
			    progname, optarg);
		    usage();
		}
		else if (strcmp(optarg, os_namev[i]) == 0) {
		    opsys = i;
		    break;
		}
	    }
	    break;
	default:
	    usage();
	}
    argc -= optind;
    argv += optind;

    if (argc == 0)
	usage();

    /*
     * Process file(s)
     */
    do {
	okay = 0;

	if ((fd = open(*argv, O_RDWR | O_EXLOCK, 0)) > 0 &&
	    lseek(fd, (off_t)0, SEEK_SET) == 0 &&
	    fstat(fd, &st) == 0) {

	    /* Make sure this is a 32bit ELF or OLF version 1 file */
	    if (read(fd, &ehdr, sizeof(Elf32_Ehdr)) == sizeof(Elf32_Ehdr)&&
		(IS_ELF(ehdr) || IS_OLF(ehdr)) &&
		ehdr.e_ident[EI_CLASS] == ELFCLASS32 &&
		ehdr.e_ident[EI_VERSION] == 1) {

		/* Is this elf2olf? */
		if(!olf2elf) {

		    /* Tag, your it... */
		    ehdr.e_ident[OI_MAG0]    = OLFMAG0;
		    ehdr.e_ident[OI_MAG1]    = OLFMAG1;
		    ehdr.e_ident[OI_MAG2]    = OLFMAG2;
		    ehdr.e_ident[OI_MAG3]    = OLFMAG3;
		    ehdr.e_ident[OI_OS]      = opsys;
		    ehdr.e_ident[OI_DYNAMIC] = ODYNAMIC_N;
		    ehdr.e_ident[OI_STRIP]   = OSTRIP;

		    /* We'll need this endian */
		    e = ehdr.e_ident[EI_DATA];

		    /* Now we need to figure out wether or */
		    /* not we're really stripped. */
		    if (lseek(fd, (off_t)word(ehdr.e_shoff, e),
			      SEEK_SET) == word(ehdr.e_shoff, e)) {

			/* 
			 * search through section header table
			 * looking for a section header of type
			 * SHT_SYMTAB and SHT_DYNAMIC. If there is 
			 * one present we're NOT stripped and/or 
			 * dynamic.
			 */
			for (i = 0; i < half(ehdr.e_shnum, e); i++) {
			    if (read(fd, &shdr, sizeof(Elf32_Shdr)) == sizeof(Elf32_Shdr)){
				if (word(shdr.sh_type, e) == SHT_SYMTAB)
				    ehdr.e_ident[OI_STRIP] = OSTRIP_N;
				else if (word(shdr.sh_type, e) == SHT_DYNAMIC)
				    ehdr.e_ident[OI_DYNAMIC] = ODYNAMIC;
			    } else
				pwarn(progname, *argv, errno);
			} /* while less than number of section headers */

			/* We're ready to modify */
			okay = 1;

		    } else /* Bogus section header table seek */
			pwarn(progname, *argv, errno);

		} else { /* olf2elf */
		    ehdr.e_ident[EI_MAG0]    = ELFMAG0;
		    ehdr.e_ident[EI_MAG1]    = ELFMAG1;
		    ehdr.e_ident[EI_MAG2]    = ELFMAG2;
		    ehdr.e_ident[EI_MAG3]    = ELFMAG3;
		    ehdr.e_ident[OI_OS]      = 0;
		    ehdr.e_ident[OI_DYNAMIC] = 0;
		    ehdr.e_ident[OI_STRIP]   = 0;

		    okay = 1;
		} /* olf2elf */
	    } else /* Bogus non-ELF file encountered */
		pwarn(progname, *argv, ENOEXEC);

	    /*
	     * Do It.
	     */
	    if (okay) {
		if (lseek(fd, (off_t)0, SEEK_SET) == 0) {
		    if (write(fd, &ehdr, sizeof(Elf32_Ehdr)) == sizeof(Elf32_Ehdr)) {
			if (verbose) {
			    if (!olf2elf) {
				printf("ELF %s => OLF %d-bit %s %s linked %s OLF.\n",
				       *argv,
				       (ehdr.e_ident[OI_CLASS] == OLFCLASS32)?\
				           32 : 64,
				       os_namev[ehdr.e_ident[OI_OS]],
				       ehdr.e_ident[OI_DYNAMIC] ? \
				           "dynamically" : "statically", 
				       !ehdr.e_ident[OI_STRIP] ?
				           "stripped" : "unstripped");
			    } else
				printf("OLF %s => ELF.\n", *argv);
			}
		    } else /* bad write */
			pwarn(progname, *argv, errno);	
		} else /* bad seek */
		    pwarn(progname, *argv, errno);
	    } /* okay? */
	    fsync(fd);
	    close(fd);

	} else /* couldn't handle file */
	    pwarn(progname, *argv, errno);
    } while (*(++argv) != NULL);
    

    return (retval);
}

void
pwarn(name, fname, errval)
    char *name;
    char *fname;
    int errval;
{
    fprintf(stderr, "%s: %s: %s.\n", name, fname, strerror(errval));
    retval = 1;
}

void
usage(void)
{
    int i;
    int col = 8;

    if (olf2elf) {
	fprintf(stderr,	"usage: %s [-v] file ...\n", progname);
    } else {
	fprintf(stderr,	"usage: %s [-v] [-o opsys] elffile ...\n", progname);
	fprintf(stderr, "where opsys is:\n\t");
	for (i = 1; os_namev[i] != NULL; i++) {
	    col = col + strlen(os_namev[i]) + 2;
	    if (col > 78) {
		fprintf(stderr, "\n\t");
		col = 8;
	    }
	    fprintf(stderr, "%s%s", os_namev[i],
		os_namev[i+1] ? ", " : "\n");
	}
    }
    exit(1);
}