| /* | 
 |     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; | 
 | } |