|  | /* | 
|  | genidl - Generate interface listing from a Portable Executable. | 
|  | Copyright (C) 2009-2016  mingw-w64 project | 
|  |  | 
|  | 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 3 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, see <http://www.gnu.org/licenses/>. | 
|  | */ | 
|  |  | 
|  | #define _CRT_SECURE_NO_WARNINGS | 
|  | #include "genidl_cfg.h" | 
|  | #include "genidl_readpe.h" | 
|  | #include "genidl_typeinfo.h" | 
|  | #include "genidl_typinfo.h" | 
|  | #include "fsredir.h" | 
|  |  | 
|  | /* Configure globals.  */ | 
|  | int show_dump_too = 0; | 
|  | int generate_header = 0; | 
|  | int is_verbose = 0; | 
|  |  | 
|  | /* Process files.  */ | 
|  | size_t file_args_cnt = 0; | 
|  | char **file_args = NULL; | 
|  | const char *basedumpname = ""; | 
|  |  | 
|  | static char *get_idl_basename (const char *file); | 
|  |  | 
|  | #ifdef REDIRECTOR | 
|  | static int use_redirector = 0; /* Use/Disable FS redirector */ | 
|  | #endif | 
|  |  | 
|  | static void | 
|  | show_usage (void) | 
|  | { | 
|  | fprintf (stderr, "Usage: genidl [OPTION]... [FILE]...\n"); | 
|  | fprintf (stderr, "Dumps IDL information from typelib data found in PE32/PE32+ executables and\n" | 
|  | "TLB files.\n"); | 
|  | fprintf (stderr, "\n"); | 
|  | fprintf (stderr, "Options:\n" | 
|  | "  -b ARG, --basedumpname=ARG\n" | 
|  | "                           Specify ARG as prefix of generated idl files.\n" | 
|  | "  -H, --header				Generate header\n" | 
|  | "  -d, --dump               Dump additional internal debugging information.\n" | 
|  | "  -v, --verbose            Show additional status prints.\n" | 
|  | "  -h, --help               Show this help.\n" | 
|  | #ifdef REDIRECTOR | 
|  | "  -r, --disable-fs-redirector\n" | 
|  | "                           Disable Win64 FS redirection, for 32-bit\n" | 
|  | "                           gendef on 64-bit Windows\n" | 
|  | #endif | 
|  | ); | 
|  | fprintf (stderr, "\nReport bugs to <mingw-w64-public@lists.sourceforge.net>\n"); | 
|  | exit (1); | 
|  | } | 
|  |  | 
|  | static int | 
|  | scanArgs (int argc, char **argv) | 
|  | { | 
|  | int seen_error = 0; | 
|  | if (!argc) | 
|  | return -1; | 
|  | file_args = (char **) malloc (sizeof (char *) * argc); | 
|  | while (argc > 0) | 
|  | { | 
|  | char *h = *argv; | 
|  | if (h[0]  == '-') | 
|  | { | 
|  | h++; | 
|  | switch (*h) { | 
|  | case '-': /* Long arguments section */ | 
|  | h++; | 
|  | switch (*h) { | 
|  | case 'H': | 
|  | generate_header = 1; | 
|  | break; | 
|  | case 'd': | 
|  | if(! strcmp (h, "dump")) | 
|  | { | 
|  | show_dump_too = 1; | 
|  | break; | 
|  | } | 
|  | else | 
|  | goto unknown_fail; | 
|  | case 'h': | 
|  | if (!strcmp (h, "header")) | 
|  | { | 
|  | generate_header = 1; | 
|  | break; | 
|  | } | 
|  | if (!strcmp (h, "help")) return -2; | 
|  | goto unknown_fail; | 
|  | case 'v': | 
|  | if (! strcmp (h, "verbose")) | 
|  | is_verbose++; | 
|  | else | 
|  | goto unknown_fail; | 
|  | break; | 
|  | case 'b': | 
|  | if (! strncmp(h, "basedumpname=", 13)) | 
|  | { | 
|  | basedumpname = &(h[13]); | 
|  | break; | 
|  | } | 
|  | else | 
|  | goto unknown_fail; | 
|  | #ifdef REDIRECTOR | 
|  | case 'r': | 
|  | if(! strcmp (h, "disable-fs-redirector")) | 
|  | { | 
|  | use_redirector = 1; | 
|  | break; | 
|  | } | 
|  | else | 
|  | goto unknown_fail; | 
|  | #endif | 
|  | default: goto unknown_fail; | 
|  | } | 
|  | break; | 
|  | /* Short arguments section */ | 
|  | case 'd': | 
|  | if (h[1] == 0) | 
|  | show_dump_too = 1; | 
|  | else | 
|  | goto unknown_fail; | 
|  | break; | 
|  | case 'h': | 
|  | if (h[1] == 0) | 
|  | return -2; | 
|  | goto unknown_fail; | 
|  | case 'b': | 
|  | if (h[1] == 0) | 
|  | { | 
|  | basedumpname = *(++argv); | 
|  | --argc; | 
|  | break; | 
|  | } | 
|  | goto unknown_fail; | 
|  | case 'v': | 
|  | if (h[1] != 0) | 
|  | goto unknown_fail; | 
|  | is_verbose++; | 
|  | break; | 
|  | #ifdef REDIRECTOR | 
|  | case 'r': | 
|  | if (h[1] != 0) | 
|  | goto unknown_fail; | 
|  | use_redirector = 1; | 
|  | break; | 
|  | #endif | 
|  | default: | 
|  | unknown_fail: | 
|  | fprintf (stderr, "Option %s' is unknown.\n", *argv); | 
|  | seen_error = 1; | 
|  | break; | 
|  | } | 
|  | } | 
|  | else | 
|  | { | 
|  | file_args[file_args_cnt++] = *argv; | 
|  | } | 
|  | --argc; | 
|  | ++argv; | 
|  | } | 
|  | if (seen_error || file_args_cnt == 0) | 
|  | return -2; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int main(int argc,char **argv) | 
|  | { | 
|  | FILE *fp,*gp; | 
|  | long p; | 
|  | int32_t be64; | 
|  | size_t len; | 
|  | size_t i; | 
|  | unsigned char *dta = NULL; | 
|  | genidl_read_config ("./genidl.conf"); | 
|  | if (scanArgs (--argc, ++argv) < 0) | 
|  | { | 
|  | show_usage (); | 
|  | } | 
|  | #ifdef REDIRECTOR | 
|  | doredirect(use_redirector); | 
|  | #endif | 
|  | for (i = 0; i < file_args_cnt; i++) | 
|  | { | 
|  | char s[1024], *idl_basename,*org_basename; | 
|  | int start, end; | 
|  | gp = fopen (file_args[i], "rb"); | 
|  | if (!gp) | 
|  | { | 
|  | fprintf (stderr, "Failed to open file ,%s'.\n", file_args[i]); | 
|  | continue; | 
|  | } | 
|  | p = genidl_ispe (gp, &be64); | 
|  |  | 
|  | if (is_verbose) | 
|  | fprintf (stderr, "Found PE at %ld (%s bits)\n", p, !be64 ? "32" : "64"); | 
|  |  | 
|  | end = genidl_pe_typelib_resource_count (gp); | 
|  |  | 
|  | if (is_verbose) | 
|  | fprintf (stderr, "Contains %d typelib resource(s)\n", end); | 
|  | org_basename = get_idl_basename (file_args[i]); | 
|  | idl_basename = strdup (org_basename); | 
|  | if (strrchr (idl_basename, '.') != NULL) | 
|  | *strrchr (idl_basename, '.') = 0; | 
|  |  | 
|  | for (start = 0; start < end; start++) | 
|  | { | 
|  | genidl_pe_typelib_resource_read (gp, start, &dta, &len); | 
|  | if (generate_header == 0) | 
|  | { | 
|  | if (end != 1) | 
|  | sprintf (s, "%s%s_%d.idl", idl_basename, basedumpname, start); | 
|  | else | 
|  | sprintf (s, "%s%s.idl", idl_basename, basedumpname); | 
|  | fp = fopen (s, "wb"); | 
|  | if (fp) | 
|  | { | 
|  | sTI2TypLib *tl = TI2_typlib_init (dta, (size_t) len); | 
|  | if (tl) | 
|  | { | 
|  | TI2_typlib_idl (fp, tl, org_basename); | 
|  | TI2_typlib_dest (tl); | 
|  | } | 
|  | if (show_dump_too) | 
|  | dumpInfo (fp, dta, len); | 
|  | fclose (fp); | 
|  | } | 
|  | } | 
|  | else if (generate_header == 1) | 
|  | { | 
|  | if (end != 1) | 
|  | sprintf (s, "%s%s_%d.h", idl_basename, basedumpname, start); | 
|  | else | 
|  | sprintf (s, "%s%s.h", idl_basename, basedumpname); | 
|  | fp = fopen (s, "wb"); | 
|  | if (fp) | 
|  | { | 
|  | sTI2TypLib *tl = TI2_typlib_init (dta, (size_t) len); | 
|  | if (tl) | 
|  | { | 
|  | TI2_typlib_hdr (fp, tl, org_basename); | 
|  | TI2_typlib_dest (tl); | 
|  | } | 
|  | fclose (fp); | 
|  | } | 
|  | } | 
|  | } | 
|  | free (idl_basename); | 
|  | free (org_basename); | 
|  | fclose (gp); | 
|  | } | 
|  | /* genidl_save_config_fp (stderr); */ | 
|  | genidl_save_config ("./genidl.conf"); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static char * | 
|  | get_idl_basename (const char *file) | 
|  | { | 
|  | char *h, *p; | 
|  | if (!file || *file == 0) | 
|  | return strdup (""); | 
|  | h = strrchr (file, '\\'); | 
|  | p = strrchr (file, '/'); | 
|  | if (!h && !p) | 
|  | h = strdup (file); | 
|  | else if (!h) | 
|  | h = strdup (p + 1); | 
|  | else if (!p) | 
|  | h = strdup (h + 1); | 
|  | else if (p < h) | 
|  | h = strdup (h + 1); | 
|  | else | 
|  | h = strdup (p + 1); | 
|  | genidl_strlwr (h); | 
|  | return h; | 
|  | } | 
|  |  | 
|  | char * | 
|  | genidl_strlwr (char *s) | 
|  | { | 
|  | char *h = s; | 
|  | if (!h) | 
|  | return NULL; | 
|  | while (*h != 0) | 
|  | { | 
|  | if (h[0] >= 'A' && h[0] <= 'Z') | 
|  | h[0] = (h[0] - 'A') + 'a'; | 
|  | ++h; | 
|  | } | 
|  |  | 
|  | return s; | 
|  | } |