1 /* dlltool.c -- tool to generate stuff for PE style DLLs
2 Copyright (C) 1995, 96, 97, 98, 99, 2000 Free Software Foundation, Inc.
4 This file is part of GNU Binutils.
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
23 This program allows you to build the files necessary to create
24 DLLs to run on a system which understands PE format image files.
27 See "Peering Inside the PE: A Tour of the Win32 Portable Executable
28 File Format", MSJ 1994, Volume 9 for more information.
29 Also see "Microsoft Portable Executable and Common Object File Format,
30 Specification 4.1" for more information.
32 A DLL contains an export table which contains the information
33 which the runtime loader needs to tie up references from a
36 The export table is generated by this program by reading
37 in a .DEF file or scanning the .a and .o files which will be in the
38 DLL. A .o file can contain information in special ".drectve" sections
39 with export information.
41 A DEF file contains any number of the following commands:
44 NAME <name> [ , <base> ]
45 The result is going to be <name>.EXE
47 LIBRARY <name> [ , <base> ]
48 The result is going to be <name>.DLL
50 EXPORTS ( <name1> [ = <name2> ] [ @ <integer> ] [ NONAME ] [CONSTANT] [DATA] ) *
51 Declares name1 as an exported symbol from the
52 DLL, with optional ordinal number <integer>
54 IMPORTS ( ( <internal-name> = <module-name> . <integer> )
55 | ( [ <internal-name> = ] <module-name> . <external-name> )) *
56 Declares that <external-name> or the exported function whoes ordinal number
57 is <integer> is to be imported from the file <module-name>. If
58 <internal-name> is specified then this is the name that the imported
59 function will be refered to in the body of the DLL.
62 Puts <string> into output .exp file in the .rdata section
64 [STACKSIZE|HEAPSIZE] <number-reserve> [ , <number-commit> ]
65 Generates --stack|--heap <number-reserve>,<number-commit>
66 in the output .drectve section. The linker will
67 see this and act upon it.
70 SECTIONS ( <sectionname> <attr>+ )*
71 <attr> = READ | WRITE | EXECUTE | SHARED
72 Generates --attr <sectionname> <attr> in the output
73 .drectve section. The linker will see this and act
77 A -export:<name> in a .drectve section in an input .o or .a
78 file to this program is equivalent to a EXPORTS <name>
83 The program generates output files with the prefix supplied
84 on the command line, or in the def file, or taken from the first
87 The .exp.s file contains the information necessary to export
88 the routines in the DLL. The .lib.s file contains the information
89 necessary to use the DLL's routines from a referencing program.
96 asm (".section .drectve");
97 asm (".ascii \"-export:adef\"");
101 printf ("hello from the dll %s\n", s);
106 printf ("hello from the dll and the other entry point %s\n", s);
110 asm (".section .drectve");
111 asm (".ascii \"-export:cdef\"");
112 asm (".ascii \"-export:ddef\"");
116 printf ("hello from the dll %s\n", s);
121 printf ("hello from the dll and the other entry point %s\n", s);
139 HEAPSIZE 0x40000, 0x2000
143 SECTIONS donkey READ WRITE
146 # Compile up the parts of the dll and the program
148 gcc -c file1.c file2.c themain.c
150 # Optional: put the dll objects into a library
151 # (you don't have to, you could name all the object
152 # files on the dlltool line)
154 ar qcv thedll.in file1.o file2.o
157 # Run this tool over the DLL's .def file and generate an exports
158 # file (thedll.o) and an imports file (thedll.a).
159 # (You may have to use -S to tell dlltool where to find the assembler).
161 dlltool --def thedll.def --output-exp thedll.o --output-lib thedll.a
163 # Build the dll with the library and the export table
165 ld -o thedll.dll thedll.o thedll.in
167 # Link the executable with the import library
169 gcc -o themain.exe themain.o thedll.a
171 This example can be extended if relocations are needed in the DLL:
173 # Compile up the parts of the dll and the program
175 gcc -c file1.c file2.c themain.c
177 # Run this tool over the DLL's .def file and generate an imports file.
179 dlltool --def thedll.def --output-lib thedll.lib
181 # Link the executable with the import library and generate a base file
184 gcc -o themain.exe themain.o thedll.lib -Wl,--base-file -Wl,themain.base
186 # Run this tool over the DLL's .def file and generate an exports file
187 # which includes the relocations from the base file.
189 dlltool --def thedll.def --base-file themain.base --output-exp thedll.exp
191 # Build the dll with file1.o, file2.o and the export table
193 ld -o thedll.dll thedll.exp file1.o file2.o
196 /* .idata section description
198 The .idata section is the import table. It is a collection of several
199 subsections used to keep the pieces for each dll together: .idata$[234567].
200 IE: Each dll's .idata$2's are catenated together, each .idata$3's, etc.
202 .idata$2 = Import Directory Table
203 = array of IMAGE_IMPORT_DESCRIPTOR's.
205 DWORD Import Lookup Table; - pointer to .idata$4
206 DWORD TimeDateStamp; - currently always 0
207 DWORD ForwarderChain; - currently always 0
208 DWORD Name; - pointer to dll's name
209 PIMAGE_THUNK_DATA FirstThunk; - pointer to .idata$5
211 .idata$3 = null terminating entry for .idata$2.
213 .idata$4 = Import Lookup Table
214 = array of array of pointers to hint name table.
215 There is one for each dll being imported from, and each dll's set is
216 terminated by a trailing NULL.
218 .idata$5 = Import Address Table
219 = array of array of pointers to hint name table.
220 There is one for each dll being imported from, and each dll's set is
221 terminated by a trailing NULL.
222 Initially, this table is identical to the Import Lookup Table. However,
223 at load time, the loader overwrites the entries with the address of the
226 .idata$6 = Hint Name Table
227 = Array of { short, asciz } entries, one for each imported function.
228 The `short' is the function's ordinal number.
230 .idata$7 = dll name (eg: "kernel32.dll"). (.idata$6 for ppc)
233 /* AIX requires this to be the first thing in the file. */
240 #define show_allnames 0
242 #define PAGE_SIZE 4096
243 #define PAGE_MASK (-PAGE_SIZE)
245 #include "libiberty.h"
248 #include "demangle.h"
249 #include "dyn-string.h"
254 #include <sys/stat.h>
256 #ifdef ANSI_PROTOTYPES
263 #include "coff/arm.h"
264 #include "coff/internal.h"
267 /* Forward references. */
268 static char *look_for_prog
PARAMS ((const char *, const char *, int));
269 static char *deduce_name
PARAMS ((const char *));
271 #ifdef DLLTOOL_MCORE_ELF
272 static void mcore_elf_cache_filename (char *);
273 static void mcore_elf_gen_out_file (void);
276 #ifdef HAVE_SYS_WAIT_H
277 #include <sys/wait.h>
278 #else /* ! HAVE_SYS_WAIT_H */
279 #if ! defined (_WIN32) || defined (__CYGWIN32__)
281 #define WIFEXITED(w) (((w)&0377) == 0)
284 #define WIFSIGNALED(w) (((w)&0377) != 0177 && ((w)&~0377) == 0)
287 #define WTERMSIG(w) ((w) & 0177)
290 #define WEXITSTATUS(w) (((w) >> 8) & 0377)
292 #else /* defined (_WIN32) && ! defined (__CYGWIN32__) */
294 #define WIFEXITED(w) (((w) & 0xff) == 0)
297 #define WIFSIGNALED(w) (((w) & 0xff) != 0 && ((w) & 0xff) != 0x7f)
300 #define WTERMSIG(w) ((w) & 0x7f)
303 #define WEXITSTATUS(w) (((w) & 0xff00) >> 8)
305 #endif /* defined (_WIN32) && ! defined (__CYGWIN32__) */
306 #endif /* ! HAVE_SYS_WAIT_H */
308 /* ifunc and ihead data structures: ttk@cygnus.com 1997
310 When IMPORT declarations are encountered in a .def file the
311 function import information is stored in a structure referenced by
312 the global variable IMPORT_LIST. The structure is a linked list
313 containing the names of the dll files each function is imported
314 from and a linked list of functions being imported from that dll
315 file. This roughly parallels the structure of the .idata section
316 in the PE object file.
318 The contents of .def file are interpreted from within the
319 process_def_file function. Every time an IMPORT declaration is
320 encountered, it is broken up into its component parts and passed to
321 def_import. IMPORT_LIST is initialized to NULL in function main. */
323 typedef struct ifunct
325 char *name
; /* name of function being imported */
326 int ord
; /* two-byte ordinal value associated with function */
330 typedef struct iheadt
332 char *dllname
; /* name of dll file imported from */
333 long nfuncs
; /* number of functions in list */
334 struct ifunct
*funchead
; /* first function in list */
335 struct ifunct
*functail
; /* last function in list */
336 struct iheadt
*next
; /* next dll file in list */
339 /* Structure containing all import information as defined in .def file
340 (qv "ihead structure"). */
342 static iheadtype
*import_list
= NULL
;
344 static char *as_name
= NULL
;
345 static char * as_flags
= "";
347 static int no_idata4
;
348 static int no_idata5
;
349 static char *exp_name
;
350 static char *imp_name
;
351 static char *head_label
;
352 static char *imp_name_lab
;
353 static char *dll_name
;
355 static int add_indirect
= 0;
356 static int add_underscore
= 0;
357 static int dontdeltemps
= 0;
359 /* True if we should export all symbols. Otherwise, we only export
360 symbols listed in .drectve sections or in the def file. */
361 static boolean export_all_symbols
;
363 /* True if we should exclude the symbols in DEFAULT_EXCLUDES when
364 exporting all symbols. */
365 static boolean do_default_excludes
;
367 /* Default symbols to exclude when exporting all the symbols. */
368 static const char *default_excludes
= "DllMain@12,DllEntryPoint@0,impure_ptr";
370 /* True if we should add __imp_<SYMBOL> to import libraries for backward
371 compatibility to old Cygwin releases. */
372 static boolean create_compat_implib
;
374 static char *def_file
;
376 extern char * program_name
;
380 static int add_stdcall_alias
;
382 static FILE *output_def
;
383 static FILE *base_file
;
386 #ifdef DLLTOOL_ARM_EPOC
387 static const char *mname
= "arm-epoc";
389 static const char *mname
= "arm";
394 static const char *mname
= "i386";
398 static const char *mname
= "ppc";
402 static const char *mname
= "sh";
406 static const char *mname
= "mips";
410 static const char * mname
= "mcore-le";
413 #ifdef DLLTOOL_MCORE_ELF
414 static const char * mname
= "mcore-elf";
415 static char * mcore_elf_out_file
= NULL
;
416 static char * mcore_elf_linker
= NULL
;
417 static char * mcore_elf_linker_flags
= NULL
;
419 #define DRECTVE_SECTION_NAME ((machine == MMCORE_ELF || machine == MMCORE_ELF_LE) ? ".exports" : ".drectve")
422 #ifndef DRECTVE_SECTION_NAME
423 #define DRECTVE_SECTION_NAME ".drectve"
426 #define PATHMAX 250 /* What's the right name for this ? */
428 #define TMP_ASM "dc.s"
429 #define TMP_HEAD_S "dh.s"
430 #define TMP_HEAD_O "dh.o"
431 #define TMP_TAIL_S "dt.s"
432 #define TMP_TAIL_O "dt.o"
433 #define TMP_STUB "ds"
435 /* This bit of assemly does jmp * .... */
436 static const unsigned char i386_jtab
[] =
438 0xff, 0x25, 0x00, 0x00, 0x00, 0x00, 0x90, 0x90
441 static const unsigned char arm_jtab
[] =
443 0x00, 0xc0, 0x9f, 0xe5, /* ldr ip, [pc] */
444 0x00, 0xf0, 0x9c, 0xe5, /* ldr pc, [ip] */
448 static const unsigned char arm_interwork_jtab
[] =
450 0x04, 0xc0, 0x9f, 0xe5, /* ldr ip, [pc] */
451 0x00, 0xc0, 0x9c, 0xe5, /* ldr ip, [ip] */
452 0x1c, 0xff, 0x2f, 0xe1, /* bx ip */
456 static const unsigned char thumb_jtab
[] =
458 0x40, 0xb4, /* push {r6} */
459 0x02, 0x4e, /* ldr r6, [pc, #8] */
460 0x36, 0x68, /* ldr r6, [r6] */
461 0xb4, 0x46, /* mov ip, r6 */
462 0x40, 0xbc, /* pop {r6} */
463 0x60, 0x47, /* bx ip */
467 static const unsigned char mcore_be_jtab
[] =
469 0x71, 0x02, /* lrw r1,2 */
470 0x81, 0x01, /* ld.w r1,(r1,0) */
471 0x00, 0xC1, /* jmp r1 */
472 0x12, 0x00, /* nop */
473 0x00, 0x00, 0x00, 0x00 /* <address> */
476 static const unsigned char mcore_le_jtab
[] =
478 0x02, 0x71, /* lrw r1,2 */
479 0x01, 0x81, /* ld.w r1,(r1,0) */
480 0xC1, 0x00, /* jmp r1 */
481 0x00, 0x12, /* nop */
482 0x00, 0x00, 0x00, 0x00 /* <address> */
485 /* This is the glue sequence for PowerPC PE. There is a */
486 /* tocrel16-tocdefn reloc against the first instruction. */
487 /* We also need a IMGLUE reloc against the glue function */
488 /* to restore the toc saved by the third instruction in */
490 static const unsigned char ppc_jtab
[] =
492 0x00, 0x00, 0x62, 0x81, /* lwz r11,0(r2) */
493 /* Reloc TOCREL16 __imp_xxx */
494 0x00, 0x00, 0x8B, 0x81, /* lwz r12,0(r11) */
495 0x04, 0x00, 0x41, 0x90, /* stw r2,4(r1) */
496 0xA6, 0x03, 0x89, 0x7D, /* mtctr r12 */
497 0x04, 0x00, 0x4B, 0x80, /* lwz r2,4(r11) */
498 0x20, 0x04, 0x80, 0x4E /* bctr */
502 /* the glue instruction, picks up the toc from the stw in */
503 /* the above code: "lwz r2,4(r1)" */
504 static bfd_vma ppc_glue_insn
= 0x80410004;
510 const char *how_byte
;
511 const char *how_short
;
512 const char *how_long
;
513 const char *how_asciz
;
514 const char *how_comment
;
515 const char *how_jump
;
516 const char *how_global
;
517 const char *how_space
;
518 const char *how_align_short
;
519 const char *how_align_long
;
520 const char *how_default_as_switches
;
521 const char *how_bfd_target
;
522 enum bfd_architecture how_bfd_arch
;
523 const unsigned char *how_jtab
;
524 int how_jtab_size
; /* size of the jtab entry */
525 int how_jtab_roff
; /* offset into it for the ind 32 reloc into idata 5 */
528 static const struct mac
533 "arm", ".byte", ".short", ".long", ".asciz", "@",
534 "ldr\tip,[pc]\n\tldr\tpc,[ip]\n\t.long",
535 ".global", ".space", ".align\t2",".align\t4", "-mapcs-32",
536 "pe-arm-little", bfd_arch_arm
,
537 arm_jtab
, sizeof (arm_jtab
), 8
542 "i386", ".byte", ".short", ".long", ".asciz", "#",
543 "jmp *", ".global", ".space", ".align\t2",".align\t4", "",
544 "pe-i386",bfd_arch_i386
,
545 i386_jtab
, sizeof (i386_jtab
), 2
550 "ppc", ".byte", ".short", ".long", ".asciz", "#",
551 "jmp *", ".global", ".space", ".align\t2",".align\t4", "",
552 "pe-powerpcle",bfd_arch_powerpc
,
553 ppc_jtab
, sizeof (ppc_jtab
), 0
558 "thumb", ".byte", ".short", ".long", ".asciz", "@",
559 "push\t{r6}\n\tldr\tr6, [pc, #8]\n\tldr\tr6, [r6]\n\tmov\tip, r6\n\tpop\t{r6}\n\tbx\tip",
560 ".global", ".space", ".align\t2",".align\t4", "-mthumb-interwork",
561 "pe-arm-little", bfd_arch_arm
,
562 thumb_jtab
, sizeof (thumb_jtab
), 12
565 #define MARM_INTERWORK 4
567 "arm_interwork", ".byte", ".short", ".long", ".asciz", "@",
568 "ldr\tip,[pc]\n\tldr\tip,[ip]\n\tbx\tip\n\t.long",
569 ".global", ".space", ".align\t2",".align\t4", "-mthumb-interwork",
570 "pe-arm-little", bfd_arch_arm
,
571 arm_interwork_jtab
, sizeof (arm_interwork_jtab
), 12
576 "mcore-be", ".byte", ".short", ".long", ".asciz", "//",
577 "lrw r1,[1f]\n\tld.w r1,(r1,0)\n\tjmp r1\n\tnop\n1:.long",
578 ".global", ".space", ".align\t2",".align\t4", "",
579 "pe-mcore-big", bfd_arch_mcore
,
580 mcore_be_jtab
, sizeof (mcore_be_jtab
), 8
585 "mcore-le", ".byte", ".short", ".long", ".asciz", "//",
586 "lrw r1,[1f]\n\tld.w r1,(r1,0)\n\tjmp r1\n\tnop\n1:.long",
587 ".global", ".space", ".align\t2",".align\t4", "-EL",
588 "pe-mcore-little", bfd_arch_mcore
,
589 mcore_le_jtab
, sizeof (mcore_le_jtab
), 8
594 "mcore-elf-be", ".byte", ".short", ".long", ".asciz", "//",
595 "lrw r1,[1f]\n\tld.w r1,(r1,0)\n\tjmp r1\n\tnop\n1:.long",
596 ".global", ".space", ".align\t2",".align\t4", "",
597 "elf32-mcore-big", bfd_arch_mcore
,
598 mcore_be_jtab
, sizeof (mcore_be_jtab
), 8
602 #define MMCORE_ELF_LE 8
603 "mcore-elf-le", ".byte", ".short", ".long", ".asciz", "//",
604 "lrw r1,[1f]\n\tld.w r1,(r1,0)\n\tjmp r1\n\tnop\n1:.long",
605 ".global", ".space", ".align\t2",".align\t4", "-EL",
606 "elf32-mcore-little", bfd_arch_mcore
,
607 mcore_le_jtab
, sizeof (mcore_le_jtab
), 8
612 "arm-epoc", ".byte", ".short", ".long", ".asciz", "@",
613 "ldr\tip,[pc]\n\tldr\tpc,[ip]\n\t.long",
614 ".global", ".space", ".align\t2",".align\t4", "",
615 "epoc-pe-arm-little", bfd_arch_arm
,
616 arm_jtab
, sizeof (arm_jtab
), 8
619 { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }
629 typedef struct export
632 const char *internal_name
;
642 /* A list of symbols which we should not export. */
646 struct string_list
*next
;
650 static struct string_list
*excludes
;
652 static const char *rvaafter
PARAMS ((int));
653 static const char *rvabefore
PARAMS ((int));
654 static const char *asm_prefix
PARAMS ((int));
655 static void append_import
PARAMS ((const char *, const char *, int));
656 static void run
PARAMS ((const char *, char *));
657 static void scan_drectve_symbols
PARAMS ((bfd
*));
658 static void scan_filtered_symbols
PARAMS ((bfd
*, PTR
, long, unsigned int));
659 static void add_excludes
PARAMS ((const char *));
660 static boolean match_exclude
PARAMS ((const char *));
661 static void set_default_excludes
PARAMS ((void));
662 static long filter_symbols
PARAMS ((bfd
*, PTR
, long, unsigned int));
663 static void scan_all_symbols
PARAMS ((bfd
*));
664 static void scan_open_obj_file
PARAMS ((bfd
*));
665 static void scan_obj_file
PARAMS ((const char *));
666 static void dump_def_info
PARAMS ((FILE *));
667 static int sfunc
PARAMS ((const void *, const void *));
668 static void flush_page
PARAMS ((FILE *, long *, int, int));
669 static void gen_def_file
PARAMS ((void));
670 static void generate_idata_ofile
PARAMS ((FILE *));
671 static void gen_exp_file
PARAMS ((void));
672 static const char *xlate
PARAMS ((const char *));
674 static void dump_iat
PARAMS ((FILE *, export_type
*));
676 static char *make_label
PARAMS ((const char *, const char *));
677 static bfd
*make_one_lib_file
PARAMS ((export_type
*, int));
678 static bfd
*make_head
PARAMS ((void));
679 static bfd
*make_tail
PARAMS ((void));
680 static void gen_lib_file
PARAMS ((void));
681 static int pfunc
PARAMS ((const void *, const void *));
682 static int nfunc
PARAMS ((const void *, const void *));
683 static void remove_null_names
PARAMS ((export_type
**));
684 static void dtab
PARAMS ((export_type
**));
685 static void process_duplicates
PARAMS ((export_type
**));
686 static void fill_ordinals
PARAMS ((export_type
**));
687 static int alphafunc
PARAMS ((const void *, const void *));
688 static void mangle_defs
PARAMS ((void));
689 static void usage
PARAMS ((FILE *, int));
690 static void inform
PARAMS ((const char *, ...));
695 inform (const char * message
, ...)
697 inform (message
, va_alist
)
698 const char * message
;
708 va_start (args
, message
);
713 report (message
, args
);
736 /* xgettext:c-format */
737 fatal (_("Internal error: Unknown machine type: %d"), machine
);
761 /* xgettext:c-format */
762 fatal (_("Internal error: Unknown machine type: %d"), machine
);
787 /* xgettext:c-format */
788 fatal (_("Internal error: Unknown machine type: %d"), machine
);
794 #define ASM_BYTE mtable[machine].how_byte
795 #define ASM_SHORT mtable[machine].how_short
796 #define ASM_LONG mtable[machine].how_long
797 #define ASM_TEXT mtable[machine].how_asciz
798 #define ASM_C mtable[machine].how_comment
799 #define ASM_JUMP mtable[machine].how_jump
800 #define ASM_GLOBAL mtable[machine].how_global
801 #define ASM_SPACE mtable[machine].how_space
802 #define ASM_ALIGN_SHORT mtable[machine].how_align_short
803 #define ASM_RVA_BEFORE rvabefore(machine)
804 #define ASM_RVA_AFTER rvaafter(machine)
805 #define ASM_PREFIX asm_prefix(machine)
806 #define ASM_ALIGN_LONG mtable[machine].how_align_long
807 #define HOW_BFD_READ_TARGET 0 /* always default*/
808 #define HOW_BFD_WRITE_TARGET mtable[machine].how_bfd_target
809 #define HOW_BFD_ARCH mtable[machine].how_bfd_arch
810 #define HOW_JTAB mtable[machine].how_jtab
811 #define HOW_JTAB_SIZE mtable[machine].how_jtab_size
812 #define HOW_JTAB_ROFF mtable[machine].how_jtab_roff
813 #define ASM_SWITCHES mtable[machine].how_default_as_switches
818 process_def_file (name
)
821 FILE *f
= fopen (name
, FOPEN_RT
);
824 /* xgettext:c-format */
825 fatal (_("Can't open def file: %s"), name
);
829 /* xgettext:c-format */
830 inform (_("Processing def file: %s"), name
);
834 inform (_("Processed def file"));
837 /**********************************************************************/
839 /* Communications with the parser */
841 static const char *d_name
; /* Arg to NAME or LIBRARY */
842 static int d_nfuncs
; /* Number of functions exported */
843 static int d_named_nfuncs
; /* Number of named functions exported */
844 static int d_low_ord
; /* Lowest ordinal index */
845 static int d_high_ord
; /* Highest ordinal index */
846 static export_type
*d_exports
; /*list of exported functions */
847 static export_type
**d_exports_lexically
; /* vector of exported functions in alpha order */
848 static dlist_type
*d_list
; /* Descriptions */
849 static dlist_type
*a_list
; /* Stuff to go in directives */
856 const char * err ATTRIBUTE_UNUSED
;
858 /* xgettext:c-format */
859 non_fatal (_("Syntax error in def file %s:%d"), def_file
, linenumber
);
865 def_exports (name
, internal_name
, ordinal
, noname
, constant
, data
)
867 const char *internal_name
;
873 struct export
*p
= (struct export
*) xmalloc (sizeof (*p
));
876 p
->internal_name
= internal_name
? internal_name
: name
;
877 p
->ordinal
= ordinal
;
878 p
->constant
= constant
;
887 def_name (name
, base
)
891 /* xgettext:c-format */
892 inform (_("NAME: %s base: %x"), name
, base
);
895 non_fatal (_("Can't have LIBRARY and NAME"));
898 /* if --dllname not provided, use the one in the DEF file.
899 FIXME: Is this appropriate for executables? */
901 dll_name
= xstrdup (name
);
906 def_library (name
, base
)
910 /* xgettext:c-format */
911 inform (_("LIBRARY: %s base: %x"), name
, base
);
914 non_fatal (_("Can't have LIBRARY and NAME"));
917 /* if --dllname not provided, use the one in the DEF file. */
919 dll_name
= xstrdup (name
);
924 def_description (desc
)
927 dlist_type
*d
= (dlist_type
*) xmalloc (sizeof (dlist_type
));
928 d
->text
= xstrdup (desc
);
937 dlist_type
*d
= (dlist_type
*) xmalloc (sizeof (dlist_type
));
938 d
->text
= xstrdup (dir
);
944 def_heapsize (reserve
, commit
)
950 sprintf (b
, "-heap 0x%x,0x%x ", reserve
, commit
);
952 sprintf (b
, "-heap 0x%x ", reserve
);
953 new_directive (xstrdup (b
));
957 def_stacksize (reserve
, commit
)
963 sprintf (b
, "-stack 0x%x,0x%x ", reserve
, commit
);
965 sprintf (b
, "-stack 0x%x ", reserve
);
966 new_directive (xstrdup (b
));
969 /* append_import simply adds the given import definition to the global
970 import_list. It is used by def_import. */
973 append_import (symbol_name
, dll_name
, func_ordinal
)
974 const char *symbol_name
;
975 const char *dll_name
;
981 for (pq
= &import_list
; *pq
!= NULL
; pq
= &(*pq
)->next
)
983 if (strcmp ((*pq
)->dllname
, dll_name
) == 0)
986 q
->functail
->next
= xmalloc (sizeof (ifunctype
));
987 q
->functail
= q
->functail
->next
;
988 q
->functail
->ord
= func_ordinal
;
989 q
->functail
->name
= xstrdup (symbol_name
);
990 q
->functail
->next
= NULL
;
996 q
= xmalloc (sizeof (iheadtype
));
997 q
->dllname
= xstrdup (dll_name
);
999 q
->funchead
= xmalloc (sizeof (ifunctype
));
1000 q
->functail
= q
->funchead
;
1002 q
->functail
->name
= xstrdup (symbol_name
);
1003 q
->functail
->ord
= func_ordinal
;
1004 q
->functail
->next
= NULL
;
1009 /* def_import is called from within defparse.y when an IMPORT
1010 declaration is encountered. Depending on the form of the
1011 declaration, the module name may or may not need ".dll" to be
1012 appended to it, the name of the function may be stored in internal
1013 or entry, and there may or may not be an ordinal value associated
1016 /* A note regarding the parse modes:
1017 In defparse.y we have to accept import declarations which follow
1018 any one of the following forms:
1019 <func_name_in_app> = <dll_name>.<func_name_in_dll>
1020 <func_name_in_app> = <dll_name>.<number>
1021 <dll_name>.<func_name_in_dll>
1023 Furthermore, the dll's name may or may not end with ".dll", which
1024 complicates the parsing a little. Normally the dll's name is
1025 passed to def_import() in the "module" parameter, but when it ends
1026 with ".dll" it gets passed in "module" sans ".dll" and that needs
1029 def_import gets five parameters:
1030 APP_NAME - the name of the function in the application, if
1031 present, or NULL if not present.
1032 MODULE - the name of the dll, possibly sans extension (ie, '.dll').
1033 DLLEXT - the extension of the dll, if present, NULL if not present.
1034 ENTRY - the name of the function in the dll, if present, or NULL.
1035 ORD_VAL - the numerical tag of the function in the dll, if present,
1036 or NULL. Exactly one of <entry> or <ord_val> must be
1037 present (i.e., not NULL). */
1040 def_import (app_name
, module
, dllext
, entry
, ord_val
)
1041 const char *app_name
;
1047 const char *application_name
;
1051 application_name
= entry
;
1054 if (app_name
!= NULL
)
1055 application_name
= app_name
;
1057 application_name
= "";
1062 buf
= (char *) alloca (strlen (module
) + strlen (dllext
) + 2);
1063 sprintf (buf
, "%s.%s", module
, dllext
);
1067 append_import (application_name
, module
, ord_val
);
1071 def_version (major
, minor
)
1075 printf ("VERSION %d.%d\n", major
, minor
);
1079 def_section (name
, attr
)
1096 sprintf (buf
, "-attr %s %s", name
, atts
);
1097 new_directive (xstrdup (buf
));
1105 def_section ("CODE", attr
);
1112 def_section ("DATA", attr
);
1115 /**********************************************************************/
1123 int pid
, wait_status
;
1126 char *errmsg_fmt
, *errmsg_arg
;
1127 char *temp_base
= choose_temp_base ();
1129 inform ("run: %s %s", what
, args
);
1131 /* Count the args */
1133 for (s
= args
; *s
; s
++)
1137 argv
= alloca (sizeof (char *) * (i
+ 3));
1146 while (*s
!= ' ' && *s
!= 0)
1154 pid
= pexecute (argv
[0], (char * const *) argv
, program_name
, temp_base
,
1155 &errmsg_fmt
, &errmsg_arg
, PEXECUTE_ONE
| PEXECUTE_SEARCH
);
1159 inform (strerror (errno
));
1161 fatal (errmsg_fmt
, errmsg_arg
);
1164 pid
= pwait (pid
, & wait_status
, 0);
1168 /* xgettext:c-format */
1169 fatal (_("wait: %s"), strerror (errno
));
1171 else if (WIFSIGNALED (wait_status
))
1173 /* xgettext:c-format */
1174 fatal (_("subprocess got fatal signal %d"), WTERMSIG (wait_status
));
1176 else if (WIFEXITED (wait_status
))
1178 if (WEXITSTATUS (wait_status
) != 0)
1179 /* xgettext:c-format */
1180 non_fatal (_("%s exited with status %d"),
1181 what
, WEXITSTATUS (wait_status
));
1187 /* Look for a list of symbols to export in the .drectve section of
1188 ABFD. Pass each one to def_exports. */
1191 scan_drectve_symbols (abfd
)
1200 /* Look for .drectve's */
1201 s
= bfd_get_section_by_name (abfd
, DRECTVE_SECTION_NAME
);
1206 size
= bfd_get_section_size_before_reloc (s
);
1207 buf
= xmalloc (size
);
1209 bfd_get_section_contents (abfd
, s
, buf
, 0, size
);
1211 /* xgettext:c-format */
1212 inform (_("Sucking in info from %s section in %s"),
1213 DRECTVE_SECTION_NAME
, bfd_get_filename (abfd
));
1215 /* Search for -export: strings. The exported symbols can optionally
1216 have type tags (eg., -export:foo,data), so handle those as well.
1217 Currently only data tag is supported. */
1223 && strncmp (p
, "-export:", 8) == 0)
1227 flagword flags
= BSF_FUNCTION
;
1231 while (p
< e
&& *p
!= ',' && *p
!= ' ' && *p
!= '-')
1233 c
= xmalloc (p
- name
+ 1);
1234 memcpy (c
, name
, p
- name
);
1236 if (p
< e
&& *p
== ',') /* found type tag. */
1238 char *tag_start
= ++p
;
1239 while (p
< e
&& *p
!= ' ' && *p
!= '-')
1241 if (strncmp (tag_start
, "data", 4) == 0)
1242 flags
&= ~BSF_FUNCTION
;
1245 /* FIXME: The 5th arg is for the `constant' field.
1246 What should it be? Not that it matters since it's not
1247 currently useful. */
1248 def_exports (c
, 0, -1, 0, 0, ! (flags
& BSF_FUNCTION
));
1250 if (add_stdcall_alias
&& strchr (c
, '@'))
1252 char *exported_name
= xstrdup (c
);
1253 char *atsym
= strchr (exported_name
, '@');
1255 /* Note: stdcall alias symbols can never be data. */
1256 def_exports (exported_name
, xstrdup (c
), -1, 0, 0, 0);
1265 /* Look through the symbols in MINISYMS, and add each one to list of
1266 symbols to export. */
1269 scan_filtered_symbols (abfd
, minisyms
, symcount
, size
)
1276 bfd_byte
*from
, *fromend
;
1278 store
= bfd_make_empty_symbol (abfd
);
1280 bfd_fatal (bfd_get_filename (abfd
));
1282 from
= (bfd_byte
*) minisyms
;
1283 fromend
= from
+ symcount
* size
;
1284 for (; from
< fromend
; from
+= size
)
1287 const char *symbol_name
;
1289 sym
= bfd_minisymbol_to_symbol (abfd
, false, from
, store
);
1291 bfd_fatal (bfd_get_filename (abfd
));
1293 symbol_name
= bfd_asymbol_name (sym
);
1294 if (bfd_get_symbol_leading_char (abfd
) == symbol_name
[0])
1297 def_exports (xstrdup (symbol_name
) , 0, -1, 0, 0,
1298 ! (sym
->flags
& BSF_FUNCTION
));
1300 if (add_stdcall_alias
&& strchr (symbol_name
, '@'))
1302 char *exported_name
= xstrdup (symbol_name
);
1303 char *atsym
= strchr (exported_name
, '@');
1305 /* Note: stdcall alias symbols can never be data. */
1306 def_exports (exported_name
, xstrdup (symbol_name
), -1, 0, 0, 0);
1311 /* Add a list of symbols to exclude. */
1314 add_excludes (new_excludes
)
1315 const char *new_excludes
;
1318 char *exclude_string
;
1320 local_copy
= xstrdup (new_excludes
);
1322 exclude_string
= strtok (local_copy
, ",:");
1323 for (; exclude_string
; exclude_string
= strtok (NULL
, ",:"))
1325 struct string_list
*new_exclude
;
1327 new_exclude
= ((struct string_list
*)
1328 xmalloc (sizeof (struct string_list
)));
1329 new_exclude
->string
= (char *) xmalloc (strlen (exclude_string
) + 2);
1330 /* FIXME: Is it always right to add a leading underscore? */
1331 sprintf (new_exclude
->string
, "_%s", exclude_string
);
1332 new_exclude
->next
= excludes
;
1333 excludes
= new_exclude
;
1335 /* xgettext:c-format */
1336 inform (_("Excluding symbol: %s"), exclude_string
);
1342 /* See if STRING is on the list of symbols to exclude. */
1345 match_exclude (string
)
1348 struct string_list
*excl_item
;
1350 for (excl_item
= excludes
; excl_item
; excl_item
= excl_item
->next
)
1351 if (strcmp (string
, excl_item
->string
) == 0)
1356 /* Add the default list of symbols to exclude. */
1359 set_default_excludes (void)
1361 add_excludes (default_excludes
);
1364 /* Choose which symbols to export. */
1367 filter_symbols (abfd
, minisyms
, symcount
, size
)
1373 bfd_byte
*from
, *fromend
, *to
;
1376 store
= bfd_make_empty_symbol (abfd
);
1378 bfd_fatal (bfd_get_filename (abfd
));
1380 from
= (bfd_byte
*) minisyms
;
1381 fromend
= from
+ symcount
* size
;
1382 to
= (bfd_byte
*) minisyms
;
1384 for (; from
< fromend
; from
+= size
)
1389 sym
= bfd_minisymbol_to_symbol (abfd
, false, (const PTR
) from
, store
);
1391 bfd_fatal (bfd_get_filename (abfd
));
1393 /* Check for external and defined only symbols. */
1394 keep
= (((sym
->flags
& BSF_GLOBAL
) != 0
1395 || (sym
->flags
& BSF_WEAK
) != 0
1396 || bfd_is_com_section (sym
->section
))
1397 && ! bfd_is_und_section (sym
->section
));
1399 keep
= keep
&& ! match_exclude (sym
->name
);
1403 memcpy (to
, from
, size
);
1408 return (to
- (bfd_byte
*) minisyms
) / size
;
1411 /* Export all symbols in ABFD, except for ones we were told not to
1415 scan_all_symbols (abfd
)
1422 /* Ignore bfds with an import descriptor table. We assume that any
1423 such BFD contains symbols which are exported from another DLL,
1424 and we don't want to reexport them from here. */
1425 if (bfd_get_section_by_name (abfd
, ".idata$4"))
1428 if (! (bfd_get_file_flags (abfd
) & HAS_SYMS
))
1430 /* xgettext:c-format */
1431 non_fatal (_("%s: no symbols"), bfd_get_filename (abfd
));
1435 symcount
= bfd_read_minisymbols (abfd
, false, &minisyms
, &size
);
1437 bfd_fatal (bfd_get_filename (abfd
));
1441 /* xgettext:c-format */
1442 non_fatal (_("%s: no symbols"), bfd_get_filename (abfd
));
1446 /* Discard the symbols we don't want to export. It's OK to do this
1447 in place; we'll free the storage anyway. */
1449 symcount
= filter_symbols (abfd
, minisyms
, symcount
, size
);
1450 scan_filtered_symbols (abfd
, minisyms
, symcount
, size
);
1455 /* Look at the object file to decide which symbols to export. */
1458 scan_open_obj_file (abfd
)
1461 if (export_all_symbols
)
1462 scan_all_symbols (abfd
);
1464 scan_drectve_symbols (abfd
);
1466 /* FIXME: we ought to read in and block out the base relocations */
1468 /* xgettext:c-format */
1469 inform (_("Done reading %s"), bfd_get_filename (abfd
));
1473 scan_obj_file (filename
)
1474 const char *filename
;
1476 bfd
* f
= bfd_openr (filename
, 0);
1479 /* xgettext:c-format */
1480 fatal (_("Unable to open object file: %s"), filename
);
1482 /* xgettext:c-format */
1483 inform (_("Scanning object file %s"), filename
);
1485 if (bfd_check_format (f
, bfd_archive
))
1487 bfd
*arfile
= bfd_openr_next_archived_file (f
, 0);
1490 if (bfd_check_format (arfile
, bfd_object
))
1491 scan_open_obj_file (arfile
);
1493 arfile
= bfd_openr_next_archived_file (f
, arfile
);
1496 #ifdef DLLTOOL_MCORE_ELF
1497 if (mcore_elf_out_file
)
1498 inform (_("Cannot produce mcore-elf dll from archive file: %s"), filename
);
1501 else if (bfd_check_format (f
, bfd_object
))
1503 scan_open_obj_file (f
);
1505 #ifdef DLLTOOL_MCORE_ELF
1506 if (mcore_elf_out_file
)
1507 mcore_elf_cache_filename ((char *) filename
);
1514 /**********************************************************************/
1522 fprintf (f
, "%s ", ASM_C
);
1523 for (i
= 0; oav
[i
]; i
++)
1524 fprintf (f
, "%s ", oav
[i
]);
1526 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
1528 fprintf (f
, "%s %d = %s %s @ %d %s%s%s\n",
1534 exp
->noname
? "NONAME " : "",
1535 exp
->constant
? "CONSTANT" : "",
1536 exp
->data
? "DATA" : "");
1540 /* Generate the .exp file */
1547 return *(const long *) a
- *(const long *) b
;
1551 flush_page (f
, need
, page_addr
, on_page
)
1559 /* Flush this page */
1560 fprintf (f
, "\t%s\t0x%08x\t%s Starting RVA for chunk\n",
1564 fprintf (f
, "\t%s\t0x%x\t%s Size of block\n",
1566 (on_page
* 2) + (on_page
& 1) * 2 + 8,
1569 for (i
= 0; i
< on_page
; i
++)
1571 long needed
= need
[i
];
1574 needed
= ((needed
- page_addr
) | 0x3000) & 0xffff;
1576 fprintf (f
, "\t%s\t0x%lx\n", ASM_SHORT
, needed
);
1581 fprintf (f
, "\t%s\t0x%x\n", ASM_SHORT
, 0 | 0x0000);
1590 inform (_("Adding exports to output file"));
1592 fprintf (output_def
, ";");
1593 for (i
= 0; oav
[i
]; i
++)
1594 fprintf (output_def
, " %s", oav
[i
]);
1596 fprintf (output_def
, "\nEXPORTS\n");
1598 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
1600 char *quote
= strchr (exp
->name
, '.') ? "\"" : "";
1601 char *res
= cplus_demangle (exp
->internal_name
, DMGL_ANSI
| DMGL_PARAMS
);
1603 if (strcmp (exp
->name
, exp
->internal_name
) == 0)
1606 fprintf (output_def
, "\t%s%s%s @ %d%s%s ; %s\n",
1611 exp
->noname
? " NONAME" : "",
1612 exp
->data
? " DATA" : "",
1617 char *quote1
= strchr (exp
->internal_name
, '.') ? "\"" : "";
1619 fprintf (output_def
, "\t%s%s%s = %s%s%s @ %d%s%s ; %s\n",
1627 exp
->noname
? " NONAME" : "",
1628 exp
->data
? " DATA" : "",
1635 inform (_("Added exports to output file"));
1638 /* generate_idata_ofile generates the portable assembly source code
1639 for the idata sections. It appends the source code to the end of
1643 generate_idata_ofile (filvar
)
1652 if (import_list
== NULL
)
1655 fprintf (filvar
, "%s Import data sections\n", ASM_C
);
1656 fprintf (filvar
, "\n\t.section\t.idata$2\n");
1657 fprintf (filvar
, "\t%s\tdoi_idata\n", ASM_GLOBAL
);
1658 fprintf (filvar
, "doi_idata:\n");
1661 for (headptr
= import_list
; headptr
!= NULL
; headptr
= headptr
->next
)
1663 fprintf (filvar
, "\t%slistone%d%s\t%s %s\n",
1664 ASM_RVA_BEFORE
, nheads
, ASM_RVA_AFTER
,
1665 ASM_C
, headptr
->dllname
);
1666 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
);
1667 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
);
1668 fprintf (filvar
, "\t%sdllname%d%s\n",
1669 ASM_RVA_BEFORE
, nheads
, ASM_RVA_AFTER
);
1670 fprintf (filvar
, "\t%slisttwo%d%s\n\n",
1671 ASM_RVA_BEFORE
, nheads
, ASM_RVA_AFTER
);
1675 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
); /* NULL record at */
1676 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
); /* end of idata$2 */
1677 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
); /* section */
1678 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
);
1679 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
);
1681 fprintf (filvar
, "\n\t.section\t.idata$4\n");
1683 for (headptr
= import_list
; headptr
!= NULL
; headptr
= headptr
->next
)
1685 fprintf (filvar
, "listone%d:\n", headindex
);
1686 for ( funcindex
= 0; funcindex
< headptr
->nfuncs
; funcindex
++ )
1687 fprintf (filvar
, "\t%sfuncptr%d_%d%s\n",
1688 ASM_RVA_BEFORE
, headindex
, funcindex
, ASM_RVA_AFTER
);
1689 fprintf (filvar
,"\t%s\t0\n", ASM_LONG
); /* NULL terminating list */
1693 fprintf (filvar
, "\n\t.section\t.idata$5\n");
1695 for (headptr
= import_list
; headptr
!= NULL
; headptr
= headptr
->next
)
1697 fprintf (filvar
, "listtwo%d:\n", headindex
);
1698 for ( funcindex
= 0; funcindex
< headptr
->nfuncs
; funcindex
++ )
1699 fprintf (filvar
, "\t%sfuncptr%d_%d%s\n",
1700 ASM_RVA_BEFORE
, headindex
, funcindex
, ASM_RVA_AFTER
);
1701 fprintf (filvar
, "\t%s\t0\n", ASM_LONG
); /* NULL terminating list */
1705 fprintf (filvar
, "\n\t.section\t.idata$6\n");
1707 for (headptr
= import_list
; headptr
!= NULL
; headptr
= headptr
->next
)
1710 for (funcptr
= headptr
->funchead
; funcptr
!= NULL
;
1711 funcptr
= funcptr
->next
)
1713 fprintf (filvar
,"funcptr%d_%d:\n", headindex
, funcindex
);
1714 fprintf (filvar
,"\t%s\t%d\n", ASM_SHORT
,
1715 ((funcptr
->ord
) & 0xFFFF));
1716 fprintf (filvar
,"\t%s\t\"%s\"\n", ASM_TEXT
, funcptr
->name
);
1717 fprintf (filvar
,"\t%s\t0\n", ASM_BYTE
);
1723 fprintf (filvar
, "\n\t.section\t.idata$7\n");
1725 for (headptr
= import_list
; headptr
!= NULL
; headptr
= headptr
->next
)
1727 fprintf (filvar
,"dllname%d:\n", headindex
);
1728 fprintf (filvar
,"\t%s\t\"%s\"\n", ASM_TEXT
, headptr
->dllname
);
1729 fprintf (filvar
,"\t%s\t0\n", ASM_BYTE
);
1734 /* Assemble the specified file. */
1736 assemble_file (source
, dest
)
1737 const char * source
;
1742 cmd
= (char *) alloca (strlen (ASM_SWITCHES
) + strlen (as_flags
)
1743 + strlen (source
) + strlen (dest
) + 50);
1745 sprintf (cmd
, "%s %s -o %s %s", ASM_SWITCHES
, as_flags
, dest
, source
);
1758 /* xgettext:c-format */
1759 inform (_("Generating export file: %s"), exp_name
);
1761 f
= fopen (TMP_ASM
, FOPEN_WT
);
1763 /* xgettext:c-format */
1764 fatal (_("Unable to open temporary assembler file: %s"), TMP_ASM
);
1766 /* xgettext:c-format */
1767 inform (_("Opened temporary file: %s"), TMP_ASM
);
1773 fprintf (f
, "\t.section .edata\n\n");
1774 fprintf (f
, "\t%s 0 %s Allways 0\n", ASM_LONG
, ASM_C
);
1775 fprintf (f
, "\t%s 0x%lx %s Time and date\n", ASM_LONG
, (long) time(0),
1777 fprintf (f
, "\t%s 0 %s Major and Minor version\n", ASM_LONG
, ASM_C
);
1778 fprintf (f
, "\t%sname%s %s Ptr to name of dll\n", ASM_RVA_BEFORE
, ASM_RVA_AFTER
, ASM_C
);
1779 fprintf (f
, "\t%s %d %s Starting ordinal of exports\n", ASM_LONG
, d_low_ord
, ASM_C
);
1782 fprintf (f
, "\t%s %d %s Number of functions\n", ASM_LONG
, d_high_ord
- d_low_ord
+ 1, ASM_C
);
1783 fprintf(f
,"\t%s named funcs %d, low ord %d, high ord %d\n",
1785 d_named_nfuncs
, d_low_ord
, d_high_ord
);
1786 fprintf (f
, "\t%s %d %s Number of names\n", ASM_LONG
,
1787 show_allnames
? d_high_ord
- d_low_ord
+ 1 : d_named_nfuncs
, ASM_C
);
1788 fprintf (f
, "\t%safuncs%s %s Address of functions\n", ASM_RVA_BEFORE
, ASM_RVA_AFTER
, ASM_C
);
1790 fprintf (f
, "\t%sanames%s %s Address of Name Pointer Table\n",
1791 ASM_RVA_BEFORE
, ASM_RVA_AFTER
, ASM_C
);
1793 fprintf (f
, "\t%sanords%s %s Address of ordinals\n", ASM_RVA_BEFORE
, ASM_RVA_AFTER
, ASM_C
);
1795 fprintf (f
, "name: %s \"%s\"\n", ASM_TEXT
, dll_name
);
1798 fprintf(f
,"%s Export address Table\n", ASM_C
);
1799 fprintf(f
,"\t%s\n", ASM_ALIGN_LONG
);
1800 fprintf (f
, "afuncs:\n");
1803 for (exp
= d_exports
; exp
; exp
= exp
->next
)
1805 if (exp
->ordinal
!= i
)
1808 fprintf (f
, "\t%s\t%d\t%s %d..%d missing\n",
1810 (exp
->ordinal
- i
) * 4,
1812 i
, exp
->ordinal
- 1);
1815 while (i
< exp
->ordinal
)
1817 fprintf(f
,"\t%s\t0\n", ASM_LONG
);
1821 fprintf (f
, "\t%s%s%s%s\t%s %d\n", ASM_RVA_BEFORE
,
1823 exp
->internal_name
, ASM_RVA_AFTER
, ASM_C
, exp
->ordinal
);
1827 fprintf (f
,"%s Export Name Pointer Table\n", ASM_C
);
1828 fprintf (f
, "anames:\n");
1830 for (i
= 0; (exp
= d_exports_lexically
[i
]); i
++)
1832 if (!exp
->noname
|| show_allnames
)
1833 fprintf (f
, "\t%sn%d%s\n",
1834 ASM_RVA_BEFORE
, exp
->ordinal
, ASM_RVA_AFTER
);
1837 fprintf (f
,"%s Export Oridinal Table\n", ASM_C
);
1838 fprintf (f
, "anords:\n");
1839 for (i
= 0; (exp
= d_exports_lexically
[i
]); i
++)
1841 if (!exp
->noname
|| show_allnames
)
1842 fprintf (f
, "\t%s %d\n", ASM_SHORT
, exp
->ordinal
- d_low_ord
);
1845 fprintf(f
,"%s Export Name Table\n", ASM_C
);
1846 for (i
= 0; (exp
= d_exports_lexically
[i
]); i
++)
1847 if (!exp
->noname
|| show_allnames
)
1848 fprintf (f
, "n%d: %s \"%s\"\n",
1849 exp
->ordinal
, ASM_TEXT
, exp
->name
);
1853 fprintf (f
, "\t.section %s\n", DRECTVE_SECTION_NAME
);
1854 for (dl
= a_list
; dl
; dl
= dl
->next
)
1856 fprintf (f
, "\t%s\t\"%s\"\n", ASM_TEXT
, dl
->text
);
1862 fprintf (f
, "\t.section .rdata\n");
1863 for (dl
= d_list
; dl
; dl
= dl
->next
)
1868 /* We don't output as ascii because there can
1869 be quote characters in the string. */
1871 for (p
= dl
->text
; *p
; p
++)
1874 fprintf (f
, "\t%s\t", ASM_BYTE
);
1877 fprintf (f
, "%d", *p
);
1880 fprintf (f
, ",0\n");
1894 /* Add to the output file a way of getting to the exported names
1895 without using the import library. */
1898 fprintf (f
, "\t.section\t.rdata\n");
1899 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
1900 if (!exp
->noname
|| show_allnames
)
1902 /* We use a single underscore for MS compatibility, and a
1903 double underscore for backward compatibility with old
1905 if (create_compat_implib
)
1906 fprintf (f
, "\t%s\t__imp_%s\n", ASM_GLOBAL
, exp
->name
);
1907 fprintf (f
, "\t%s\t_imp__%s\n", ASM_GLOBAL
, exp
->name
);
1908 if (create_compat_implib
)
1909 fprintf (f
, "__imp_%s:\n", exp
->name
);
1910 fprintf (f
, "_imp__%s:\n", exp
->name
);
1911 fprintf (f
, "\t%s\t%s\n", ASM_LONG
, exp
->name
);
1915 /* Dump the reloc section if a base file is provided */
1919 long need
[PAGE_SIZE
];
1926 fprintf (f
, "\t.section\t.init\n");
1927 fprintf (f
, "lab:\n");
1929 fseek (base_file
, 0, SEEK_END
);
1930 numbytes
= ftell (base_file
);
1931 fseek (base_file
, 0, SEEK_SET
);
1932 copy
= xmalloc (numbytes
);
1933 fread (copy
, 1, numbytes
, base_file
);
1934 num_entries
= numbytes
/ sizeof (long);
1937 fprintf (f
, "\t.section\t.reloc\n");
1943 qsort (copy
, num_entries
, sizeof (long), sfunc
);
1944 /* Delete duplcates */
1945 for (src
= 0; src
< num_entries
; src
++)
1947 if (last
!= copy
[src
])
1948 last
= copy
[dst
++] = copy
[src
];
1952 page_addr
= addr
& PAGE_MASK
; /* work out the page addr */
1954 for (j
= 0; j
< num_entries
; j
++)
1957 if ((addr
& PAGE_MASK
) != page_addr
)
1959 flush_page (f
, need
, page_addr
, on_page
);
1961 page_addr
= addr
& PAGE_MASK
;
1963 need
[on_page
++] = addr
;
1965 flush_page (f
, need
, page_addr
, on_page
);
1967 /* fprintf (f, "\t%s\t0,0\t%s End\n", ASM_LONG, ASM_C);*/
1971 generate_idata_ofile (f
);
1975 /* assemble the file */
1976 assemble_file (TMP_ASM
, exp_name
);
1978 if (dontdeltemps
== 0)
1981 inform (_("Generated exports file"));
1990 char *copy
= xmalloc (strlen (name
) + 2);
1992 strcpy (copy
+ 1, name
);
1999 p
= strchr (name
, '@');
2006 /**********************************************************************/
2015 if (exp
->noname
&& !show_allnames
)
2017 fprintf (f
, "\t%s\t0x%08x\n",
2019 exp
->ordinal
| 0x80000000); /* hint or orindal ?? */
2023 fprintf (f
, "\t%sID%d%s\n", ASM_RVA_BEFORE
,
2041 unsigned char *data
;
2056 #define INIT_SEC_DATA(id, name, flags, align) { id, name, flags, align, NULL, NULL, NULL, 0, NULL }
2057 static sinfo secdata
[NSECS
] =
2059 INIT_SEC_DATA (TEXT
, ".text", SEC_CODE
| SEC_HAS_CONTENTS
, 2),
2060 INIT_SEC_DATA (DATA
, ".data", SEC_DATA
, 2),
2061 INIT_SEC_DATA (BSS
, ".bss", 0, 2),
2062 INIT_SEC_DATA (IDATA7
, ".idata$7", SEC_HAS_CONTENTS
, 2),
2063 INIT_SEC_DATA (IDATA5
, ".idata$5", SEC_HAS_CONTENTS
, 2),
2064 INIT_SEC_DATA (IDATA4
, ".idata$4", SEC_HAS_CONTENTS
, 2),
2065 INIT_SEC_DATA (IDATA6
, ".idata$6", SEC_HAS_CONTENTS
, 1)
2070 /* Sections numbered to make the order the same as other PowerPC NT */
2071 /* compilers. This also keeps funny alignment thingies from happening. */
2084 static sinfo secdata
[NSECS
] =
2086 { TEXT
, ".text", SEC_CODE
| SEC_HAS_CONTENTS
, 3},
2087 { PDATA
, ".pdata", SEC_HAS_CONTENTS
, 2},
2088 { RDATA
, ".reldata", SEC_HAS_CONTENTS
, 2},
2089 { IDATA5
, ".idata$5", SEC_HAS_CONTENTS
, 2},
2090 { IDATA4
, ".idata$4", SEC_HAS_CONTENTS
, 2},
2091 { IDATA6
, ".idata$6", SEC_HAS_CONTENTS
, 1},
2092 { IDATA7
, ".idata$7", SEC_HAS_CONTENTS
, 2},
2093 { DATA
, ".data", SEC_DATA
, 2},
2094 { BSS
, ".bss", 0, 2}
2100 This is what we're trying to make. We generate the imp symbols with
2101 both single and double underscores, for compatibility.
2104 .global _GetFileVersionInfoSizeW@8
2105 .global __imp_GetFileVersionInfoSizeW@8
2106 _GetFileVersionInfoSizeW@8:
2107 jmp * __imp_GetFileVersionInfoSizeW@8
2108 .section .idata$7 # To force loading of head
2109 .long __version_a_head
2110 # Import Address Table
2112 __imp_GetFileVersionInfoSizeW@8:
2115 # Import Lookup Table
2121 .asciz "GetFileVersionInfoSizeW"
2124 For the PowerPC, here's the variation on the above scheme:
2126 # Rather than a simple "jmp *", the code to get to the dll function
2129 lwz r11,[tocv]__imp_function_name(r2)
2130 # RELOC: 00000000 TOCREL16,TOCDEFN __imp_function_name
2139 make_label (prefix
, name
)
2143 int len
= strlen (ASM_PREFIX
) + strlen (prefix
) + strlen (name
);
2144 char *copy
= xmalloc (len
+1 );
2145 strcpy (copy
, ASM_PREFIX
);
2146 strcat (copy
, prefix
);
2147 strcat (copy
, name
);
2152 make_one_lib_file (exp
, i
)
2160 const char *prefix
= "d";
2163 name
= (char *) alloca (strlen (prefix
) + 10);
2164 sprintf (name
, "%ss%05d.s", prefix
, i
);
2165 f
= fopen (name
, FOPEN_WT
);
2166 fprintf (f
, "\t.text\n");
2167 fprintf (f
, "\t%s\t%s%s\n", ASM_GLOBAL
, ASM_PREFIX
, exp
->name
);
2168 if (create_compat_implib
)
2169 fprintf (f
, "\t%s\t__imp_%s\n", ASM_GLOBAL
, exp
->name
);
2170 fprintf (f
, "\t%s\t_imp__%s\n", ASM_GLOBAL
, exp
->name
);
2171 if (create_compat_implib
)
2172 fprintf (f
, "%s%s:\n\t%s\t__imp_%s\n", ASM_PREFIX
,
2173 exp
->name
, ASM_JUMP
, exp
->name
);
2175 fprintf (f
, "\t.section\t.idata$7\t%s To force loading of head\n", ASM_C
);
2176 fprintf (f
, "\t%s\t%s\n", ASM_LONG
, head_label
);
2179 fprintf (f
,"%s Import Address Table\n", ASM_C
);
2181 fprintf (f
, "\t.section .idata$5\n");
2182 if (create_compat_implib
)
2183 fprintf (f
, "__imp_%s:\n", exp
->name
);
2184 fprintf (f
, "_imp__%s:\n", exp
->name
);
2188 fprintf (f
, "\n%s Import Lookup Table\n", ASM_C
);
2189 fprintf (f
, "\t.section .idata$4\n");
2193 if(!exp
->noname
|| show_allnames
)
2195 fprintf (f
, "%s Hint/Name table\n", ASM_C
);
2196 fprintf (f
, "\t.section .idata$6\n");
2197 fprintf (f
, "ID%d:\t%s\t%d\n", exp
->ordinal
, ASM_SHORT
, exp
->hint
);
2198 fprintf (f
, "\t%s\t\"%s\"\n", ASM_TEXT
, xlate (exp
->name
));
2203 dest
= (char *) alloca (strlen (prefix
) + 10);
2204 sprintf (dest
, "%ss%05d.o", prefix
, i
);
2205 assemble_file (name
, dest
);
2210 asymbol
* exp_label
;
2213 asymbol
* iname_lab
;
2214 asymbol
** iname_lab_pp
;
2215 asymbol
** iname_pp
;
2224 asymbol
* ptrs
[NSECS
+ 4 + EXTRA
+ 1];
2226 char * outname
= xmalloc (10);
2230 sprintf (outname
, "%s%05d.o", TMP_STUB
, i
);
2232 abfd
= bfd_openw (outname
, HOW_BFD_WRITE_TARGET
);
2235 /* xgettext:c-format */
2236 fatal (_("bfd_open failed open stub file: %s"), outname
);
2238 /* xgettext:c-format */
2239 inform (_("Creating stub file: %s"), outname
);
2241 bfd_set_format (abfd
, bfd_object
);
2242 bfd_set_arch_mach (abfd
, HOW_BFD_ARCH
, 0);
2245 if (machine
== MARM_INTERWORK
|| machine
== MTHUMB
)
2246 bfd_set_private_flags (abfd
, F_INTERWORK
);
2249 /* First make symbols for the sections */
2250 for (i
= 0; i
< NSECS
; i
++)
2252 sinfo
*si
= secdata
+ i
;
2255 si
->sec
= bfd_make_section_old_way (abfd
, si
->name
);
2256 bfd_set_section_flags (abfd
,
2260 bfd_set_section_alignment(abfd
, si
->sec
, si
->align
);
2261 si
->sec
->output_section
= si
->sec
;
2262 si
->sym
= bfd_make_empty_symbol(abfd
);
2263 si
->sym
->name
= si
->sec
->name
;
2264 si
->sym
->section
= si
->sec
;
2265 si
->sym
->flags
= BSF_LOCAL
;
2267 ptrs
[oidx
] = si
->sym
;
2268 si
->sympp
= ptrs
+ oidx
;
2277 exp_label
= bfd_make_empty_symbol (abfd
);
2278 exp_label
->name
= make_label ("", exp
->name
);
2280 /* On PowerPC, the function name points to a descriptor in
2281 the rdata section, the first element of which is a
2282 pointer to the code (..function_name), and the second
2283 points to the .toc */
2285 if (machine
== MPPC
)
2286 exp_label
->section
= secdata
[RDATA
].sec
;
2289 exp_label
->section
= secdata
[TEXT
].sec
;
2291 exp_label
->flags
= BSF_GLOBAL
;
2292 exp_label
->value
= 0;
2295 if (machine
== MTHUMB
)
2296 bfd_coff_set_symbol_class (abfd
, exp_label
, C_THUMBEXTFUNC
);
2298 ptrs
[oidx
++] = exp_label
;
2301 /* Generate imp symbols with one underscore for Microsoft
2302 compatibility, and with two underscores for backward
2303 compatibility with old versions of cygwin. */
2304 if (create_compat_implib
)
2306 iname
= bfd_make_empty_symbol (abfd
);
2307 iname
->name
= make_label ("__imp_", exp
->name
);
2308 iname
->section
= secdata
[IDATA5
].sec
;
2309 iname
->flags
= BSF_GLOBAL
;
2313 iname2
= bfd_make_empty_symbol (abfd
);
2314 iname2
->name
= make_label ("_imp__", exp
->name
);
2315 iname2
->section
= secdata
[IDATA5
].sec
;
2316 iname2
->flags
= BSF_GLOBAL
;
2319 iname_lab
= bfd_make_empty_symbol(abfd
);
2321 iname_lab
->name
= head_label
;
2322 iname_lab
->section
= (asection
*)&bfd_und_section
;
2323 iname_lab
->flags
= 0;
2324 iname_lab
->value
= 0;
2327 iname_pp
= ptrs
+ oidx
;
2328 if (create_compat_implib
)
2329 ptrs
[oidx
++] = iname
;
2330 ptrs
[oidx
++] = iname2
;
2332 iname_lab_pp
= ptrs
+ oidx
;
2333 ptrs
[oidx
++] = iname_lab
;
2336 /* The symbol refering to the code (.text) */
2338 asymbol
*function_name
;
2340 function_name
= bfd_make_empty_symbol(abfd
);
2341 function_name
->name
= make_label ("..", exp
->name
);
2342 function_name
->section
= secdata
[TEXT
].sec
;
2343 function_name
->flags
= BSF_GLOBAL
;
2344 function_name
->value
= 0;
2346 fn_pp
= ptrs
+ oidx
;
2347 ptrs
[oidx
++] = function_name
;
2350 /* The .toc symbol */
2352 asymbol
*toc_symbol
; /* The .toc symbol */
2354 toc_symbol
= bfd_make_empty_symbol (abfd
);
2355 toc_symbol
->name
= make_label (".", "toc");
2356 toc_symbol
->section
= (asection
*)&bfd_und_section
;
2357 toc_symbol
->flags
= BSF_GLOBAL
;
2358 toc_symbol
->value
= 0;
2360 toc_pp
= ptrs
+ oidx
;
2361 ptrs
[oidx
++] = toc_symbol
;
2367 for (i
= 0; i
< NSECS
; i
++)
2369 sinfo
*si
= secdata
+ i
;
2370 asection
*sec
= si
->sec
;
2379 si
->size
= HOW_JTAB_SIZE
;
2380 si
->data
= xmalloc (HOW_JTAB_SIZE
);
2381 memcpy (si
->data
, HOW_JTAB
, HOW_JTAB_SIZE
);
2383 /* add the reloc into idata$5 */
2384 rel
= xmalloc (sizeof (arelent
));
2386 rpp
= xmalloc (sizeof (arelent
*) * 2);
2390 rel
->address
= HOW_JTAB_ROFF
;
2393 if (machine
== MPPC
)
2395 rel
->howto
= bfd_reloc_type_lookup (abfd
,
2396 BFD_RELOC_16_GOTOFF
);
2397 rel
->sym_ptr_ptr
= iname_pp
;
2401 rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2402 rel
->sym_ptr_ptr
= secdata
[IDATA5
].sympp
;
2404 sec
->orelocation
= rpp
;
2405 sec
->reloc_count
= 1;
2410 /* An idata$4 or idata$5 is one word long, and has an
2413 si
->data
= xmalloc (4);
2418 si
->data
[0] = exp
->ordinal
;
2419 si
->data
[1] = exp
->ordinal
>> 8;
2420 si
->data
[2] = exp
->ordinal
>> 16;
2425 sec
->reloc_count
= 1;
2426 memset (si
->data
, 0, si
->size
);
2427 rel
= xmalloc (sizeof (arelent
));
2428 rpp
= xmalloc (sizeof (arelent
*) * 2);
2433 rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_RVA
);
2434 rel
->sym_ptr_ptr
= secdata
[IDATA6
].sympp
;
2435 sec
->orelocation
= rpp
;
2443 /* This used to add 1 to exp->hint. I don't know
2444 why it did that, and it does not match what I see
2445 in programs compiled with the MS tools. */
2446 int idx
= exp
->hint
;
2447 si
->size
= strlen (xlate (exp
->name
)) + 3;
2448 si
->data
= xmalloc (si
->size
);
2449 si
->data
[0] = idx
& 0xff;
2450 si
->data
[1] = idx
>> 8;
2451 strcpy (si
->data
+ 2, xlate (exp
->name
));
2456 si
->data
=xmalloc(4);
2457 memset (si
->data
, 0, si
->size
);
2458 rel
= xmalloc (sizeof (arelent
));
2459 rpp
= xmalloc (sizeof (arelent
*) * 2);
2463 rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_RVA
);
2464 rel
->sym_ptr_ptr
= iname_lab_pp
;
2465 sec
->orelocation
= rpp
;
2466 sec
->reloc_count
= 1;
2472 /* The .pdata section is 5 words long. */
2473 /* Think of it as: */
2476 /* bfd_vma BeginAddress, [0x00] */
2477 /* EndAddress, [0x04] */
2478 /* ExceptionHandler, [0x08] */
2479 /* HandlerData, [0x0c] */
2480 /* PrologEndAddress; [0x10] */
2483 /* So this pdata section setups up this as a glue linkage to
2484 a dll routine. There are a number of house keeping things
2487 1. In the name of glue trickery, the ADDR32 relocs for 0,
2488 4, and 0x10 are set to point to the same place:
2490 2. There is one more reloc needed in the pdata section.
2491 The actual glue instruction to restore the toc on
2492 return is saved as the offset in an IMGLUE reloc.
2493 So we need a total of four relocs for this section.
2495 3. Lastly, the HandlerData field is set to 0x03, to indicate
2496 that this is a glue routine.
2498 arelent
*imglue
, *ba_rel
, *ea_rel
, *pea_rel
;
2500 /* alignment must be set to 2**2 or you get extra stuff */
2501 bfd_set_section_alignment(abfd
, sec
, 2);
2504 si
->data
=xmalloc(4 * 5);
2505 memset (si
->data
, 0, si
->size
);
2506 rpp
= xmalloc (sizeof (arelent
*) * 5);
2507 rpp
[0] = imglue
= xmalloc (sizeof (arelent
));
2508 rpp
[1] = ba_rel
= xmalloc (sizeof (arelent
));
2509 rpp
[2] = ea_rel
= xmalloc (sizeof (arelent
));
2510 rpp
[3] = pea_rel
= xmalloc (sizeof (arelent
));
2513 /* stick the toc reload instruction in the glue reloc */
2514 bfd_put_32(abfd
, ppc_glue_insn
, (char *) &imglue
->address
);
2517 imglue
->howto
= bfd_reloc_type_lookup (abfd
,
2518 BFD_RELOC_32_GOTOFF
);
2519 imglue
->sym_ptr_ptr
= fn_pp
;
2521 ba_rel
->address
= 0;
2523 ba_rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2524 ba_rel
->sym_ptr_ptr
= fn_pp
;
2526 bfd_put_32(abfd
, 0x18, si
->data
+ 0x04);
2527 ea_rel
->address
= 4;
2529 ea_rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2530 ea_rel
->sym_ptr_ptr
= fn_pp
;
2532 /* mark it as glue */
2533 bfd_put_32(abfd
, 0x03, si
->data
+ 0x0c);
2535 /* mark the prolog end address */
2536 bfd_put_32(abfd
, 0x0D, si
->data
+ 0x10);
2537 pea_rel
->address
= 0x10;
2538 pea_rel
->addend
= 0;
2539 pea_rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2540 pea_rel
->sym_ptr_ptr
= fn_pp
;
2542 sec
->orelocation
= rpp
;
2543 sec
->reloc_count
= 4;
2547 /* Each external function in a PowerPC PE file has a two word
2548 descriptor consisting of:
2549 1. The address of the code.
2550 2. The address of the appropriate .toc
2551 We use relocs to build this.
2555 si
->data
= xmalloc (8);
2556 memset (si
->data
, 0, si
->size
);
2558 rpp
= xmalloc (sizeof (arelent
*) * 3);
2559 rpp
[0] = rel
= xmalloc (sizeof (arelent
));
2560 rpp
[1] = xmalloc (sizeof (arelent
));
2565 rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2566 rel
->sym_ptr_ptr
= fn_pp
;
2572 rel
->howto
= bfd_reloc_type_lookup (abfd
, BFD_RELOC_32
);
2573 rel
->sym_ptr_ptr
= toc_pp
;
2575 sec
->orelocation
= rpp
;
2576 sec
->reloc_count
= 2;
2578 #endif /* DLLTOOL_PPC */
2584 /* Size up all the sections */
2585 for (i
= 0; i
< NSECS
; i
++)
2587 sinfo
*si
= secdata
+ i
;
2589 bfd_set_section_size (abfd
, si
->sec
, si
->size
);
2590 bfd_set_section_vma (abfd
, si
->sec
, vma
);
2592 /* vma += si->size;*/
2595 /* Write them out */
2596 for (i
= 0; i
< NSECS
; i
++)
2598 sinfo
*si
= secdata
+ i
;
2600 if (i
== IDATA5
&& no_idata5
)
2603 if (i
== IDATA4
&& no_idata4
)
2606 bfd_set_section_contents (abfd
, si
->sec
,
2611 bfd_set_symtab (abfd
, ptrs
, oidx
);
2613 abfd
= bfd_openr (outname
, HOW_BFD_READ_TARGET
);
2622 FILE *f
= fopen (TMP_HEAD_S
, FOPEN_WT
);
2626 fatal (_("failed to open temporary head file: %s"), TMP_HEAD_S
);
2630 fprintf (f
, "%s IMAGE_IMPORT_DESCRIPTOR\n", ASM_C
);
2631 fprintf (f
, "\t.section .idata$2\n");
2633 fprintf(f
,"\t%s\t%s\n", ASM_GLOBAL
,head_label
);
2635 fprintf (f
, "%s:\n", head_label
);
2637 fprintf (f
, "\t%shname%s\t%sPtr to image import by name list\n",
2638 ASM_RVA_BEFORE
, ASM_RVA_AFTER
, ASM_C
);
2640 fprintf (f
, "\t%sthis should be the timestamp, but NT sometimes\n", ASM_C
);
2641 fprintf (f
, "\t%sdoesn't load DLLs when this is set.\n", ASM_C
);
2642 fprintf (f
, "\t%s\t0\t%s loaded time\n", ASM_LONG
, ASM_C
);
2643 fprintf (f
, "\t%s\t0\t%s Forwarder chain\n", ASM_LONG
, ASM_C
);
2644 fprintf (f
, "\t%s__%s_iname%s\t%s imported dll's name\n",
2649 fprintf (f
, "\t%sfthunk%s\t%s pointer to firstthunk\n",
2651 ASM_RVA_AFTER
, ASM_C
);
2653 fprintf (f
, "%sStuff for compatibility\n", ASM_C
);
2657 fprintf (f
, "\t.section\t.idata$5\n");
2658 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2659 fprintf (f
, "fthunk:\n");
2664 fprintf (f
, "\t.section\t.idata$4\n");
2666 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2667 fprintf (f
, "\t.section .idata$4\n");
2668 fprintf (f
, "hname:\n");
2673 assemble_file (TMP_HEAD_S
, TMP_HEAD_O
);
2675 return bfd_openr (TMP_HEAD_O
, HOW_BFD_READ_TARGET
);
2681 FILE *f
= fopen (TMP_TAIL_S
, FOPEN_WT
);
2685 fatal (_("failed to open temporary tail file: %s"), TMP_TAIL_S
);
2691 fprintf (f
, "\t.section .idata$4\n");
2692 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2697 fprintf (f
, "\t.section .idata$5\n");
2698 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2702 /* Normally, we need to see a null descriptor built in idata$3 to
2703 act as the terminator for the list. The ideal way, I suppose,
2704 would be to mark this section as a comdat type 2 section, so
2705 only one would appear in the final .exe (if our linker supported
2706 comdat, that is) or cause it to be inserted by something else (say
2710 fprintf (f
, "\t.section .idata$3\n");
2711 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2712 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2713 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2714 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2715 fprintf (f
, "\t%s\t0\n", ASM_LONG
);
2719 /* Other PowerPC NT compilers use idata$6 for the dllname, so I
2720 do too. Original, huh? */
2721 fprintf (f
, "\t.section .idata$6\n");
2723 fprintf (f
, "\t.section .idata$7\n");
2726 fprintf (f
, "\t%s\t__%s_iname\n", ASM_GLOBAL
, imp_name_lab
);
2727 fprintf (f
, "__%s_iname:\t%s\t\"%s\"\n",
2728 imp_name_lab
, ASM_TEXT
, dll_name
);
2732 assemble_file (TMP_TAIL_S
, TMP_TAIL_O
);
2734 return bfd_openr (TMP_TAIL_O
, HOW_BFD_READ_TARGET
);
2749 outarch
= bfd_openw (imp_name
, HOW_BFD_WRITE_TARGET
);
2752 /* xgettext:c-format */
2753 fatal (_("Can't open .lib file: %s"), imp_name
);
2755 /* xgettext:c-format */
2756 inform (_("Creating library file: %s"), imp_name
);
2758 bfd_set_format (outarch
, bfd_archive
);
2759 outarch
->has_armap
= 1;
2761 /* Work out a reasonable size of things to put onto one line. */
2763 ar_head
= make_head ();
2764 ar_tail
= make_tail();
2766 if (ar_head
== NULL
|| ar_tail
== NULL
)
2769 for (i
= 0; (exp
= d_exports_lexically
[i
]); i
++)
2771 bfd
*n
= make_one_lib_file (exp
, i
);
2776 /* Now stick them all into the archive */
2778 ar_head
->next
= head
;
2779 ar_tail
->next
= ar_head
;
2782 if (! bfd_set_archive_head (outarch
, head
))
2783 bfd_fatal ("bfd_set_archive_head");
2785 if (! bfd_close (outarch
))
2786 bfd_fatal (imp_name
);
2788 while (head
!= NULL
)
2790 bfd
*n
= head
->next
;
2795 /* Delete all the temp files */
2797 if (dontdeltemps
== 0)
2799 unlink (TMP_HEAD_O
);
2800 unlink (TMP_HEAD_S
);
2801 unlink (TMP_TAIL_O
);
2802 unlink (TMP_TAIL_S
);
2805 if (dontdeltemps
< 2)
2809 name
= (char *) alloca (sizeof TMP_STUB
+ 10);
2810 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
2812 sprintf (name
, "%s%05d.o", TMP_STUB
, i
);
2813 if (unlink (name
) < 0)
2814 /* xgettext:c-format */
2815 non_fatal (_("cannot delete %s: %s"), name
, strerror (errno
));
2819 inform (_("Created lib file"));
2822 /**********************************************************************/
2824 /* Run through the information gathered from the .o files and the
2825 .def file and work out the best stuff */
2831 export_type
*ap
= *(export_type
**) a
;
2832 export_type
*bp
= *(export_type
**) b
;
2833 if (ap
->ordinal
== bp
->ordinal
)
2836 /* unset ordinals go to the bottom */
2837 if (ap
->ordinal
== -1)
2839 if (bp
->ordinal
== -1)
2841 return (ap
->ordinal
- bp
->ordinal
);
2849 export_type
*ap
= *(export_type
**) a
;
2850 export_type
*bp
= *(export_type
**) b
;
2852 return (strcmp (ap
->name
, bp
->name
));
2856 remove_null_names (ptr
)
2861 for (dst
= src
= 0; src
< d_nfuncs
; src
++)
2865 ptr
[dst
] = ptr
[src
];
2882 for (i
= 0; i
< d_nfuncs
; i
++)
2886 printf ("%d %s @ %d %s%s%s\n",
2887 i
, ptr
[i
]->name
, ptr
[i
]->ordinal
,
2888 ptr
[i
]->noname
? "NONAME " : "",
2889 ptr
[i
]->constant
? "CONSTANT" : "",
2890 ptr
[i
]->data
? "DATA" : "");
2899 process_duplicates (d_export_vec
)
2900 export_type
**d_export_vec
;
2908 /* Remove duplicates */
2909 qsort (d_export_vec
, d_nfuncs
, sizeof (export_type
*), nfunc
);
2911 dtab (d_export_vec
);
2912 for (i
= 0; i
< d_nfuncs
- 1; i
++)
2914 if (strcmp (d_export_vec
[i
]->name
,
2915 d_export_vec
[i
+ 1]->name
) == 0)
2918 export_type
*a
= d_export_vec
[i
];
2919 export_type
*b
= d_export_vec
[i
+ 1];
2923 /* xgettext:c-format */
2924 inform (_("Warning, ignoring duplicate EXPORT %s %d,%d"),
2925 a
->name
, a
->ordinal
, b
->ordinal
);
2927 if (a
->ordinal
!= -1
2928 && b
->ordinal
!= -1)
2929 /* xgettext:c-format */
2930 fatal (_("Error, duplicate EXPORT with oridinals: %s"),
2933 /* Merge attributes */
2934 b
->ordinal
= a
->ordinal
> 0 ? a
->ordinal
: b
->ordinal
;
2935 b
->constant
|= a
->constant
;
2936 b
->noname
|= a
->noname
;
2938 d_export_vec
[i
] = 0;
2941 dtab (d_export_vec
);
2942 remove_null_names (d_export_vec
);
2943 dtab (d_export_vec
);
2948 /* Count the names */
2949 for (i
= 0; i
< d_nfuncs
; i
++)
2951 if (!d_export_vec
[i
]->noname
)
2957 fill_ordinals (d_export_vec
)
2958 export_type
**d_export_vec
;
2965 qsort (d_export_vec
, d_nfuncs
, sizeof (export_type
*), pfunc
);
2967 /* fill in the unset ordinals with ones from our range */
2969 ptr
= (char *) xmalloc (size
);
2971 memset (ptr
, 0, size
);
2973 /* Mark in our large vector all the numbers that are taken */
2974 for (i
= 0; i
< d_nfuncs
; i
++)
2976 if (d_export_vec
[i
]->ordinal
!= -1)
2978 ptr
[d_export_vec
[i
]->ordinal
] = 1;
2979 if (lowest
== -1 || d_export_vec
[i
]->ordinal
< lowest
)
2981 lowest
= d_export_vec
[i
]->ordinal
;
2986 /* Start at 1 for compatibility with MS toolchain. */
2990 /* Now fill in ordinals where the user wants us to choose. */
2991 for (i
= 0; i
< d_nfuncs
; i
++)
2993 if (d_export_vec
[i
]->ordinal
== -1)
2997 /* First try within or after any user supplied range. */
2998 for (j
= lowest
; j
< size
; j
++)
3002 d_export_vec
[i
]->ordinal
= j
;
3006 /* Then try before the range. */
3007 for (j
= lowest
; j
>0; j
--)
3011 d_export_vec
[i
]->ordinal
= j
;
3022 qsort (d_export_vec
, d_nfuncs
, sizeof (export_type
*), pfunc
);
3024 /* Work out the lowest and highest ordinal numbers. */
3027 if (d_export_vec
[0])
3028 d_low_ord
= d_export_vec
[0]->ordinal
;
3029 if (d_export_vec
[d_nfuncs
-1])
3030 d_high_ord
= d_export_vec
[d_nfuncs
-1]->ordinal
;
3039 const export_type
**a
= (const export_type
**) av
;
3040 const export_type
**b
= (const export_type
**) bv
;
3042 return strcmp ((*a
)->name
, (*b
)->name
);
3048 /* First work out the minimum ordinal chosen */
3054 export_type
**d_export_vec
3055 = (export_type
**) xmalloc (sizeof (export_type
*) * d_nfuncs
);
3057 inform (_("Processing definitions"));
3059 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
3061 d_export_vec
[i
] = exp
;
3064 process_duplicates (d_export_vec
);
3065 fill_ordinals (d_export_vec
);
3067 /* Put back the list in the new order */
3069 for (i
= d_nfuncs
- 1; i
>= 0; i
--)
3071 d_export_vec
[i
]->next
= d_exports
;
3072 d_exports
= d_export_vec
[i
];
3075 /* Build list in alpha order */
3076 d_exports_lexically
= (export_type
**)
3077 xmalloc (sizeof (export_type
*) * (d_nfuncs
+ 1));
3079 for (i
= 0, exp
= d_exports
; exp
; i
++, exp
= exp
->next
)
3081 d_exports_lexically
[i
] = exp
;
3083 d_exports_lexically
[i
] = 0;
3085 qsort (d_exports_lexically
, i
, sizeof (export_type
*), alphafunc
);
3087 /* Fill exp entries with their hint values */
3089 for (i
= 0; i
< d_nfuncs
; i
++)
3091 if (!d_exports_lexically
[i
]->noname
|| show_allnames
)
3092 d_exports_lexically
[i
]->hint
= hint
++;
3095 inform (_("Processed definitions"));
3098 /**********************************************************************/
3101 usage (file
, status
)
3105 /* xgetext:c-format */
3106 fprintf (file
, _("Usage %s <options> <object-files>\n"), program_name
);
3107 /* xgetext:c-format */
3108 fprintf (file
, _(" -m --machine <machine> Create as DLL for <machine>. [default: %s]\n"), mname
);
3109 fprintf (file
, _(" possible <machine>: arm[_interwork], i386, mcore[-elf]{-le|-be}, ppc, thumb\n"));
3110 fprintf (file
, _(" -e --output-exp <outname> Generate an export file.\n"));
3111 fprintf (file
, _(" -l --output-lib <outname> Generate an interface library.\n"));
3112 fprintf (file
, _(" -a --add-indirect Add dll indirects to export file.\n"));
3113 fprintf (file
, _(" -D --dllname <name> Name of input dll to put into interface lib.\n"));
3114 fprintf (file
, _(" -d --input-def <deffile> Name of .def file to be read in.\n"));
3115 fprintf (file
, _(" -z --output-def <deffile> Name of .def file to be created.\n"));
3116 fprintf (file
, _(" --export-all-symbols Export all symbols to .def\n"));
3117 fprintf (file
, _(" --no-export-all-symbols Only export listed symbols\n"));
3118 fprintf (file
, _(" --exclude-symbols <list> Don't export <list>\n"));
3119 fprintf (file
, _(" --no-default-excludes Clear default exclude symbols\n"));
3120 fprintf (file
, _(" -b --base-file <basefile> Read linker generated base file.\n"));
3121 fprintf (file
, _(" -x --no-idata4 Don't generate idata$4 section.\n"));
3122 fprintf (file
, _(" -c --no-idata5 Don't generate idata$5 section.\n"));
3123 fprintf (file
, _(" -U --add-underscore Add underscores to symbols in interface library.\n"));
3124 fprintf (file
, _(" -k --kill-at Kill @<n> from exported names.\n"));
3125 fprintf (file
, _(" -A --add-stdcall-alias Add aliases without @<n>.\n"));
3126 fprintf (file
, _(" -S --as <name> Use <name> for assembler.\n"));
3127 fprintf (file
, _(" -f --as-flags <flags> Pass <flags> to the assembler.\n"));
3128 fprintf (file
, _(" -C --compat-implib Create backward compatible import library.\n"));
3129 fprintf (file
, _(" -n --no-delete Keep temp files (repeat for extra preservation).\n"));
3130 fprintf (file
, _(" -v --verbose Be verbose.\n"));
3131 fprintf (file
, _(" -V --version Display the program version.\n"));
3132 fprintf (file
, _(" -h --help Display this information.\n"));
3133 #ifdef DLLTOOL_MCORE_ELF
3134 fprintf (file
, _(" -M --mcore-elf <outname> Process mcore-elf object files into <outname>.\n"));
3135 fprintf (file
, _(" -L --linker <name> Use <name> as the linker.\n"));
3136 fprintf (file
, _(" -F --linker-flags <flags> Pass <flags> to the linker.\n"));
3141 #define OPTION_EXPORT_ALL_SYMS 150
3142 #define OPTION_NO_EXPORT_ALL_SYMS (OPTION_EXPORT_ALL_SYMS + 1)
3143 #define OPTION_EXCLUDE_SYMS (OPTION_NO_EXPORT_ALL_SYMS + 1)
3144 #define OPTION_NO_DEFAULT_EXCLUDES (OPTION_EXCLUDE_SYMS + 1)
3146 static const struct option long_options
[] =
3148 {"no-delete", no_argument
, NULL
, 'n'},
3149 {"dllname", required_argument
, NULL
, 'D'},
3150 {"no-idata4", no_argument
, NULL
, 'x'},
3151 {"no-idata5", no_argument
, NULL
, 'c'},
3152 {"output-exp", required_argument
, NULL
, 'e'},
3153 {"output-def", required_argument
, NULL
, 'z'},
3154 {"export-all-symbols", no_argument
, NULL
, OPTION_EXPORT_ALL_SYMS
},
3155 {"no-export-all-symbols", no_argument
, NULL
, OPTION_NO_EXPORT_ALL_SYMS
},
3156 {"exclude-symbols", required_argument
, NULL
, OPTION_EXCLUDE_SYMS
},
3157 {"no-default-excludes", no_argument
, NULL
, OPTION_NO_DEFAULT_EXCLUDES
},
3158 {"output-lib", required_argument
, NULL
, 'l'},
3159 {"def", required_argument
, NULL
, 'd'}, /* for compatiblity with older versions */
3160 {"input-def", required_argument
, NULL
, 'd'},
3161 {"add-underscore", no_argument
, NULL
, 'U'},
3162 {"kill-at", no_argument
, NULL
, 'k'},
3163 {"add-stdcall-alias", no_argument
, NULL
, 'A'},
3164 {"verbose", no_argument
, NULL
, 'v'},
3165 {"version", no_argument
, NULL
, 'V'},
3166 {"help", no_argument
, NULL
, 'h'},
3167 {"machine", required_argument
, NULL
, 'm'},
3168 {"add-indirect", no_argument
, NULL
, 'a'},
3169 {"base-file", required_argument
, NULL
, 'b'},
3170 {"as", required_argument
, NULL
, 'S'},
3171 {"as-flags", required_argument
, NULL
, 'f'},
3172 {"mcore-elf", required_argument
, NULL
, 'M'},
3173 {"compat-implib", no_argument
, NULL
, 'C'},
3185 program_name
= av
[0];
3188 #if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
3189 setlocale (LC_MESSAGES
, "");
3191 bindtextdomain (PACKAGE
, LOCALEDIR
);
3192 textdomain (PACKAGE
);
3194 while ((c
= getopt_long (ac
, av
,
3195 #ifdef DLLTOOL_MCORE_ELF
3196 "m:e:l:aD:d:z:b:xcCuUkAS:f:nvVhM:L:F:",
3198 "m:e:l:aD:d:z:b:xcCuUkAS:f:nvVh",
3205 case OPTION_EXPORT_ALL_SYMS
:
3206 export_all_symbols
= true;
3208 case OPTION_NO_EXPORT_ALL_SYMS
:
3209 export_all_symbols
= false;
3211 case OPTION_EXCLUDE_SYMS
:
3212 add_excludes (optarg
);
3214 case OPTION_NO_DEFAULT_EXCLUDES
:
3215 do_default_excludes
= false;
3230 /* ignored for compatibility */
3237 output_def
= fopen (optarg
, FOPEN_WT
);
3258 print_version (program_name
);
3267 add_stdcall_alias
= 1;
3276 base_file
= fopen (optarg
, FOPEN_RB
);
3279 /* xgettext:c-format */
3280 fatal (_("Unable to open base-file: %s"), optarg
);
3283 #ifdef DLLTOOL_MCORE_ELF
3285 mcore_elf_out_file
= optarg
;
3288 mcore_elf_linker
= optarg
;
3291 mcore_elf_linker_flags
= optarg
;
3295 create_compat_implib
= 1;
3303 for (i
= 0; mtable
[i
].type
; i
++)
3304 if (strcmp (mtable
[i
].type
, mname
) == 0)
3307 if (!mtable
[i
].type
)
3308 /* xgettext:c-format */
3309 fatal (_("Machine '%s' not supported"), mname
);
3313 if (!dll_name
&& exp_name
)
3315 int len
= strlen (exp_name
) + 5;
3316 dll_name
= xmalloc (len
);
3317 strcpy (dll_name
, exp_name
);
3318 strcat (dll_name
, ".dll");
3321 if (as_name
== NULL
)
3322 as_name
= deduce_name ("as");
3324 /* Don't use the default exclude list if we're reading only the
3325 symbols in the .drectve section. The default excludes are meant
3326 to avoid exporting DLL entry point and Cygwin32 impure_ptr. */
3327 if (! export_all_symbols
)
3328 do_default_excludes
= false;
3330 if (do_default_excludes
)
3331 set_default_excludes ();
3334 process_def_file (def_file
);
3339 firstarg
= av
[optind
];
3340 scan_obj_file (av
[optind
]);
3351 /* Make imp_name safe for use as a label. */
3354 imp_name_lab
= xstrdup (imp_name
);
3355 for (p
= imp_name_lab
; *p
; p
++)
3357 if (!isalpha ((unsigned char) *p
) && !isdigit ((unsigned char) *p
))
3360 head_label
= make_label("_head_", imp_name_lab
);
3367 #ifdef DLLTOOL_MCORE_ELF
3368 if (mcore_elf_out_file
)
3369 mcore_elf_gen_out_file ();
3375 /* Look for the program formed by concatenating PROG_NAME and the
3376 string running from PREFIX to END_PREFIX. If the concatenated
3377 string contains a '/', try appending EXECUTABLE_SUFFIX if it is
3381 look_for_prog (prog_name
, prefix
, end_prefix
)
3382 const char *prog_name
;
3389 cmd
= xmalloc (strlen (prefix
)
3390 + strlen (prog_name
)
3391 #ifdef HAVE_EXECUTABLE_SUFFIX
3392 + strlen (EXECUTABLE_SUFFIX
)
3395 strcpy (cmd
, prefix
);
3397 sprintf (cmd
+ end_prefix
, "%s", prog_name
);
3399 if (strchr (cmd
, '/') != NULL
)
3403 found
= (stat (cmd
, &s
) == 0
3404 #ifdef HAVE_EXECUTABLE_SUFFIX
3405 || stat (strcat (cmd
, EXECUTABLE_SUFFIX
), &s
) == 0
3411 /* xgettext:c-format */
3412 inform (_("Tried file: %s"), cmd
);
3418 /* xgettext:c-format */
3419 inform (_("Using file: %s"), cmd
);
3424 /* Deduce the name of the program we are want to invoke.
3425 PROG_NAME is the basic name of the program we want to run,
3426 eg "as" or "ld". The catch is that we might want actually
3427 run "i386-pe-as" or "ppc-pe-ld".
3429 If argv[0] contains the full path, then try to find the program
3430 in the same place, with and then without a target-like prefix.
3432 Given, argv[0] = /usr/local/bin/i586-cygwin32-dlltool,
3433 deduce_name("as") uses the following search order:
3435 /usr/local/bin/i586-cygwin32-as
3439 If there's an EXECUTABLE_SUFFIX, it'll use that as well; for each
3440 name, it'll try without and then with EXECUTABLE_SUFFIX.
3442 Given, argv[0] = i586-cygwin32-dlltool, it will not even try "as"
3443 as the fallback, but rather return i586-cygwin32-as.
3445 Oh, and given, argv[0] = dlltool, it'll return "as".
3447 Returns a dynamically allocated string. */
3450 deduce_name (prog_name
)
3451 const char *prog_name
;
3454 char *dash
, *slash
, *cp
;
3458 for (cp
= program_name
; *cp
!= '\0'; ++cp
)
3463 #if defined(__DJGPP__) || defined (__CYGWIN__) || defined(__WIN32__)
3464 *cp
== ':' || *cp
== '\\' ||
3477 /* First, try looking for a prefixed PROG_NAME in the
3478 PROGRAM_NAME directory, with the same prefix as PROGRAM_NAME. */
3479 cmd
= look_for_prog (prog_name
, program_name
, dash
- program_name
+ 1);
3482 if (slash
!= NULL
&& cmd
== NULL
)
3484 /* Next, try looking for a PROG_NAME in the same directory as
3485 that of this program. */
3486 cmd
= look_for_prog (prog_name
, program_name
, slash
- program_name
+ 1);
3491 /* Just return PROG_NAME as is. */
3492 cmd
= xstrdup (prog_name
);
3498 #ifdef DLLTOOL_MCORE_ELF
3499 typedef struct fname_cache
3502 struct fname_cache
* next
;
3506 static fname_cache fnames
;
3509 mcore_elf_cache_filename (char * filename
)
3515 while (ptr
->next
!= NULL
)
3518 ptr
->filename
= filename
;
3519 ptr
->next
= (fname_cache
*) malloc (sizeof (fname_cache
));
3520 if (ptr
->next
!= NULL
)
3521 ptr
->next
->next
= NULL
;
3524 #define MCORE_ELF_TMP_OBJ "mcoreelf.o"
3525 #define MCORE_ELF_TMP_EXP "mcoreelf.exp"
3526 #define MCORE_ELF_TMP_LIB "mcoreelf.lib"
3529 mcore_elf_gen_out_file (void)
3534 /* Step one. Run 'ld -r' on the input object files in order to resolve
3535 any internal references and to generate a single .exports section. */
3538 ds
= dyn_string_new (100);
3539 dyn_string_append (ds
, "-r ");
3541 if (mcore_elf_linker_flags
!= NULL
)
3542 dyn_string_append (ds
, mcore_elf_linker_flags
);
3544 while (ptr
->next
!= NULL
)
3546 dyn_string_append (ds
, ptr
->filename
);
3547 dyn_string_append (ds
, " ");
3552 dyn_string_append (ds
, "-o ");
3553 dyn_string_append (ds
, MCORE_ELF_TMP_OBJ
);
3555 if (mcore_elf_linker
== NULL
)
3556 mcore_elf_linker
= deduce_name ("ld");
3558 run (mcore_elf_linker
, ds
->s
);
3560 dyn_string_delete (ds
);
3562 /* Step two. Create a .exp file and a .lib file from the temporary file.
3563 Do this by recursively invoking dlltool....*/
3564 ds
= dyn_string_new (100);
3566 dyn_string_append (ds
, "-S ");
3567 dyn_string_append (ds
, as_name
);
3569 dyn_string_append (ds
, " -e ");
3570 dyn_string_append (ds
, MCORE_ELF_TMP_EXP
);
3571 dyn_string_append (ds
, " -l ");
3572 dyn_string_append (ds
, MCORE_ELF_TMP_LIB
);
3573 dyn_string_append (ds
, " " );
3574 dyn_string_append (ds
, MCORE_ELF_TMP_OBJ
);
3577 dyn_string_append (ds
, " -v");
3581 dyn_string_append (ds
, " -n");
3583 if (dontdeltemps
> 1)
3584 dyn_string_append (ds
, " -n");
3587 /* XXX - FIME: ought to check/copy other command line options as well. */
3589 run (program_name
, ds
->s
);
3591 dyn_string_delete (ds
);
3593 /* Step four. Feed the .exp and object files to ld -shared to create the dll. */
3594 ds
= dyn_string_new (100);
3596 dyn_string_append (ds
, "-shared ");
3598 if (mcore_elf_linker_flags
)
3599 dyn_string_append (ds
, mcore_elf_linker_flags
);
3601 dyn_string_append (ds
, " ");
3602 dyn_string_append (ds
, MCORE_ELF_TMP_EXP
);
3603 dyn_string_append (ds
, " ");
3604 dyn_string_append (ds
, MCORE_ELF_TMP_OBJ
);
3605 dyn_string_append (ds
, " -o ");
3606 dyn_string_append (ds
, mcore_elf_out_file
);
3608 run (mcore_elf_linker
, ds
->s
);
3610 dyn_string_delete (ds
);
3612 if (dontdeltemps
== 0)
3613 unlink (MCORE_ELF_TMP_EXP
);
3615 if (dontdeltemps
< 2)
3616 unlink (MCORE_ELF_TMP_OBJ
);
3618 #endif /* DLLTOOL_MCORE_ELF */