/* 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)\" */