main.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716
  1. /*
  2. * Main function
  3. *
  4. * Copyright 1993 Robert J. Amstadt
  5. * Copyright 1995 Martin von Loewis
  6. * Copyright 1995, 1996, 1997 Alexandre Julliard
  7. * Copyright 1997 Eric Youngdale
  8. * Copyright 1999 Ulrich Weigand
  9. *
  10. * This library is free software; you can redistribute it and/or
  11. * modify it under the terms of the GNU Lesser General Public
  12. * License as published by the Free Software Foundation; either
  13. * version 2.1 of the License, or (at your option) any later version.
  14. *
  15. * This library is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  18. * Lesser General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU Lesser General Public
  21. * License along with this library; if not, write to the Free Software
  22. * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
  23. */
  24. #include "config.h"
  25. #include "wine/port.h"
  26. #include <assert.h>
  27. #include <stdio.h>
  28. #include <signal.h>
  29. #include <errno.h>
  30. #include <string.h>
  31. #include <stdarg.h>
  32. #include <ctype.h>
  33. #ifdef HAVE_GETOPT_H
  34. # include <getopt.h>
  35. #endif
  36. #include "build.h"
  37. int UsePIC = 0;
  38. int nb_errors = 0;
  39. int display_warnings = 0;
  40. int kill_at = 0;
  41. int verbose = 0;
  42. int link_ext_symbols = 0;
  43. int force_pointer_size = 0;
  44. int unwind_tables = 0;
  45. int unix_lib = 0;
  46. #ifdef __i386__
  47. enum target_cpu target_cpu = CPU_x86;
  48. #elif defined(__x86_64__)
  49. enum target_cpu target_cpu = CPU_x86_64;
  50. #elif defined(__powerpc__)
  51. enum target_cpu target_cpu = CPU_POWERPC;
  52. #elif defined(__arm__)
  53. enum target_cpu target_cpu = CPU_ARM;
  54. #elif defined(__aarch64__)
  55. enum target_cpu target_cpu = CPU_ARM64;
  56. #else
  57. #error Unsupported CPU
  58. #endif
  59. #ifdef __APPLE__
  60. enum target_platform target_platform = PLATFORM_APPLE;
  61. #elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
  62. enum target_platform target_platform = PLATFORM_FREEBSD;
  63. #elif defined(__sun)
  64. enum target_platform target_platform = PLATFORM_SOLARIS;
  65. #elif defined(_WIN32)
  66. enum target_platform target_platform = PLATFORM_WINDOWS;
  67. #else
  68. enum target_platform target_platform = PLATFORM_UNSPECIFIED;
  69. #endif
  70. char *target_alias = NULL;
  71. char *input_file_name = NULL;
  72. char *spec_file_name = NULL;
  73. FILE *output_file = NULL;
  74. const char *output_file_name = NULL;
  75. static int fake_module;
  76. struct strarray lib_path = { 0 };
  77. struct strarray tools_path = { 0 };
  78. struct strarray as_command = { 0 };
  79. struct strarray cc_command = { 0 };
  80. struct strarray ld_command = { 0 };
  81. struct strarray nm_command = { 0 };
  82. char *cpu_option = NULL;
  83. char *fpu_option = NULL;
  84. char *arch_option = NULL;
  85. #ifdef __SOFTFP__
  86. const char *float_abi_option = "soft";
  87. #else
  88. const char *float_abi_option = "softfp";
  89. #endif
  90. #ifdef __thumb__
  91. int thumb_mode = 1;
  92. #else
  93. int thumb_mode = 0;
  94. #endif
  95. static struct strarray res_files;
  96. /* execution mode */
  97. enum exec_mode_values
  98. {
  99. MODE_NONE,
  100. MODE_DLL,
  101. MODE_EXE,
  102. MODE_DEF,
  103. MODE_IMPLIB,
  104. MODE_STATICLIB,
  105. MODE_BUILTIN,
  106. MODE_RESOURCES
  107. };
  108. static enum exec_mode_values exec_mode = MODE_NONE;
  109. static const struct
  110. {
  111. const char *name;
  112. enum target_platform platform;
  113. } platform_names[] =
  114. {
  115. { "macos", PLATFORM_APPLE },
  116. { "darwin", PLATFORM_APPLE },
  117. { "freebsd", PLATFORM_FREEBSD },
  118. { "solaris", PLATFORM_SOLARIS },
  119. { "mingw32", PLATFORM_WINDOWS },
  120. { "windows", PLATFORM_WINDOWS },
  121. { "winnt", PLATFORM_WINDOWS }
  122. };
  123. /* set the dll file name from the input file name */
  124. static void set_dll_file_name( const char *name, DLLSPEC *spec )
  125. {
  126. char *p;
  127. if (spec->file_name) return;
  128. if ((p = strrchr( name, '\\' ))) name = p + 1;
  129. if ((p = strrchr( name, '/' ))) name = p + 1;
  130. spec->file_name = xmalloc( strlen(name) + 5 );
  131. strcpy( spec->file_name, name );
  132. if ((p = strrchr( spec->file_name, '.' )))
  133. {
  134. if (!strcmp( p, ".spec" ) || !strcmp( p, ".def" )) *p = 0;
  135. }
  136. }
  137. /* set the dll name from the file name */
  138. static void init_dll_name( DLLSPEC *spec )
  139. {
  140. if (!spec->file_name && output_file_name)
  141. {
  142. char *p;
  143. spec->file_name = xstrdup( output_file_name );
  144. if ((p = strrchr( spec->file_name, '.' ))) *p = 0;
  145. }
  146. if (!spec->dll_name && spec->file_name) /* set default name from file name */
  147. {
  148. char *p;
  149. spec->dll_name = xstrdup( spec->file_name );
  150. if ((p = strrchr( spec->dll_name, '.' ))) *p = 0;
  151. }
  152. if (spec->dll_name) spec->c_name = make_c_identifier( spec->dll_name );
  153. }
  154. /* set the dll subsystem */
  155. static void set_subsystem( const char *subsystem, DLLSPEC *spec )
  156. {
  157. char *major, *minor, *str = xstrdup( subsystem );
  158. if ((major = strchr( str, ':' ))) *major++ = 0;
  159. if (!strcmp( str, "native" )) spec->subsystem = IMAGE_SUBSYSTEM_NATIVE;
  160. else if (!strcmp( str, "windows" )) spec->subsystem = IMAGE_SUBSYSTEM_WINDOWS_GUI;
  161. else if (!strcmp( str, "console" )) spec->subsystem = IMAGE_SUBSYSTEM_WINDOWS_CUI;
  162. else if (!strcmp( str, "wince" )) spec->subsystem = IMAGE_SUBSYSTEM_WINDOWS_CE_GUI;
  163. else if (!strcmp( str, "win16" )) spec->type = SPEC_WIN16;
  164. else fatal_error( "Invalid subsystem name '%s'\n", subsystem );
  165. if (major)
  166. {
  167. if ((minor = strchr( major, '.' )))
  168. {
  169. *minor++ = 0;
  170. spec->subsystem_minor = atoi( minor );
  171. }
  172. spec->subsystem_major = atoi( major );
  173. }
  174. free( str );
  175. }
  176. /* set the target CPU and platform */
  177. static void set_target( const char *target )
  178. {
  179. unsigned int i;
  180. char *p, *spec = xstrdup( target );
  181. /* target specification is in the form CPU-MANUFACTURER-OS or CPU-MANUFACTURER-KERNEL-OS */
  182. target_alias = xstrdup( target );
  183. /* get the CPU part */
  184. if ((p = strchr( spec, '-' )))
  185. {
  186. int cpu;
  187. *p++ = 0;
  188. cpu = get_cpu_from_name( spec );
  189. if (cpu == -1) fatal_error( "Unrecognized CPU '%s'\n", spec );
  190. target_cpu = cpu;
  191. }
  192. else if (!strcmp( spec, "mingw32" ))
  193. {
  194. target_cpu = CPU_x86;
  195. p = spec;
  196. }
  197. else
  198. fatal_error( "Invalid target specification '%s'\n", target );
  199. /* get the OS part */
  200. target_platform = PLATFORM_UNSPECIFIED; /* default value */
  201. for (;;)
  202. {
  203. for (i = 0; i < ARRAY_SIZE(platform_names); i++)
  204. {
  205. if (!strncmp( platform_names[i].name, p, strlen(platform_names[i].name) ))
  206. {
  207. target_platform = platform_names[i].platform;
  208. break;
  209. }
  210. }
  211. if (target_platform != PLATFORM_UNSPECIFIED || !(p = strchr( p, '-' ))) break;
  212. p++;
  213. }
  214. free( spec );
  215. }
  216. /* cleanup on program exit */
  217. static void cleanup(void)
  218. {
  219. if (output_file_name) unlink( output_file_name );
  220. }
  221. /* clean things up when aborting on a signal */
  222. static void exit_on_signal( int sig )
  223. {
  224. exit(1); /* this will call atexit functions */
  225. }
  226. /*******************************************************************
  227. * command-line option handling
  228. */
  229. static const char usage_str[] =
  230. "Usage: winebuild [OPTIONS] [FILES]\n\n"
  231. "Options:\n"
  232. " --as-cmd=AS Command to use for assembling (default: as)\n"
  233. " -b, --target=TARGET Specify target CPU and platform for cross-compiling\n"
  234. " -B PREFIX Look for build tools in the PREFIX directory\n"
  235. " --cc-cmd=CC C compiler to use for assembling (default: fall back to --as-cmd)\n"
  236. " -d, --delay-lib=LIB Import the specified library in delayed mode\n"
  237. " -D SYM Ignored for C flags compatibility\n"
  238. " -e, --entry=FUNC Set the DLL entry point function (default: DllMain)\n"
  239. " -E, --export=FILE Export the symbols defined in the .spec or .def file\n"
  240. " --external-symbols Allow linking to external symbols\n"
  241. " -f FLAGS Compiler flags (-fPIC and -fasynchronous-unwind-tables are supported)\n"
  242. " -F, --filename=DLLFILE Set the DLL filename (default: from input file name)\n"
  243. " --fake-module Create a fake binary module\n"
  244. " -h, --help Display this help message\n"
  245. " -H, --heap=SIZE Set the heap size for a Win16 dll\n"
  246. " -I DIR Ignored for C flags compatibility\n"
  247. " -k, --kill-at Kill stdcall decorations in generated .def files\n"
  248. " -K, FLAGS Compiler flags (only -KPIC is supported)\n"
  249. " --large-address-aware Support an address space larger than 2Gb\n"
  250. " --ld-cmd=LD Command to use for linking (default: ld)\n"
  251. " -l, --library=LIB Import the specified library\n"
  252. " -L, --library-path=DIR Look for imports libraries in DIR\n"
  253. " -m16, -m32, -m64 Force building 16-bit, 32-bit resp. 64-bit code\n"
  254. " -M, --main-module=MODULE Set the name of the main module for a Win16 dll\n"
  255. " --nm-cmd=NM Command to use to get undefined symbols (default: nm)\n"
  256. " --nxcompat=y|n Set the NX compatibility flag (default: yes)\n"
  257. " -N, --dll-name=DLLNAME Set the DLL name (default: from input file name)\n"
  258. " -o, --output=NAME Set the output file name (default: stdout)\n"
  259. " -r, --res=RSRC.RES Load resources from RSRC.RES\n"
  260. " --save-temps Do not delete the generated intermediate files\n"
  261. " --subsystem=SUBSYS Set the subsystem (one of native, windows, console, wince)\n"
  262. " -u, --undefined=SYMBOL Add an undefined reference to SYMBOL when linking\n"
  263. " -v, --verbose Display the programs invoked\n"
  264. " --version Print the version and exit\n"
  265. " -w, --warnings Turn on warnings\n"
  266. "\nMode options:\n"
  267. " --dll Build a library from a .spec file and object files\n"
  268. " --def Build a .def file from a .spec file\n"
  269. " --exe Build an executable from object files\n"
  270. " --implib Build an import library\n"
  271. " --staticlib Build a static library\n"
  272. " --builtin Mark a library as a Wine builtin\n"
  273. " --resources Build a .o or .res file for the resource files\n\n"
  274. "The mode options are mutually exclusive; you must specify one and only one.\n\n";
  275. enum long_options_values
  276. {
  277. LONG_OPT_DLL = 1,
  278. LONG_OPT_DEF,
  279. LONG_OPT_EXE,
  280. LONG_OPT_IMPLIB,
  281. LONG_OPT_BUILTIN,
  282. LONG_OPT_ASCMD,
  283. LONG_OPT_CCCMD,
  284. LONG_OPT_EXTERNAL_SYMS,
  285. LONG_OPT_FAKE_MODULE,
  286. LONG_OPT_LARGE_ADDRESS_AWARE,
  287. LONG_OPT_LDCMD,
  288. LONG_OPT_NMCMD,
  289. LONG_OPT_NXCOMPAT,
  290. LONG_OPT_RESOURCES,
  291. LONG_OPT_SAVE_TEMPS,
  292. LONG_OPT_STATICLIB,
  293. LONG_OPT_SUBSYSTEM,
  294. LONG_OPT_VERSION
  295. };
  296. static const char short_options[] = "B:C:D:E:F:H:I:K:L:M:N:b:d:e:f:hkl:m:o:r:u:vw";
  297. static const struct option long_options[] =
  298. {
  299. { "dll", 0, 0, LONG_OPT_DLL },
  300. { "def", 0, 0, LONG_OPT_DEF },
  301. { "exe", 0, 0, LONG_OPT_EXE },
  302. { "implib", 0, 0, LONG_OPT_IMPLIB },
  303. { "staticlib", 0, 0, LONG_OPT_STATICLIB },
  304. { "builtin", 0, 0, LONG_OPT_BUILTIN },
  305. { "as-cmd", 1, 0, LONG_OPT_ASCMD },
  306. { "cc-cmd", 1, 0, LONG_OPT_CCCMD },
  307. { "external-symbols", 0, 0, LONG_OPT_EXTERNAL_SYMS },
  308. { "fake-module", 0, 0, LONG_OPT_FAKE_MODULE },
  309. { "large-address-aware", 0, 0, LONG_OPT_LARGE_ADDRESS_AWARE },
  310. { "ld-cmd", 1, 0, LONG_OPT_LDCMD },
  311. { "nm-cmd", 1, 0, LONG_OPT_NMCMD },
  312. { "nxcompat", 1, 0, LONG_OPT_NXCOMPAT },
  313. { "resources", 0, 0, LONG_OPT_RESOURCES },
  314. { "save-temps", 0, 0, LONG_OPT_SAVE_TEMPS },
  315. { "subsystem", 1, 0, LONG_OPT_SUBSYSTEM },
  316. { "version", 0, 0, LONG_OPT_VERSION },
  317. /* aliases for short options */
  318. { "target", 1, 0, 'b' },
  319. { "delay-lib", 1, 0, 'd' },
  320. { "export", 1, 0, 'E' },
  321. { "entry", 1, 0, 'e' },
  322. { "filename", 1, 0, 'F' },
  323. { "help", 0, 0, 'h' },
  324. { "heap", 1, 0, 'H' },
  325. { "kill-at", 0, 0, 'k' },
  326. { "library", 1, 0, 'l' },
  327. { "library-path", 1, 0, 'L' },
  328. { "main-module", 1, 0, 'M' },
  329. { "dll-name", 1, 0, 'N' },
  330. { "output", 1, 0, 'o' },
  331. { "res", 1, 0, 'r' },
  332. { "undefined", 1, 0, 'u' },
  333. { "verbose", 0, 0, 'v' },
  334. { "warnings", 0, 0, 'w' },
  335. { NULL, 0, 0, 0 }
  336. };
  337. static void usage( int exit_code )
  338. {
  339. fprintf( stderr, "%s", usage_str );
  340. exit( exit_code );
  341. }
  342. static void set_exec_mode( enum exec_mode_values mode )
  343. {
  344. if (exec_mode != MODE_NONE) usage(1);
  345. exec_mode = mode;
  346. }
  347. /* get the default entry point for a given spec file */
  348. static const char *get_default_entry_point( const DLLSPEC *spec )
  349. {
  350. if (spec->characteristics & IMAGE_FILE_DLL) return "__wine_spec_dll_entry";
  351. if (spec->subsystem == IMAGE_SUBSYSTEM_NATIVE) return "__wine_spec_drv_entry";
  352. if (spec->type == SPEC_WIN16) return "__wine_spec_exe16_entry";
  353. return "__wine_spec_exe_entry";
  354. }
  355. /* parse options from the argv array and remove all the recognized ones */
  356. static char **parse_options( int argc, char **argv, DLLSPEC *spec )
  357. {
  358. char *p;
  359. int optc;
  360. int save_temps = 0;
  361. while ((optc = getopt_long( argc, argv, short_options, long_options, NULL )) != -1)
  362. {
  363. switch(optc)
  364. {
  365. case 'B':
  366. strarray_add( &tools_path, xstrdup( optarg ), NULL );
  367. break;
  368. case 'D':
  369. /* ignored */
  370. break;
  371. case 'E':
  372. spec_file_name = xstrdup( optarg );
  373. set_dll_file_name( optarg, spec );
  374. break;
  375. case 'F':
  376. spec->file_name = xstrdup( optarg );
  377. break;
  378. case 'H':
  379. if (!isdigit(optarg[0]))
  380. fatal_error( "Expected number argument with -H option instead of '%s'\n", optarg );
  381. spec->heap_size = atoi(optarg);
  382. if (spec->heap_size > 65535)
  383. fatal_error( "Invalid heap size %d, maximum is 65535\n", spec->heap_size );
  384. break;
  385. case 'I':
  386. /* ignored */
  387. break;
  388. case 'K':
  389. /* ignored, because cc generates correct code. */
  390. break;
  391. case 'L':
  392. strarray_add( &lib_path, xstrdup( optarg ), NULL );
  393. break;
  394. case 'm':
  395. if (!strcmp( optarg, "16" )) spec->type = SPEC_WIN16;
  396. else if (!strcmp( optarg, "32" )) force_pointer_size = 4;
  397. else if (!strcmp( optarg, "64" )) force_pointer_size = 8;
  398. else if (!strcmp( optarg, "arm" )) thumb_mode = 0;
  399. else if (!strcmp( optarg, "thumb" )) thumb_mode = 1;
  400. else if (!strcmp( optarg, "unix" )) unix_lib = 1;
  401. else if (!strncmp( optarg, "cpu=", 4 )) cpu_option = xstrdup( optarg + 4 );
  402. else if (!strncmp( optarg, "fpu=", 4 )) fpu_option = xstrdup( optarg + 4 );
  403. else if (!strncmp( optarg, "arch=", 5 )) arch_option = xstrdup( optarg + 5 );
  404. else if (!strncmp( optarg, "float-abi=", 10 )) float_abi_option = xstrdup( optarg + 10 );
  405. else fatal_error( "Unknown -m option '%s'\n", optarg );
  406. break;
  407. case 'M':
  408. spec->main_module = xstrdup( optarg );
  409. break;
  410. case 'N':
  411. spec->dll_name = xstrdup( optarg );
  412. break;
  413. case 'b':
  414. set_target( optarg );
  415. break;
  416. case 'd':
  417. add_delayed_import( optarg );
  418. break;
  419. case 'e':
  420. spec->init_func = xstrdup( optarg );
  421. if ((p = strchr( spec->init_func, '@' ))) *p = 0; /* kill stdcall decoration */
  422. break;
  423. case 'f':
  424. if (!strcmp( optarg, "PIC") || !strcmp( optarg, "pic")) UsePIC = 1;
  425. else if (!strcmp( optarg, "asynchronous-unwind-tables")) unwind_tables = 1;
  426. else if (!strcmp( optarg, "no-asynchronous-unwind-tables")) unwind_tables = 0;
  427. /* ignore all other flags */
  428. break;
  429. case 'h':
  430. usage(0);
  431. break;
  432. case 'k':
  433. kill_at = 1;
  434. break;
  435. case 'l':
  436. add_import_dll( optarg, NULL );
  437. break;
  438. case 'o':
  439. if (unlink( optarg ) == -1 && errno != ENOENT)
  440. fatal_error( "Unable to create output file '%s'\n", optarg );
  441. output_file_name = xstrdup( optarg );
  442. break;
  443. case 'r':
  444. strarray_add( &res_files, xstrdup( optarg ), NULL );
  445. break;
  446. case 'u':
  447. add_extra_ld_symbol( optarg );
  448. break;
  449. case 'v':
  450. verbose++;
  451. break;
  452. case 'w':
  453. display_warnings = 1;
  454. break;
  455. case LONG_OPT_DLL:
  456. set_exec_mode( MODE_DLL );
  457. break;
  458. case LONG_OPT_DEF:
  459. set_exec_mode( MODE_DEF );
  460. break;
  461. case LONG_OPT_EXE:
  462. set_exec_mode( MODE_EXE );
  463. if (!spec->subsystem) spec->subsystem = IMAGE_SUBSYSTEM_WINDOWS_GUI;
  464. break;
  465. case LONG_OPT_IMPLIB:
  466. set_exec_mode( MODE_IMPLIB );
  467. break;
  468. case LONG_OPT_STATICLIB:
  469. set_exec_mode( MODE_STATICLIB );
  470. break;
  471. case LONG_OPT_BUILTIN:
  472. set_exec_mode( MODE_BUILTIN );
  473. break;
  474. case LONG_OPT_ASCMD:
  475. as_command = strarray_fromstring( optarg, " " );
  476. break;
  477. case LONG_OPT_CCCMD:
  478. cc_command = strarray_fromstring( optarg, " " );
  479. break;
  480. case LONG_OPT_FAKE_MODULE:
  481. fake_module = 1;
  482. break;
  483. case LONG_OPT_EXTERNAL_SYMS:
  484. link_ext_symbols = 1;
  485. break;
  486. case LONG_OPT_LARGE_ADDRESS_AWARE:
  487. spec->characteristics |= IMAGE_FILE_LARGE_ADDRESS_AWARE;
  488. break;
  489. case LONG_OPT_LDCMD:
  490. ld_command = strarray_fromstring( optarg, " " );
  491. break;
  492. case LONG_OPT_NMCMD:
  493. nm_command = strarray_fromstring( optarg, " " );
  494. break;
  495. case LONG_OPT_NXCOMPAT:
  496. if (optarg[0] == 'n' || optarg[0] == 'N')
  497. spec->dll_characteristics &= ~IMAGE_DLLCHARACTERISTICS_NX_COMPAT;
  498. break;
  499. case LONG_OPT_RESOURCES:
  500. set_exec_mode( MODE_RESOURCES );
  501. break;
  502. case LONG_OPT_SAVE_TEMPS:
  503. save_temps = 1;
  504. break;
  505. case LONG_OPT_SUBSYSTEM:
  506. set_subsystem( optarg, spec );
  507. break;
  508. case LONG_OPT_VERSION:
  509. printf( "winebuild version " PACKAGE_VERSION "\n" );
  510. exit(0);
  511. case '?':
  512. usage(1);
  513. break;
  514. }
  515. }
  516. if (!save_temps) atexit( cleanup_tmp_files );
  517. if (spec->file_name && !strchr( spec->file_name, '.' ))
  518. strcat( spec->file_name, exec_mode == MODE_EXE ? ".exe" : ".dll" );
  519. init_dll_name( spec );
  520. switch (target_cpu)
  521. {
  522. case CPU_x86:
  523. if (force_pointer_size == 8) target_cpu = CPU_x86_64;
  524. break;
  525. case CPU_x86_64:
  526. if (force_pointer_size == 4) target_cpu = CPU_x86;
  527. break;
  528. default:
  529. if (force_pointer_size == 8)
  530. fatal_error( "Cannot build 64-bit code for this CPU\n" );
  531. break;
  532. }
  533. return &argv[optind];
  534. }
  535. /* load all specified resource files */
  536. static void load_resources( char *argv[], DLLSPEC *spec )
  537. {
  538. int i;
  539. char **ptr, **last;
  540. switch (spec->type)
  541. {
  542. case SPEC_WIN16:
  543. for (i = 0; i < res_files.count; i++) load_res16_file( res_files.str[i], spec );
  544. break;
  545. case SPEC_WIN32:
  546. for (i = 0; i < res_files.count; i++)
  547. {
  548. if (!load_res32_file( res_files.str[i], spec ))
  549. fatal_error( "%s is not a valid Win32 resource file\n", res_files.str[i] );
  550. }
  551. /* load any resource file found in the remaining arguments */
  552. for (ptr = last = argv; *ptr; ptr++)
  553. {
  554. if (!load_res32_file( *ptr, spec ))
  555. *last++ = *ptr; /* not a resource file, keep it in the list */
  556. }
  557. *last = NULL;
  558. break;
  559. }
  560. }
  561. /* add input files that look like import libs to the import list */
  562. static void load_import_libs( char *argv[] )
  563. {
  564. char **ptr, **last;
  565. for (ptr = last = argv; *ptr; ptr++)
  566. {
  567. if (strendswith( *ptr, ".def" ))
  568. add_import_dll( NULL, *ptr );
  569. else
  570. *last++ = *ptr; /* not an import dll, keep it in the list */
  571. }
  572. *last = NULL;
  573. }
  574. static int parse_input_file( DLLSPEC *spec )
  575. {
  576. FILE *input_file = open_input_file( NULL, spec_file_name );
  577. char *extension = strrchr( spec_file_name, '.' );
  578. int result;
  579. spec->src_name = xstrdup( input_file_name );
  580. if (extension && !strcmp( extension, ".def" ))
  581. result = parse_def_file( input_file, spec );
  582. else
  583. result = parse_spec_file( input_file, spec );
  584. close_input_file( input_file );
  585. return result;
  586. }
  587. /*******************************************************************
  588. * main
  589. */
  590. int main(int argc, char **argv)
  591. {
  592. DLLSPEC *spec = alloc_dll_spec();
  593. #ifdef SIGHUP
  594. signal( SIGHUP, exit_on_signal );
  595. #endif
  596. signal( SIGTERM, exit_on_signal );
  597. signal( SIGINT, exit_on_signal );
  598. argv = parse_options( argc, argv, spec );
  599. atexit( cleanup ); /* make sure we remove the output file on exit */
  600. switch(exec_mode)
  601. {
  602. case MODE_DLL:
  603. if (spec->subsystem != IMAGE_SUBSYSTEM_NATIVE)
  604. spec->characteristics |= IMAGE_FILE_DLL;
  605. /* fall through */
  606. case MODE_EXE:
  607. load_resources( argv, spec );
  608. if (spec_file_name && !parse_input_file( spec )) break;
  609. if (!spec->init_func) spec->init_func = xstrdup( get_default_entry_point( spec ));
  610. if (fake_module)
  611. {
  612. if (spec->type == SPEC_WIN16) output_fake_module16( spec );
  613. else output_fake_module( spec );
  614. break;
  615. }
  616. if (target_platform != PLATFORM_WINDOWS)
  617. {
  618. load_import_libs( argv );
  619. read_undef_symbols( spec, argv );
  620. resolve_imports( spec );
  621. }
  622. if (spec->type == SPEC_WIN16) output_spec16_file( spec );
  623. else output_spec32_file( spec );
  624. break;
  625. case MODE_DEF:
  626. if (argv[0]) fatal_error( "file argument '%s' not allowed in this mode\n", argv[0] );
  627. if (!spec_file_name) fatal_error( "missing .spec file\n" );
  628. if (!parse_input_file( spec )) break;
  629. open_output_file();
  630. output_def_file( spec, 0 );
  631. close_output_file();
  632. break;
  633. case MODE_IMPLIB:
  634. if (!spec_file_name) fatal_error( "missing .spec file\n" );
  635. if (!parse_input_file( spec )) break;
  636. output_static_lib( spec, argv );
  637. break;
  638. case MODE_STATICLIB:
  639. output_static_lib( NULL, argv );
  640. break;
  641. case MODE_BUILTIN:
  642. if (!argv[0]) fatal_error( "missing file argument for --builtin option\n" );
  643. make_builtin_files( argv );
  644. break;
  645. case MODE_RESOURCES:
  646. load_resources( argv, spec );
  647. output_res_o_file( spec );
  648. break;
  649. default:
  650. usage(1);
  651. break;
  652. }
  653. if (nb_errors) exit(1);
  654. output_file_name = NULL;
  655. return 0;
  656. }