Cleanup code related to --identify option.
* binutils/dlltool.c (file scope): Removed globals identify_ms, identify_member_contains_symname_result, identify_dll_name_list_head, and identify_dll_name_list_tail. Renamed existing typedef dll_name_list_type to dll_name_list_node_type. Added new typedefs dll_name_list_type, symname_search_data_type, and identify_data_type. (identify_append_dll_name_to_list): Renamed to... (dll_name_list_append): ...here. Changed signature to accept list argument rather than use global. (identify_count_dll_name_list): Renamed to... (dll_name_list_count): ...here. Changed signature to accept list argument rather than use global. (identify_print_dll_name_list): Renamed to... (dll_name_list_print): ...here. Changed signature to accept list argument rather than use global. (identify_free_dll_name_list): Renamed to... (dll_name_list_free_contents): ...here. (dll_name_list_free): New function. (dll_name_list_create): New function. (identify_process_section_p): Changed signature to accept ms_style_implib argument rather than use global. (identify_member_contains_symname): Expect incoming void * data to be symname_search_data_type. (identify_dll_for_implib): Use new functions dll_name_list_create and dll_name_list_free. Use new types symname_search_data_type and identify_data_type to communicate with search routines. (identify_search_section): Expect incoming void * data to be identify_data_type. Use its contents rather than global variables.
This commit is contained in:
parent
a67ec3f4ff
commit
258936723f
2 changed files with 179 additions and 65 deletions
|
@ -1,3 +1,38 @@
|
|||
2008-01-14 Charles Wilson <cygwin@cwilson.fastmail.fm>
|
||||
|
||||
Cleanup code related to --identify option.
|
||||
|
||||
* binutils/dlltool.c (file scope): Removed globals identify_ms,
|
||||
identify_member_contains_symname_result,
|
||||
identify_dll_name_list_head, and identify_dll_name_list_tail.
|
||||
Renamed existing typedef dll_name_list_type to
|
||||
dll_name_list_node_type.
|
||||
Added new typedefs dll_name_list_type, symname_search_data_type,
|
||||
and identify_data_type.
|
||||
(identify_append_dll_name_to_list): Renamed to...
|
||||
(dll_name_list_append): ...here. Changed signature to accept list
|
||||
argument rather than use global.
|
||||
(identify_count_dll_name_list): Renamed to...
|
||||
(dll_name_list_count): ...here. Changed signature to accept list
|
||||
argument rather than use global.
|
||||
(identify_print_dll_name_list): Renamed to...
|
||||
(dll_name_list_print): ...here. Changed signature to accept list
|
||||
argument rather than use global.
|
||||
(identify_free_dll_name_list): Renamed to...
|
||||
(dll_name_list_free_contents): ...here.
|
||||
(dll_name_list_free): New function.
|
||||
(dll_name_list_create): New function.
|
||||
(identify_process_section_p): Changed signature to accept
|
||||
ms_style_implib argument rather than use global.
|
||||
(identify_member_contains_symname): Expect incoming void * data to
|
||||
be symname_search_data_type.
|
||||
(identify_dll_for_implib): Use new functions dll_name_list_create
|
||||
and dll_name_list_free. Use new types symname_search_data_type
|
||||
and identify_data_type to communicate with search routines.
|
||||
(identify_search_section): Expect incoming void * data to be
|
||||
identify_data_type. Use its contents rather than global
|
||||
variables.
|
||||
|
||||
2009-01-13 Charles Wilson <cygwin@cwilson.fastmail.fm>
|
||||
|
||||
Add --identify-strict option. Handle ms-style implibs:
|
||||
|
|
|
@ -353,27 +353,35 @@ static int no_idata5;
|
|||
static char *exp_name;
|
||||
static char *imp_name;
|
||||
static char *identify_imp_name;
|
||||
static bfd_boolean identify_ms;
|
||||
static bfd_boolean identify_strict;
|
||||
|
||||
/* Holds a linked list of dllnames associated with the
|
||||
specified import lib. Used by the identify_* code.
|
||||
The _head entry is always empty (_head->dllname is
|
||||
NULL). */
|
||||
/* Types used to implement a linked list of dllnames associated
|
||||
with the specified import lib. Used by the identify_* code.
|
||||
The head entry is acts as a sentinal node and is always empty
|
||||
(head->dllname is NULL). */
|
||||
typedef struct dll_name_list_node_t
|
||||
{
|
||||
char * dllname;
|
||||
struct dll_name_list_node_t * next;
|
||||
} dll_name_list_node_type;
|
||||
typedef struct dll_name_list_t
|
||||
{
|
||||
char * dllname;
|
||||
struct dll_name_list_t * next;
|
||||
} dll_name_list_type;
|
||||
dll_name_list_node_type * head;
|
||||
dll_name_list_node_type * tail;
|
||||
} dll_name_list_type;
|
||||
|
||||
/* Types used to pass data to iterator functions. */
|
||||
typedef struct symname_search_data_t
|
||||
{
|
||||
const char * symname;
|
||||
bfd_boolean found;
|
||||
} symname_search_data_type;
|
||||
typedef struct identify_data_t
|
||||
{
|
||||
dll_name_list_type * list;
|
||||
bfd_boolean ms_style_implib;
|
||||
} identify_data_type;
|
||||
|
||||
static dll_name_list_type * identify_dll_name_list_head;
|
||||
static dll_name_list_type * identify_dll_name_list_tail;
|
||||
/* dll_name_list management functions. */
|
||||
static void identify_append_dll_name_to_list (bfd_byte *);
|
||||
static int identify_count_dll_name_list (void);
|
||||
static void identify_print_dll_name_list (void);
|
||||
static void identify_free_dll_name_list (dll_name_list_type *);
|
||||
static bfd_boolean identify_member_contains_symname_result = FALSE;
|
||||
|
||||
static char *head_label;
|
||||
static char *imp_name_lab;
|
||||
|
@ -752,11 +760,17 @@ static bfd *make_one_lib_file (export_type *, int);
|
|||
static bfd *make_head (void);
|
||||
static bfd *make_tail (void);
|
||||
static void gen_lib_file (void);
|
||||
static void dll_name_list_append (dll_name_list_type *, bfd_byte *);
|
||||
static int dll_name_list_count (dll_name_list_type *);
|
||||
static void dll_name_list_print (dll_name_list_type *);
|
||||
static void dll_name_list_free_contents (dll_name_list_node_type *);
|
||||
static void dll_name_list_free (dll_name_list_type *);
|
||||
static dll_name_list_type * dll_name_list_create (void);
|
||||
static void identify_dll_for_implib (void);
|
||||
static void identify_search_archive
|
||||
(bfd *, void (*) (bfd *, bfd *, void *), void *);
|
||||
static void identify_search_member (bfd *, bfd *, void *);
|
||||
static bfd_boolean identify_process_section_p (asection *);
|
||||
static bfd_boolean identify_process_section_p (asection *, bfd_boolean);
|
||||
static void identify_search_section (bfd *, asection *, void *);
|
||||
static void identify_member_contains_symname (bfd *, bfd *, void *);
|
||||
|
||||
|
@ -2967,29 +2981,39 @@ gen_lib_file (void)
|
|||
inform (_("Created lib file"));
|
||||
}
|
||||
|
||||
/* Management of the identify_dll_name_list. */
|
||||
/* Append a copy of data (cast to char *) to list. */
|
||||
|
||||
static void
|
||||
identify_append_dll_name_to_list (bfd_byte * data)
|
||||
dll_name_list_append (dll_name_list_type * list, bfd_byte * data)
|
||||
{
|
||||
/* Error checking. */
|
||||
if (! list || ! list->tail)
|
||||
return;
|
||||
|
||||
/* Allocate new node. */
|
||||
dll_name_list_type * entry =
|
||||
(dll_name_list_type *) xmalloc (sizeof (dll_name_list_type));
|
||||
dll_name_list_node_type * entry =
|
||||
(dll_name_list_node_type *) xmalloc (sizeof (dll_name_list_node_type));
|
||||
|
||||
/* Initialize its values. */
|
||||
entry->dllname = xstrdup ((char *) data);
|
||||
entry->next = NULL;
|
||||
|
||||
/* Add to tail, and move tail. */
|
||||
identify_dll_name_list_tail->next = entry;
|
||||
identify_dll_name_list_tail = entry;
|
||||
list->tail->next = entry;
|
||||
list->tail = entry;
|
||||
}
|
||||
|
||||
/* Count the number of entries in list. */
|
||||
|
||||
static int
|
||||
identify_count_dll_name_list (void)
|
||||
dll_name_list_count (dll_name_list_type * list)
|
||||
{
|
||||
/* Error checking. */
|
||||
if (! list || ! list->head)
|
||||
return 0;
|
||||
|
||||
int count = 0;
|
||||
dll_name_list_type * p = identify_dll_name_list_head;
|
||||
dll_name_list_node_type * p = list->head;
|
||||
|
||||
while (p && p->next)
|
||||
{
|
||||
|
@ -2999,10 +3023,16 @@ identify_count_dll_name_list (void)
|
|||
return count;
|
||||
}
|
||||
|
||||
/* Print each entry in list to stdout. */
|
||||
|
||||
static void
|
||||
identify_print_dll_name_list (void)
|
||||
dll_name_list_print (dll_name_list_type * list)
|
||||
{
|
||||
dll_name_list_type * p = identify_dll_name_list_head;
|
||||
/* Error checking. */
|
||||
if (! list || ! list->head)
|
||||
return;
|
||||
|
||||
dll_name_list_node_type * p = list->head;
|
||||
|
||||
while (p && p->next && p->next->dllname && *(p->next->dllname))
|
||||
{
|
||||
|
@ -3011,14 +3041,31 @@ identify_print_dll_name_list (void)
|
|||
}
|
||||
}
|
||||
|
||||
/* Free all entries in list, and list itself. */
|
||||
|
||||
static void
|
||||
dll_name_list_free (dll_name_list_type * list)
|
||||
{
|
||||
if (list)
|
||||
{
|
||||
dll_name_list_free_contents (list->head);
|
||||
list->head = NULL;
|
||||
list->tail = NULL;
|
||||
free (list);
|
||||
}
|
||||
}
|
||||
|
||||
/* Recursive function to free all nodes entry->next->next...
|
||||
as well as entry itself. */
|
||||
|
||||
static void
|
||||
identify_free_dll_name_list (dll_name_list_type * entry)
|
||||
dll_name_list_free_contents (dll_name_list_node_type * entry)
|
||||
{
|
||||
if (entry)
|
||||
{
|
||||
if (entry->next)
|
||||
{
|
||||
identify_free_dll_name_list (entry->next);
|
||||
dll_name_list_free_contents (entry->next);
|
||||
entry->next = NULL;
|
||||
}
|
||||
if (entry->dllname)
|
||||
|
@ -3030,6 +3077,28 @@ identify_free_dll_name_list (dll_name_list_type * entry)
|
|||
}
|
||||
}
|
||||
|
||||
/* Allocate and initialize a dll_name_list_type object,
|
||||
including its sentinel node. Caller is responsible
|
||||
for calling dll_name_list_free when finished with
|
||||
the list. */
|
||||
|
||||
static dll_name_list_type *
|
||||
dll_name_list_create (void)
|
||||
{
|
||||
/* Allocate list. */
|
||||
dll_name_list_type * list = xmalloc (sizeof (dll_name_list_type));
|
||||
|
||||
/* Allocate and initialize sentinel node. */
|
||||
list->head = xmalloc (sizeof (dll_name_list_node_type));
|
||||
list->head->dllname = NULL;
|
||||
list->head->next = NULL;
|
||||
|
||||
/* Bookkeeping for empty list. */
|
||||
list->tail = list->head;
|
||||
|
||||
return list;
|
||||
}
|
||||
|
||||
/* Search the symbol table of the suppled BFD for a symbol whose name matches
|
||||
OBJ (where obj is cast to const char *). If found, set global variable
|
||||
identify_member_contains_symname_result TRUE. It is the caller's
|
||||
|
@ -3045,11 +3114,11 @@ identify_member_contains_symname (bfd * abfd,
|
|||
asymbol ** symbol_table;
|
||||
long number_of_symbols;
|
||||
long i;
|
||||
const char * name = (const char *) obj;
|
||||
symname_search_data_type * search_data = (symname_search_data_type *) obj;
|
||||
|
||||
/* If we already found the symbol in a different member,
|
||||
short circuit. */
|
||||
if (identify_member_contains_symname_result)
|
||||
if (search_data->found)
|
||||
return;
|
||||
|
||||
storage_needed = bfd_get_symtab_upper_bound (abfd);
|
||||
|
@ -3066,9 +3135,11 @@ identify_member_contains_symname (bfd * abfd,
|
|||
|
||||
for (i = 0; i < number_of_symbols; i++)
|
||||
{
|
||||
if (strncmp (symbol_table[i]->name, name, strlen (name)) == 0)
|
||||
if (strncmp (symbol_table[i]->name,
|
||||
search_data->symname,
|
||||
strlen (search_data->symname)) == 0)
|
||||
{
|
||||
identify_member_contains_symname_result = TRUE;
|
||||
search_data->found = TRUE;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -3097,12 +3168,16 @@ identify_dll_for_implib (void)
|
|||
{
|
||||
bfd * abfd = NULL;
|
||||
int count = 0;
|
||||
identify_data_type identify_data;
|
||||
symname_search_data_type search_data;
|
||||
|
||||
/* Initialize identify_dll_name_list. */
|
||||
identify_dll_name_list_head = xmalloc (sizeof (dll_name_list_type));
|
||||
identify_dll_name_list_head->dllname = NULL;
|
||||
identify_dll_name_list_head->next = NULL;
|
||||
identify_dll_name_list_tail = identify_dll_name_list_head;
|
||||
/* Initialize identify_data. */
|
||||
identify_data.list = dll_name_list_create ();
|
||||
identify_data.ms_style_implib = FALSE;
|
||||
|
||||
/* Initialize search_data. */
|
||||
search_data.symname = "__NULL_IMPORT_DESCRIPTOR";
|
||||
search_data.found = FALSE;
|
||||
|
||||
bfd_init ();
|
||||
|
||||
|
@ -3119,11 +3194,11 @@ identify_dll_for_implib (void)
|
|||
}
|
||||
|
||||
/* Detect if this a Microsoft import library. */
|
||||
identify_member_contains_symname_result = FALSE;
|
||||
identify_search_archive (abfd, identify_member_contains_symname,
|
||||
"__NULL_IMPORT_DESCRIPTOR");
|
||||
if (identify_member_contains_symname_result)
|
||||
identify_ms = TRUE;
|
||||
identify_search_archive (abfd,
|
||||
identify_member_contains_symname,
|
||||
(void *)(& search_data));
|
||||
if (search_data.found)
|
||||
identify_data.ms_style_implib = TRUE;
|
||||
|
||||
/* Rewind the bfd. */
|
||||
if (! bfd_close (abfd))
|
||||
|
@ -3141,29 +3216,31 @@ identify_dll_for_implib (void)
|
|||
}
|
||||
|
||||
/* Now search for the dll name. */
|
||||
identify_search_archive (abfd, identify_search_member, NULL);
|
||||
identify_search_archive (abfd,
|
||||
identify_search_member,
|
||||
(void *)(& identify_data));
|
||||
|
||||
if (! bfd_close (abfd))
|
||||
bfd_fatal (identify_imp_name);
|
||||
|
||||
count = identify_count_dll_name_list();
|
||||
count = dll_name_list_count (identify_data.list);
|
||||
if (count > 0)
|
||||
{
|
||||
if (identify_strict && count > 1)
|
||||
{
|
||||
identify_free_dll_name_list (identify_dll_name_list_head);
|
||||
identify_dll_name_list_head = NULL;
|
||||
dll_name_list_free (identify_data.list);
|
||||
identify_data.list = NULL;
|
||||
fatal (_("Import library `%s' specifies two or more dlls"),
|
||||
identify_imp_name);
|
||||
}
|
||||
identify_print_dll_name_list();
|
||||
identify_free_dll_name_list (identify_dll_name_list_head);
|
||||
identify_dll_name_list_head = NULL;
|
||||
dll_name_list_print (identify_data.list);
|
||||
dll_name_list_free (identify_data.list);
|
||||
identify_data.list = NULL;
|
||||
}
|
||||
else
|
||||
{
|
||||
identify_free_dll_name_list (identify_dll_name_list_head);
|
||||
identify_dll_name_list_head = NULL;
|
||||
dll_name_list_free (identify_data.list);
|
||||
identify_data.list = NULL;
|
||||
fatal (_("Unable to determine dll name for `%s' (not an import library?)"),
|
||||
identify_imp_name);
|
||||
}
|
||||
|
@ -3225,10 +3302,11 @@ identify_search_member (bfd *abfd,
|
|||
}
|
||||
|
||||
/* This predicate returns true if section->name matches the desired value.
|
||||
By default, this is .idata$7 (.idata$6 on PPC, or when --identify-ms). */
|
||||
By default, this is .idata$7 (.idata$6 on PPC, or if the import
|
||||
library is ms-style). */
|
||||
|
||||
static bfd_boolean
|
||||
identify_process_section_p (asection * section)
|
||||
identify_process_section_p (asection * section, bfd_boolean ms_style_implib)
|
||||
{
|
||||
static const char * SECTION_NAME =
|
||||
#ifdef DLLTOOL_PPC
|
||||
|
@ -3238,9 +3316,9 @@ identify_process_section_p (asection * section)
|
|||
".idata$7";
|
||||
#endif
|
||||
static const char * MS_SECTION_NAME = ".idata$6";
|
||||
|
||||
|
||||
const char * section_name =
|
||||
(identify_ms ? MS_SECTION_NAME : SECTION_NAME);
|
||||
(ms_style_implib ? MS_SECTION_NAME : SECTION_NAME);
|
||||
|
||||
if (strcmp (section_name, section->name) == 0)
|
||||
return TRUE;
|
||||
|
@ -3249,31 +3327,32 @@ identify_process_section_p (asection * section)
|
|||
|
||||
/* If *section has contents and its name is .idata$7 (.data$6 on PPC or if
|
||||
import lib ms-generated) -- and it satisfies several other constraints
|
||||
-- then store the contents in the list pointed to by
|
||||
identify_dll_name_list_head. */
|
||||
-- then add the contents of the section to obj->list. */
|
||||
|
||||
static void
|
||||
identify_search_section (bfd * abfd, asection * section, void * dummy ATTRIBUTE_UNUSED)
|
||||
identify_search_section (bfd * abfd, asection * section, void * obj)
|
||||
{
|
||||
bfd_byte *data = 0;
|
||||
bfd_size_type datasize;
|
||||
identify_data_type * identify_data = (identify_data_type *)obj;
|
||||
bfd_boolean ms_style = identify_data->ms_style_implib;
|
||||
|
||||
if ((section->flags & SEC_HAS_CONTENTS) == 0)
|
||||
return;
|
||||
|
||||
if (! identify_process_section_p (section))
|
||||
if (! identify_process_section_p (section, ms_style))
|
||||
return;
|
||||
|
||||
/* Binutils import libs seem distinguish the .idata$7 section that contains
|
||||
the DLL name from other .idata$7 sections by the absence of the
|
||||
SEC_RELOC flag. */
|
||||
if (!identify_ms && ((section->flags & SEC_RELOC) == SEC_RELOC))
|
||||
if (!ms_style && ((section->flags & SEC_RELOC) == SEC_RELOC))
|
||||
return;
|
||||
|
||||
/* MS import libs seem to distinguish the .idata$6 section
|
||||
that contains the DLL name from other .idata$6 sections
|
||||
by the presence of the SEC_DATA flag. */
|
||||
if (identify_ms && ((section->flags & SEC_DATA) == 0))
|
||||
if (ms_style && ((section->flags & SEC_DATA) == 0))
|
||||
return;
|
||||
|
||||
if ((datasize = bfd_section_size (abfd, section)) == 0)
|
||||
|
@ -3290,8 +3369,8 @@ identify_search_section (bfd * abfd, asection * section, void * dummy ATTRIBUTE_
|
|||
(more than 0x302f) imports, (b) it is an ms-style
|
||||
import library, but (c) it is buggy, in that the SEC_DATA
|
||||
flag is set on the "wrong" sections. This heuristic might
|
||||
also fail to record a valid dll name if the dllname is
|
||||
uses a multibyte or unicode character set (is that valid?).
|
||||
also fail to record a valid dll name if the dllname uses
|
||||
a multibyte or unicode character set (is that valid?).
|
||||
|
||||
This heuristic is based on the fact that symbols names in
|
||||
the chosen section -- as opposed to the dll name -- begin
|
||||
|
@ -3301,7 +3380,7 @@ identify_search_section (bfd * abfd, asection * section, void * dummy ATTRIBUTE_
|
|||
dll name does not contain unprintable characters. */
|
||||
if (data[0] != '\0' && ISPRINT (data[0])
|
||||
&& ((datasize < 2) || ISPRINT (data[1])))
|
||||
identify_append_dll_name_to_list (data);
|
||||
dll_name_list_append (identify_data->list, data);
|
||||
|
||||
free (data);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue