/* * File dumping utility * * Copyright 2001,2005 Eric Pouech * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "config.h" #include "wine/port.h" #include <stdlib.h> #include <stdarg.h> #include <stdio.h> #ifdef HAVE_UNISTD_H # include <unistd.h> #endif #include <time.h> #ifdef HAVE_SYS_TYPES_H # include <sys/types.h> #endif #ifdef HAVE_SYS_STAT_H # include <sys/stat.h> #endif #ifdef HAVE_SYS_MMAN_H #include <sys/mman.h> #endif #include <fcntl.h> #define NONAMELESSUNION #define NONAMELESSSTRUCT #include "windef.h" #include "winbase.h" #include "winedump.h" #include "pe.h" #ifndef O_BINARY # define O_BINARY 0 #endif static void* dump_base; static unsigned long dump_total_len; void dump_data( const unsigned char *ptr, unsigned int size, const char *prefix ) { unsigned int i, j; printf( "%s", prefix ); if (!ptr) { printf("NULL\n"); return; } for (i = 0; i < size; i++) { printf( "%02x%c", ptr[i], (i % 16 == 7) ? '-' : ' ' ); if ((i % 16) == 15) { printf( " " ); for (j = 0; j < 16; j++) printf( "%c", isprint(ptr[i-15+j]) ? ptr[i-15+j] : '.' ); if (i < size-1) printf( "\n%s", prefix ); } } if (i % 16) { printf( "%*s ", 3 * (16-(i%16)), "" ); for (j = 0; j < i % 16; j++) printf( "%c", isprint(ptr[i-(i%16)+j]) ? ptr[i-(i%16)+j] : '.' ); } printf( "\n" ); } const char* get_time_str(DWORD _t) { time_t t = (time_t)_t; static char buf[128]; /* FIXME: I don't get the same values from MS' pedump running under Wine... * I wonder if Wine isn't broken wrt to GMT settings... */ strncpy(buf, ctime(&t), sizeof(buf)); buf[sizeof(buf) - 1] = '\0'; if (buf[strlen(buf)-1] == '\n') buf[strlen(buf)-1] = '\0'; return buf; } unsigned int strlenW( const WCHAR *str ) { const WCHAR *s = str; while (*s) s++; return s - str; } void dump_unicode_str( const WCHAR *str, int len ) { if (len == -1) len = strlenW( str ); printf( "L\""); while (len-- > 0 && *str) { WCHAR c = *str++; switch (c) { case '\n': printf( "\\n" ); break; case '\r': printf( "\\r" ); break; case '\t': printf( "\\t" ); break; case '"': printf( "\\\"" ); break; case '\\': printf( "\\\\" ); break; default: if (c >= ' ' && c <= 126) putchar(c); else printf( "\\u%04x",c); } } printf( "\"" ); } void* PRD(unsigned long prd, unsigned long len) { return (prd + len > dump_total_len) ? NULL : (char*)dump_base + prd; } unsigned long Offset(void* ptr) { if (ptr < dump_base) {printf("<<<<<ptr below\n");return 0;} if ((char *)ptr >= (char*)dump_base + dump_total_len) {printf("<<<<<ptr above\n");return 0;} return (char*)ptr - (char*)dump_base; } static void do_dump( enum FileSig sig, void* pmt ) { if (sig == SIG_NE) { ne_dump( dump_base, dump_total_len ); return; } if (sig == SIG_LE) { le_dump( dump_base, dump_total_len ); return; } pe_dump(pmt); } static enum FileSig check_headers(void** pmt) { WORD* pw; DWORD* pdw; IMAGE_DOS_HEADER* dh; enum FileSig sig; pw = PRD(0, sizeof(WORD)); if (!pw) {printf("Can't get main signature, aborting\n"); return 0;} *pmt = NULL; switch (*pw) { case IMAGE_DOS_SIGNATURE: sig = SIG_DOS; dh = PRD(0, sizeof(IMAGE_DOS_HEADER)); if (dh && dh->e_lfanew >= sizeof(*dh)) /* reasonable DOS header ? */ { /* the signature is the first DWORD */ pdw = PRD(dh->e_lfanew, sizeof(DWORD)); if (pdw) { if (*pdw == IMAGE_NT_SIGNATURE) { *pmt = PRD(dh->e_lfanew, sizeof(DWORD)+sizeof(IMAGE_FILE_HEADER)); sig = SIG_PE; } else if (*(WORD *)pdw == IMAGE_OS2_SIGNATURE) { sig = SIG_NE; } else if (*(WORD *)pdw == IMAGE_VXD_SIGNATURE) { sig = SIG_LE; } else { printf("No PE Signature found\n"); } } else { printf("Can't get the extented signature, aborting\n"); } } break; case 0x4944: /* "DI" */ sig = SIG_DBG; break; case 0x444D: /* "MD" */ pdw = PRD(0, sizeof(DWORD)); if (pdw && *pdw == 0x504D444D) /* "MDMP" */ sig = SIG_MDMP; else sig = SIG_UNKNOWN; break; default: printf("No known main signature (%.2s/%x), aborting\n", (char*)pw, *pw); sig = SIG_UNKNOWN; } return sig; } int dump_analysis(const char* name, void (*fn)(enum FileSig, void*), enum FileSig wanted_sig) { int fd; enum FileSig effective_sig; int ret = 1; struct stat s; void* pmt; setbuf(stdout, NULL); fd = open(name, O_RDONLY | O_BINARY); if (fd == -1) fatal("Can't open file"); if (fstat(fd, &s) < 0) fatal("Can't get size"); dump_total_len = s.st_size; #ifdef HAVE_MMAP if ((dump_base = mmap(NULL, dump_total_len, PROT_READ, MAP_PRIVATE, fd, 0)) == (void *)-1) #endif { if (!(dump_base = malloc( dump_total_len ))) fatal( "Out of memory" ); if ((unsigned long)read( fd, dump_base, dump_total_len ) != dump_total_len) fatal( "Cannot read file" ); } effective_sig = check_headers(&pmt); if (effective_sig == SIG_UNKNOWN) { printf("Can't get a recognized file signature, aborting\n"); ret = 0; } else if (wanted_sig == SIG_UNKNOWN || wanted_sig == effective_sig) { switch (effective_sig) { case SIG_UNKNOWN: /* shouldn't happen... */ ret = 0; break; case SIG_PE: case SIG_NE: case SIG_LE: printf("Contents of \"%s\": %ld bytes\n\n", name, dump_total_len); (*fn)(effective_sig, pmt); break; case SIG_DBG: dump_separate_dbg(); break; case SIG_DOS: ret = 0; break; case SIG_MDMP: mdmp_dump(); break; } } else { printf("Can't get a suitable file signature, aborting\n"); ret = 0; } if (ret) printf("Done dumping %s\n", name); #ifdef HAVE_MMAP if (munmap(dump_base, dump_total_len) == -1) #endif { free( dump_base ); } close(fd); return ret; } void dump_file(const char* name) { dump_analysis(name, do_dump, SIG_UNKNOWN); }