/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

/*
 * Test program to mangle 1 bit in a binary
 */

#include "nspr.h"
#include "plstr.h"
#include "plgetopt.h"
#include "prio.h"

static PRFileDesc *pr_stderr;
static void
usage(char *program_name)
{

    PR_fprintf(pr_stderr, "Usage:");
    PR_fprintf(pr_stderr, "%s -i shared_library_name -o byte_offset -b bit\n", program_name);
}

int
main(int argc, char **argv)
{
    /* buffers and locals */
    PLOptState *optstate;
    char *programName;
    char cbuf;

    /* parameter set variables */
    const char *libFile = NULL;
    int bitOffset = -1;

    /* return values */
    int retval = 2; /* 0 - test succeeded.
                     * 1 - illegal args
                     * 2 - function failed */
    PRFileDesc *fd = NULL;
    int bytesRead;
    int bytesWritten;
    PROffset32 offset = -1;
    PROffset32 pos;

    programName = PL_strrchr(argv[0], '/');
    programName = programName ? (programName + 1) : argv[0];

    pr_stderr = PR_STDERR;

    optstate = PL_CreateOptState(argc, argv, "i:o:b:");
    if (optstate == NULL) {
        return 1;
    }

    while (PL_GetNextOpt(optstate) == PL_OPT_OK) {
        switch (optstate->option) {
            case 'i':
                libFile = optstate->value;
                break;

            case 'o':
                offset = atoi(optstate->value);
                break;

            case 'b':
                bitOffset = atoi(optstate->value);
                break;
        }
    }

    if (libFile == NULL) {
        usage(programName);
        return 1;
    }
    if ((bitOffset >= 8) || (bitOffset < 0)) {
        usage(programName);
        return 1;
    }

    /* open the target signature file */
    fd = PR_OpenFile(libFile, PR_RDWR, 0666);
    if (fd == NULL) {
        /* lperror(libFile); */
        PR_fprintf(pr_stderr, "Couldn't Open %s\n", libFile);
        goto loser;
    }

    if (offset < 0) { /* convert to positive offset */
        pos = PR_Seek(fd, offset, PR_SEEK_END);
        if (pos == -1) {
            PR_fprintf(pr_stderr, "Seek for read on %s (to %d) failed\n",
                       libFile, offset);
            goto loser;
        }
        offset = pos;
    }

    /* read the byte */
    pos = PR_Seek(fd, offset, PR_SEEK_SET);
    if (pos != offset) {
        PR_fprintf(pr_stderr, "Seek for read on %s (to %d) failed\n",
                   libFile, offset);
        goto loser;
    }
    bytesRead = PR_Read(fd, &cbuf, 1);
    if (bytesRead != 1) {
        PR_fprintf(pr_stderr, "Read on %s (to %d) failed\n", libFile, offset);
        goto loser;
    }

    PR_fprintf(pr_stderr, "Changing byte 0x%08x (%d): from %02x (%d) to ",
               offset, offset, (unsigned char)cbuf, (unsigned char)cbuf);
    /* change it */
    cbuf ^= 1 << bitOffset;
    PR_fprintf(pr_stderr, "%02x (%d)\n",
               (unsigned char)cbuf, (unsigned char)cbuf);

    /* write it back out */
    pos = PR_Seek(fd, offset, PR_SEEK_SET);
    if (pos != offset) {
        PR_fprintf(pr_stderr, "Seek for write on %s (to %d) failed\n",
                   libFile, offset);
        goto loser;
    }
    bytesWritten = PR_Write(fd, &cbuf, 1);
    if (bytesWritten != 1) {
        PR_fprintf(pr_stderr, "Write on %s (to %d) failed\n", libFile, offset);
        goto loser;
    }

    retval = 0;

loser:
    if (fd)
        PR_Close(fd);
    PR_Cleanup();
    return retval;
}

/*#DEFINES += -DSHLIB_SUFFIX=\"$(DLL_SUFFIX)\" -DSHLIB_PREFIX=\"$(DLL_PREFIX)\" */