OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

Compare Revisions

  • This comparison shows the changes necessary to convert path
    /
    from Rev 287 to Rev 288
    Reverse comparison

Rev 287 → Rev 288

/openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.c
0,0 → 1,2103
/* Top-level LTO routines.
Copyright 2009, 2010 Free Software Foundation, Inc.
Contributed by CodeSourcery, Inc.
 
This file is part of GCC.
 
GCC is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free
Software Foundation; either version 3, or (at your option) any later
version.
 
GCC is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
 
You should have received a copy of the GNU General Public License
along with GCC; see the file COPYING3. If not see
<http://www.gnu.org/licenses/>. */
 
#include "config.h"
#include "system.h"
#include "coretypes.h"
#include "opts.h"
#include "toplev.h"
#include "tree.h"
#include "diagnostic.h"
#include "tm.h"
#include "libiberty.h"
#include "cgraph.h"
#include "ggc.h"
#include "tree-ssa-operands.h"
#include "tree-pass.h"
#include "langhooks.h"
#include "vec.h"
#include "bitmap.h"
#include "pointer-set.h"
#include "ipa-prop.h"
#include "common.h"
#include "timevar.h"
#include "gimple.h"
#include "lto.h"
#include "lto-tree.h"
#include "lto-streamer.h"
 
/* This needs to be included after config.h. Otherwise, _GNU_SOURCE will not
be defined in time to set __USE_GNU in the system headers, and strsignal
will not be declared. */
#if HAVE_MMAP_FILE
#include <sys/mman.h>
#endif
 
/* Handle opening elf files on hosts, such as Windows, that may use
text file handling that will break binary access. */
 
#ifndef O_BINARY
# define O_BINARY 0
#endif
 
 
DEF_VEC_P(bitmap);
DEF_VEC_ALLOC_P(bitmap,heap);
 
/* Read the constructors and inits. */
 
static void
lto_materialize_constructors_and_inits (struct lto_file_decl_data * file_data)
{
size_t len;
const char *data = lto_get_section_data (file_data,
LTO_section_static_initializer,
NULL, &len);
lto_input_constructors_and_inits (file_data, data);
lto_free_section_data (file_data, LTO_section_static_initializer, NULL,
data, len);
}
 
/* Read the function body for the function associated with NODE if possible. */
 
static void
lto_materialize_function (struct cgraph_node *node)
{
tree decl;
struct lto_file_decl_data *file_data;
const char *data, *name;
size_t len;
tree step;
 
/* Ignore clone nodes. Read the body only from the original one.
We may find clone nodes during LTRANS after WPA has made inlining
decisions. */
if (node->clone_of)
return;
 
decl = node->decl;
file_data = node->local.lto_file_data;
name = IDENTIFIER_POINTER (DECL_ASSEMBLER_NAME (decl));
 
/* We may have renamed the declaration, e.g., a static function. */
name = lto_get_decl_name_mapping (file_data, name);
 
data = lto_get_section_data (file_data, LTO_section_function_body,
name, &len);
if (data)
{
struct function *fn;
 
gcc_assert (!DECL_IS_BUILTIN (decl));
 
/* This function has a definition. */
TREE_STATIC (decl) = 1;
 
gcc_assert (DECL_STRUCT_FUNCTION (decl) == NULL);
allocate_struct_function (decl, false);
 
/* Load the function body only if not operating in WPA mode. In
WPA mode, the body of the function is not needed. */
if (!flag_wpa)
{
lto_input_function_body (file_data, decl, data);
lto_stats.num_function_bodies++;
}
 
fn = DECL_STRUCT_FUNCTION (decl);
lto_free_section_data (file_data, LTO_section_function_body, name,
data, len);
 
/* Look for initializers of constant variables and private
statics. */
for (step = fn->local_decls; step; step = TREE_CHAIN (step))
{
tree decl = TREE_VALUE (step);
if (TREE_CODE (decl) == VAR_DECL
&& (TREE_STATIC (decl) && !DECL_EXTERNAL (decl))
&& flag_unit_at_a_time)
varpool_finalize_decl (decl);
}
}
else
DECL_EXTERNAL (decl) = 1;
 
/* Let the middle end know about the function. */
rest_of_decl_compilation (decl, 1, 0);
if (cgraph_node (decl)->needed)
cgraph_mark_reachable_node (cgraph_node (decl));
}
 
 
/* Decode the content of memory pointed to by DATA in the the
in decl state object STATE. DATA_IN points to a data_in structure for
decoding. Return the address after the decoded object in the input. */
 
static const uint32_t *
lto_read_in_decl_state (struct data_in *data_in, const uint32_t *data,
struct lto_in_decl_state *state)
{
uint32_t ix;
tree decl;
uint32_t i, j;
ix = *data++;
decl = lto_streamer_cache_get (data_in->reader_cache, (int) ix);
if (TREE_CODE (decl) != FUNCTION_DECL)
{
gcc_assert (decl == void_type_node);
decl = NULL_TREE;
}
state->fn_decl = decl;
 
for (i = 0; i < LTO_N_DECL_STREAMS; i++)
{
uint32_t size = *data++;
tree *decls = (tree *) xcalloc (size, sizeof (tree));
 
for (j = 0; j < size; j++)
{
decls[j] = lto_streamer_cache_get (data_in->reader_cache, data[j]);
 
/* Register every type in the global type table. If the
type existed already, use the existing type. */
if (TYPE_P (decls[j]))
decls[j] = gimple_register_type (decls[j]);
}
 
state->streams[i].size = size;
state->streams[i].trees = decls;
data += size;
}
 
return data;
}
 
 
/* Read all the symbols from buffer DATA, using descriptors in DECL_DATA.
RESOLUTIONS is the set of symbols picked by the linker (read from the
resolution file when the linker plugin is being used). */
 
static void
lto_read_decls (struct lto_file_decl_data *decl_data, const void *data,
VEC(ld_plugin_symbol_resolution_t,heap) *resolutions)
{
const struct lto_decl_header *header = (const struct lto_decl_header *) data;
const int32_t decl_offset = sizeof (struct lto_decl_header);
const int32_t main_offset = decl_offset + header->decl_state_size;
const int32_t string_offset = main_offset + header->main_size;
struct lto_input_block ib_main;
struct data_in *data_in;
unsigned int i;
const uint32_t *data_ptr, *data_end;
uint32_t num_decl_states;
 
LTO_INIT_INPUT_BLOCK (ib_main, (const char *) data + main_offset, 0,
header->main_size);
 
data_in = lto_data_in_create (decl_data, (const char *) data + string_offset,
header->string_size, resolutions);
 
/* Read the global declarations and types. */
while (ib_main.p < ib_main.len)
{
tree t = lto_input_tree (&ib_main, data_in);
gcc_assert (t && ib_main.p <= ib_main.len);
}
 
/* Read in lto_in_decl_state objects. */
data_ptr = (const uint32_t *) ((const char*) data + decl_offset);
data_end =
(const uint32_t *) ((const char*) data_ptr + header->decl_state_size);
num_decl_states = *data_ptr++;
gcc_assert (num_decl_states > 0);
decl_data->global_decl_state = lto_new_in_decl_state ();
data_ptr = lto_read_in_decl_state (data_in, data_ptr,
decl_data->global_decl_state);
 
/* Read in per-function decl states and enter them in hash table. */
decl_data->function_decl_states =
htab_create (37, lto_hash_in_decl_state, lto_eq_in_decl_state, free);
 
for (i = 1; i < num_decl_states; i++)
{
struct lto_in_decl_state *state = lto_new_in_decl_state ();
void **slot;
 
data_ptr = lto_read_in_decl_state (data_in, data_ptr, state);
slot = htab_find_slot (decl_data->function_decl_states, state, INSERT);
gcc_assert (*slot == NULL);
*slot = state;
}
 
if (data_ptr != data_end)
internal_error ("bytecode stream: garbage at the end of symbols section");
/* Set the current decl state to be the global state. */
decl_data->current_decl_state = decl_data->global_decl_state;
 
lto_data_in_delete (data_in);
}
 
/* strtoll is not portable. */
int64_t
lto_parse_hex (const char *p) {
uint64_t ret = 0;
for (; *p != '\0'; ++p)
{
char c = *p;
unsigned char part;
ret <<= 4;
if (c >= '0' && c <= '9')
part = c - '0';
else if (c >= 'a' && c <= 'f')
part = c - 'a' + 10;
else if (c >= 'A' && c <= 'F')
part = c - 'A' + 10;
else
internal_error ("could not parse hex number");
ret |= part;
}
return ret;
}
 
/* Read resolution for file named FILE_NAME. The resolution is read from
RESOLUTION. An array with the symbol resolution is returned. The array
size is written to SIZE. */
 
static VEC(ld_plugin_symbol_resolution_t,heap) *
lto_resolution_read (FILE *resolution, lto_file *file)
{
/* We require that objects in the resolution file are in the same
order as the lto1 command line. */
unsigned int name_len;
char *obj_name;
unsigned int num_symbols;
unsigned int i;
VEC(ld_plugin_symbol_resolution_t,heap) *ret = NULL;
unsigned max_index = 0;
 
if (!resolution)
return NULL;
 
name_len = strlen (file->filename);
obj_name = XNEWVEC (char, name_len + 1);
fscanf (resolution, " "); /* Read white space. */
 
fread (obj_name, sizeof (char), name_len, resolution);
obj_name[name_len] = '\0';
if (strcmp (obj_name, file->filename) != 0)
internal_error ("unexpected file name %s in linker resolution file. "
"Expected %s", obj_name, file->filename);
if (file->offset != 0)
{
int t;
char offset_p[17];
int64_t offset;
t = fscanf (resolution, "@0x%16s", offset_p);
if (t != 1)
internal_error ("could not parse file offset");
offset = lto_parse_hex (offset_p);
if (offset != file->offset)
internal_error ("unexpected offset");
}
 
free (obj_name);
 
fscanf (resolution, "%u", &num_symbols);
 
for (i = 0; i < num_symbols; i++)
{
int t;
unsigned index;
char r_str[27];
enum ld_plugin_symbol_resolution r;
unsigned int j;
unsigned int lto_resolution_str_len =
sizeof (lto_resolution_str) / sizeof (char *);
 
t = fscanf (resolution, "%u %26s %*[^\n]\n", &index, r_str);
if (t != 2)
internal_error ("Invalid line in the resolution file.");
if (index > max_index)
max_index = index;
 
for (j = 0; j < lto_resolution_str_len; j++)
{
if (strcmp (lto_resolution_str[j], r_str) == 0)
{
r = (enum ld_plugin_symbol_resolution) j;
break;
}
}
if (j == lto_resolution_str_len)
internal_error ("Invalid resolution in the resolution file.");
 
VEC_safe_grow_cleared (ld_plugin_symbol_resolution_t, heap, ret,
max_index + 1);
VEC_replace (ld_plugin_symbol_resolution_t, ret, index, r);
}
 
return ret;
}
 
/* Generate a TREE representation for all types and external decls
entities in FILE.
 
Read all of the globals out of the file. Then read the cgraph
and process the .o index into the cgraph nodes so that it can open
the .o file to load the functions and ipa information. */
 
static struct lto_file_decl_data *
lto_file_read (lto_file *file, FILE *resolution_file)
{
struct lto_file_decl_data *file_data;
const char *data;
size_t len;
VEC(ld_plugin_symbol_resolution_t,heap) *resolutions;
resolutions = lto_resolution_read (resolution_file, file);
 
file_data = XCNEW (struct lto_file_decl_data);
file_data->file_name = file->filename;
file_data->section_hash_table = lto_obj_build_section_table (file);
file_data->renaming_hash_table = lto_create_renaming_table ();
 
data = lto_get_section_data (file_data, LTO_section_decls, NULL, &len);
lto_read_decls (file_data, data, resolutions);
lto_free_section_data (file_data, LTO_section_decls, NULL, data, len);
 
return file_data;
}
 
#if HAVE_MMAP_FILE && HAVE_SYSCONF && defined _SC_PAGE_SIZE
#define LTO_MMAP_IO 1
#endif
 
#if LTO_MMAP_IO
/* Page size of machine is used for mmap and munmap calls. */
static size_t page_mask;
#endif
 
/* Get the section data of length LEN from FILENAME starting at
OFFSET. The data segment must be freed by the caller when the
caller is finished. Returns NULL if all was not well. */
 
static char *
lto_read_section_data (struct lto_file_decl_data *file_data,
intptr_t offset, size_t len)
{
char *result;
static int fd = -1;
static char *fd_name;
#if LTO_MMAP_IO
intptr_t computed_len;
intptr_t computed_offset;
intptr_t diff;
#endif
 
/* Keep a single-entry file-descriptor cache. The last file we
touched will get closed at exit.
??? Eventually we want to add a more sophisticated larger cache
or rather fix function body streaming to not stream them in
practically random order. */
if (fd != -1
&& strcmp (fd_name, file_data->file_name) != 0)
{
free (fd_name);
close (fd);
fd = -1;
}
if (fd == -1)
{
fd_name = xstrdup (file_data->file_name);
fd = open (file_data->file_name, O_RDONLY|O_BINARY);
if (fd == -1)
return NULL;
}
 
#if LTO_MMAP_IO
if (!page_mask)
{
size_t page_size = sysconf (_SC_PAGE_SIZE);
page_mask = ~(page_size - 1);
}
 
computed_offset = offset & page_mask;
diff = offset - computed_offset;
computed_len = len + diff;
 
result = (char *) mmap (NULL, computed_len, PROT_READ, MAP_PRIVATE,
fd, computed_offset);
if (result == MAP_FAILED)
return NULL;
 
return result + diff;
#else
result = (char *) xmalloc (len);
if (lseek (fd, offset, SEEK_SET) != offset
|| read (fd, result, len) != (ssize_t) len)
{
free (result);
return NULL;
}
 
return result;
#endif
}
 
 
/* Get the section data from FILE_DATA of SECTION_TYPE with NAME.
NAME will be NULL unless the section type is for a function
body. */
 
static const char *
get_section_data (struct lto_file_decl_data *file_data,
enum lto_section_type section_type,
const char *name,
size_t *len)
{
htab_t section_hash_table = file_data->section_hash_table;
struct lto_section_slot *f_slot;
struct lto_section_slot s_slot;
const char *section_name = lto_get_section_name (section_type, name);
char *data = NULL;
 
*len = 0;
s_slot.name = section_name;
f_slot = (struct lto_section_slot *) htab_find (section_hash_table, &s_slot);
if (f_slot)
{
data = lto_read_section_data (file_data, f_slot->start, f_slot->len);
*len = f_slot->len;
}
 
free (CONST_CAST (char *, section_name));
return data;
}
 
 
/* Free the section data from FILE_DATA of SECTION_TYPE with NAME that
starts at OFFSET and has LEN bytes. */
 
static void
free_section_data (struct lto_file_decl_data *file_data ATTRIBUTE_UNUSED,
enum lto_section_type section_type ATTRIBUTE_UNUSED,
const char *name ATTRIBUTE_UNUSED,
const char *offset, size_t len ATTRIBUTE_UNUSED)
{
#if LTO_MMAP_IO
intptr_t computed_len;
intptr_t computed_offset;
intptr_t diff;
#endif
 
#if LTO_MMAP_IO
computed_offset = ((intptr_t) offset) & page_mask;
diff = (intptr_t) offset - computed_offset;
computed_len = len + diff;
 
munmap ((caddr_t) computed_offset, computed_len);
#else
free (CONST_CAST(char *, offset));
#endif
}
 
/* Vector of all cgraph node sets. */
static GTY (()) VEC(cgraph_node_set, gc) *lto_cgraph_node_sets;
 
 
/* Group cgrah nodes by input files. This is used mainly for testing
right now. */
 
static void
lto_1_to_1_map (void)
{
struct cgraph_node *node;
struct lto_file_decl_data *file_data;
struct pointer_map_t *pmap;
cgraph_node_set set;
void **slot;
 
timevar_push (TV_WHOPR_WPA);
 
lto_cgraph_node_sets = VEC_alloc (cgraph_node_set, gc, 1);
 
/* If the cgraph is empty, create one cgraph node set so that there is still
an output file for any variables that need to be exported in a DSO. */
if (!cgraph_nodes)
{
set = cgraph_node_set_new ();
VEC_safe_push (cgraph_node_set, gc, lto_cgraph_node_sets, set);
goto finish;
}
 
pmap = pointer_map_create ();
 
for (node = cgraph_nodes; node; node = node->next)
{
/* We only need to partition the nodes that we read from the
gimple bytecode files. */
file_data = node->local.lto_file_data;
if (file_data == NULL)
continue;
 
slot = pointer_map_contains (pmap, file_data);
if (slot)
set = (cgraph_node_set) *slot;
else
{
set = cgraph_node_set_new ();
slot = pointer_map_insert (pmap, file_data);
*slot = set;
VEC_safe_push (cgraph_node_set, gc, lto_cgraph_node_sets, set);
}
 
cgraph_node_set_add (set, node);
}
 
pointer_map_destroy (pmap);
 
finish:
timevar_pop (TV_WHOPR_WPA);
 
lto_stats.num_cgraph_partitions += VEC_length (cgraph_node_set,
lto_cgraph_node_sets);
}
 
 
/* Add inlined clone NODE and its master clone to SET, if NODE itself has
inlined callees, recursively add the callees. */
 
static void
lto_add_inline_clones (cgraph_node_set set, struct cgraph_node *node,
bitmap original_decls, bitmap inlined_decls)
{
struct cgraph_node *callee;
struct cgraph_edge *edge;
 
cgraph_node_set_add (set, node);
 
if (!bitmap_bit_p (original_decls, DECL_UID (node->decl)))
bitmap_set_bit (inlined_decls, DECL_UID (node->decl));
 
/* Check to see if NODE has any inlined callee. */
for (edge = node->callees; edge != NULL; edge = edge->next_callee)
{
callee = edge->callee;
if (callee->global.inlined_to != NULL)
lto_add_inline_clones (set, callee, original_decls, inlined_decls);
}
}
 
/* Compute the transitive closure of inlining of SET based on the
information in the callgraph. Returns a bitmap of decls that have
been inlined into SET indexed by UID. */
 
static bitmap
lto_add_all_inlinees (cgraph_node_set set)
{
cgraph_node_set_iterator csi;
struct cgraph_node *node;
bitmap original_nodes = lto_bitmap_alloc ();
bitmap original_decls = lto_bitmap_alloc ();
bitmap inlined_decls = lto_bitmap_alloc ();
bool changed;
 
/* We are going to iterate SET while adding to it, mark all original
nodes so that we only add node inlined to original nodes. */
for (csi = csi_start (set); !csi_end_p (csi); csi_next (&csi))
{
bitmap_set_bit (original_nodes, csi_node (csi)->uid);
bitmap_set_bit (original_decls, DECL_UID (csi_node (csi)->decl));
}
 
/* Some of the original nodes might not be needed anymore.
Remove them. */
do
{
changed = false;
for (csi = csi_start (set); !csi_end_p (csi); csi_next (&csi))
{
struct cgraph_node *inlined_to;
node = csi_node (csi);
 
/* NODE was not inlined. We still need it. */
if (!node->global.inlined_to)
continue;
 
inlined_to = node->global.inlined_to;
 
/* NODE should have only one caller. */
gcc_assert (!node->callers->next_caller);
 
if (!bitmap_bit_p (original_nodes, inlined_to->uid))
{
bitmap_clear_bit (original_nodes, node->uid);
cgraph_node_set_remove (set, node);
changed = true;
}
}
}
while (changed);
 
/* Transitively add to SET all the inline clones for every node that
has been inlined. */
for (csi = csi_start (set); !csi_end_p (csi); csi_next (&csi))
{
node = csi_node (csi);
if (bitmap_bit_p (original_nodes, node->uid))
lto_add_inline_clones (set, node, original_decls, inlined_decls);
}
 
lto_bitmap_free (original_nodes);
lto_bitmap_free (original_decls);
 
return inlined_decls;
}
 
/* Owing to inlining, we may need to promote a file-scope variable
to a global variable. Consider this case:
 
a.c:
static int var;
 
void
foo (void)
{
var++;
}
 
b.c:
 
extern void foo (void);
 
void
bar (void)
{
foo ();
}
 
If WPA inlines FOO inside BAR, then the static variable VAR needs to
be promoted to global because BAR and VAR may be in different LTRANS
files. */
 
/* This struct keeps track of states used in globalization. */
 
typedef struct
{
/* Current cgraph node set. */
cgraph_node_set set;
 
/* Function DECLs of cgraph nodes seen. */
bitmap seen_node_decls;
 
/* Use in walk_tree to avoid multiple visits of a node. */
struct pointer_set_t *visited;
 
/* static vars in this set. */
bitmap static_vars_in_set;
 
/* static vars in all previous set. */
bitmap all_static_vars;
 
/* all vars in all previous set. */
bitmap all_vars;
} globalize_context_t;
 
/* Callback for walk_tree. Examine the tree pointer to by TP and see if
if its a file-scope static variable of function that need to be turned
into a global. */
 
static tree
globalize_cross_file_statics (tree *tp, int *walk_subtrees ATTRIBUTE_UNUSED,
void *data)
{
globalize_context_t *context = (globalize_context_t *) data;
tree t = *tp;
 
if (t == NULL_TREE)
return NULL;
 
/* The logic for globalization of VAR_DECLs and FUNCTION_DECLs are
different. For functions, we can simply look at the cgraph node sets
to tell if there are references to static functions outside the set.
The cgraph node sets do not keep track of vars, we need to traverse
the trees to determine what vars need to be globalized. */
if (TREE_CODE (t) == VAR_DECL)
{
if (!TREE_PUBLIC (t))
{
/* This file-scope static variable is reachable from more
that one set. Make it global but with hidden visibility
so that we do not export it in dynamic linking. */
if (bitmap_bit_p (context->all_static_vars, DECL_UID (t)))
{
TREE_PUBLIC (t) = 1;
DECL_VISIBILITY (t) = VISIBILITY_HIDDEN;
}
bitmap_set_bit (context->static_vars_in_set, DECL_UID (t));
}
bitmap_set_bit (context->all_vars, DECL_UID (t));
walk_tree (&DECL_INITIAL (t), globalize_cross_file_statics, context,
context->visited);
}
else if (TREE_CODE (t) == FUNCTION_DECL && !TREE_PUBLIC (t))
{
if (!cgraph_node_in_set_p (cgraph_node (t), context->set))
{
/* This file-scope static function is reachable from a set
which does not contain the function DECL. Make it global
but with hidden visibility. */
TREE_PUBLIC (t) = 1;
DECL_VISIBILITY (t) = VISIBILITY_HIDDEN;
}
}
 
return NULL;
}
 
/* Helper of lto_scan_statics_in_cgraph_node below. Scan TABLE for
static decls that may be used in more than one LTRANS file.
CONTEXT is a globalize_context_t for storing scanning states. */
 
static void
lto_scan_statics_in_ref_table (struct lto_tree_ref_table *table,
globalize_context_t *context)
{
unsigned i;
 
for (i = 0; i < table->size; i++)
walk_tree (&table->trees[i], globalize_cross_file_statics, context,
context->visited);
}
 
/* Promote file-scope decl reachable from NODE if necessary to global.
CONTEXT is a globalize_context_t storing scanning states. */
 
static void
lto_scan_statics_in_cgraph_node (struct cgraph_node *node,
globalize_context_t *context)
{
struct lto_in_decl_state *state;
/* Do nothing if NODE has no function body. */
if (!node->analyzed)
return;
/* Return if the DECL of nodes has been visited before. */
if (bitmap_bit_p (context->seen_node_decls, DECL_UID (node->decl)))
return;
 
bitmap_set_bit (context->seen_node_decls, DECL_UID (node->decl));
 
state = lto_get_function_in_decl_state (node->local.lto_file_data,
node->decl);
gcc_assert (state);
 
lto_scan_statics_in_ref_table (&state->streams[LTO_DECL_STREAM_VAR_DECL],
context);
lto_scan_statics_in_ref_table (&state->streams[LTO_DECL_STREAM_FN_DECL],
context);
}
 
/* Scan all global variables that we have not yet seen so far. CONTEXT
is a globalize_context_t storing scanning states. */
 
static void
lto_scan_statics_in_remaining_global_vars (globalize_context_t *context)
{
tree var, var_context;
struct varpool_node *vnode;
 
FOR_EACH_STATIC_VARIABLE (vnode)
{
var = vnode->decl;
var_context = DECL_CONTEXT (var);
if (TREE_STATIC (var)
&& TREE_PUBLIC (var)
&& (!var_context || TREE_CODE (var_context) != FUNCTION_DECL)
&& !bitmap_bit_p (context->all_vars, DECL_UID (var)))
walk_tree (&var, globalize_cross_file_statics, context,
context->visited);
}
}
 
/* Find out all static decls that need to be promoted to global because
of cross file sharing. This function must be run in the WPA mode after
all inlinees are added. */
 
static void
lto_promote_cross_file_statics (void)
{
unsigned i, n_sets;
cgraph_node_set set;
cgraph_node_set_iterator csi;
globalize_context_t context;
 
memset (&context, 0, sizeof (context));
context.all_vars = lto_bitmap_alloc ();
context.all_static_vars = lto_bitmap_alloc ();
 
n_sets = VEC_length (cgraph_node_set, lto_cgraph_node_sets);
for (i = 0; i < n_sets; i++)
{
set = VEC_index (cgraph_node_set, lto_cgraph_node_sets, i);
context.set = set;
context.visited = pointer_set_create ();
context.static_vars_in_set = lto_bitmap_alloc ();
context.seen_node_decls = lto_bitmap_alloc ();
 
for (csi = csi_start (set); !csi_end_p (csi); csi_next (&csi))
lto_scan_statics_in_cgraph_node (csi_node (csi), &context);
 
if (i == n_sets - 1)
lto_scan_statics_in_remaining_global_vars (&context);
 
bitmap_ior_into (context.all_static_vars, context.static_vars_in_set);
 
pointer_set_destroy (context.visited);
lto_bitmap_free (context.static_vars_in_set);
lto_bitmap_free (context.seen_node_decls);
}
 
lto_bitmap_free (context.all_vars);
lto_bitmap_free (context.all_static_vars);
}
 
 
/* Given a file name FNAME, return a string with FNAME prefixed with '*'. */
 
static char *
prefix_name_with_star (const char *fname)
{
char *star_fname;
size_t len;
len = strlen (fname) + 1 + 1;
star_fname = XNEWVEC (char, len);
snprintf (star_fname, len, "*%s", fname);
 
return star_fname;
}
 
 
/* Return a copy of FNAME without the .o extension. */
 
static char *
strip_extension (const char *fname)
{
char *s = XNEWVEC (char, strlen (fname) - 2 + 1);
gcc_assert (strstr (fname, ".o"));
snprintf (s, strlen (fname) - 2 + 1, "%s", fname);
 
return s;
}
 
 
/* Return a file name associated with cgraph node set SET. This may
be a new temporary file name if SET needs to be processed by
LTRANS, or the original file name if all the nodes in SET belong to
the same input file. */
 
static char *
get_filename_for_set (cgraph_node_set set)
{
char *fname = NULL;
static const size_t max_fname_len = 100;
 
if (cgraph_node_set_needs_ltrans_p (set))
{
/* Create a new temporary file to store SET. To facilitate
debugging, use file names from SET as part of the new
temporary file name. */
cgraph_node_set_iterator si;
struct pointer_set_t *pset = pointer_set_create ();
for (si = csi_start (set); !csi_end_p (si); csi_next (&si))
{
struct cgraph_node *n = csi_node (si);
const char *node_fname;
char *f;
 
/* Don't use the same file name more than once. */
if (pointer_set_insert (pset, n->local.lto_file_data))
continue;
 
/* The first file name found in SET determines the output
directory. For the remaining files, we use their
base names. */
node_fname = n->local.lto_file_data->file_name;
if (fname == NULL)
{
fname = strip_extension (node_fname);
continue;
}
 
f = strip_extension (lbasename (node_fname));
 
/* If the new name causes an excessively long file name,
make the last component "___" to indicate overflow. */
if (strlen (fname) + strlen (f) > max_fname_len - 3)
{
fname = reconcat (fname, fname, "___", NULL);
break;
}
else
{
fname = reconcat (fname, fname, "_", f, NULL);
free (f);
}
}
 
pointer_set_destroy (pset);
 
/* Add the extension .wpa.o to indicate that this file has been
produced by WPA. */
fname = reconcat (fname, fname, ".wpa.o", NULL);
gcc_assert (fname);
}
else
{
/* Since SET does not need to be processed by LTRANS, use
the original file name and mark it with a '*' prefix so that
lto_execute_ltrans knows not to process it. */
cgraph_node_set_iterator si = csi_start (set);
struct cgraph_node *first = csi_node (si);
fname = prefix_name_with_star (first->local.lto_file_data->file_name);
}
 
return fname;
}
 
static lto_file *current_lto_file;
 
 
/* Write all output files in WPA mode. Returns a NULL-terminated array of
output file names. */
 
static char **
lto_wpa_write_files (void)
{
char **output_files;
unsigned i, n_sets, last_out_file_ix, num_out_files;
lto_file *file;
cgraph_node_set set;
bitmap decls;
VEC(bitmap,heap) *inlined_decls = NULL;
 
timevar_push (TV_WHOPR_WPA);
 
/* Include all inlined functions and determine what sets need to be
compiled by LTRANS. After this loop, only those sets that
contain callgraph nodes from more than one file will need to be
compiled by LTRANS. */
for (i = 0; VEC_iterate (cgraph_node_set, lto_cgraph_node_sets, i, set); i++)
{
decls = lto_add_all_inlinees (set);
VEC_safe_push (bitmap, heap, inlined_decls, decls);
lto_stats.num_output_cgraph_nodes += VEC_length (cgraph_node_ptr,
set->nodes);
}
 
/* After adding all inlinees, find out statics that need to be promoted
to globals because of cross-file inlining. */
lto_promote_cross_file_statics ();
 
timevar_pop (TV_WHOPR_WPA);
 
timevar_push (TV_WHOPR_WPA_IO);
 
/* The number of output files depends on the number of input files
and how many callgraph node sets we create. Reserve enough space
for the maximum of these two. */
num_out_files = MAX (VEC_length (cgraph_node_set, lto_cgraph_node_sets),
num_in_fnames);
output_files = XNEWVEC (char *, num_out_files + 1);
 
n_sets = VEC_length (cgraph_node_set, lto_cgraph_node_sets);
for (i = 0; i < n_sets; i++)
{
char *temp_filename;
 
set = VEC_index (cgraph_node_set, lto_cgraph_node_sets, i);
temp_filename = get_filename_for_set (set);
output_files[i] = temp_filename;
 
if (cgraph_node_set_needs_ltrans_p (set))
{
/* Write all the nodes in SET to TEMP_FILENAME. */
file = lto_obj_file_open (temp_filename, true);
if (!file)
fatal_error ("lto_obj_file_open() failed");
 
lto_set_current_out_file (file);
lto_new_extern_inline_states ();
 
decls = VEC_index (bitmap, inlined_decls, i);
lto_force_functions_extern_inline (decls);
 
ipa_write_summaries_of_cgraph_node_set (set);
lto_delete_extern_inline_states ();
 
lto_set_current_out_file (NULL);
lto_obj_file_close (file);
}
}
 
last_out_file_ix = n_sets;
 
lto_stats.num_output_files += n_sets;
 
output_files[last_out_file_ix] = NULL;
 
for (i = 0; VEC_iterate (bitmap, inlined_decls, i, decls); i++)
lto_bitmap_free (decls);
VEC_free (bitmap, heap, inlined_decls);
 
timevar_pop (TV_WHOPR_WPA_IO);
 
return output_files;
}
 
/* Template of LTRANS dumpbase suffix. */
#define DUMPBASE_SUFFIX ".ltrans18446744073709551615"
 
/* Perform local transformations (LTRANS) on the files in the NULL-terminated
FILES array. These should have been written previously by
lto_wpa_write_files (). Transformations are performed via executing
COLLECT_GCC for reach file. */
 
static void
lto_execute_ltrans (char *const *files)
{
struct pex_obj *pex;
const char *collect_gcc_options, *collect_gcc;
struct obstack env_obstack;
const char **argv;
const char **argv_ptr;
const char *errmsg;
size_t i, j;
int err;
int status;
FILE *ltrans_output_list_stream = NULL;
bool seen_dumpbase = false;
char *dumpbase_suffix = NULL;
 
timevar_push (TV_WHOPR_WPA_LTRANS_EXEC);
 
/* Get the driver and options. */
collect_gcc = getenv ("COLLECT_GCC");
if (!collect_gcc)
fatal_error ("environment variable COLLECT_GCC must be set");
 
/* Set the CFLAGS environment variable. */
collect_gcc_options = getenv ("COLLECT_GCC_OPTIONS");
if (!collect_gcc_options)
fatal_error ("environment variable COLLECT_GCC_OPTIONS must be set");
 
/* Count arguments. */
i = 0;
for (j = 0; collect_gcc_options[j] != '\0'; ++j)
if (collect_gcc_options[j] == '\'')
++i;
 
if (i % 2 != 0)
fatal_error ("malformed COLLECT_GCC_OPTIONS");
 
/* Initalize the arguments for the LTRANS driver. */
argv = XNEWVEC (const char *, 8 + i / 2);
argv_ptr = argv;
*argv_ptr++ = collect_gcc;
*argv_ptr++ = "-xlto";
for (j = 0; collect_gcc_options[j] != '\0'; ++j)
if (collect_gcc_options[j] == '\'')
{
char *option;
 
++j;
i = j;
while (collect_gcc_options[j] != '\'')
++j;
obstack_init (&env_obstack);
obstack_grow (&env_obstack, &collect_gcc_options[i], j - i);
if (seen_dumpbase)
obstack_grow (&env_obstack, DUMPBASE_SUFFIX,
sizeof (DUMPBASE_SUFFIX));
else
obstack_1grow (&env_obstack, 0);
option = XOBFINISH (&env_obstack, char *);
if (seen_dumpbase)
{
dumpbase_suffix = option + 7 + j - i;
seen_dumpbase = false;
}
 
/* LTRANS does not need -fwpa nor -fltrans-*. */
if (strncmp (option, "-fwpa", 5) != 0
&& strncmp (option, "-fltrans-", 9) != 0)
{
if (strncmp (option, "-dumpbase", 9) == 0)
seen_dumpbase = true;
*argv_ptr++ = option;
}
}
*argv_ptr++ = "-fltrans";
 
/* Open the LTRANS output list. */
if (ltrans_output_list)
{
ltrans_output_list_stream = fopen (ltrans_output_list, "w");
if (ltrans_output_list_stream == NULL)
error ("opening LTRANS output list %s: %m", ltrans_output_list);
}
 
for (i = 0; files[i]; ++i)
{
size_t len;
 
/* If the file is prefixed with a '*', it means that we do not
need to re-compile it with LTRANS because it has not been
modified by WPA. Skip it from the command line to
lto_execute_ltrans, but add it to ltrans_output_list_stream
so it is linked after we are done. */
if (files[i][0] == '*')
{
size_t len = strlen (files[i]) - 1;
if (ltrans_output_list_stream)
if (fwrite (&files[i][1], 1, len, ltrans_output_list_stream) < len
|| fwrite ("\n", 1, 1, ltrans_output_list_stream) < 1)
error ("writing to LTRANS output list %s: %m",
ltrans_output_list);
}
else
{
char *output_name;
 
/* Otherwise, add FILES[I] to lto_execute_ltrans command line
and add the resulting file to LTRANS output list. */
 
/* Replace the .o suffix with a .ltrans.o suffix and write
the resulting name to the LTRANS output list. */
obstack_init (&env_obstack);
obstack_grow (&env_obstack, files[i], strlen (files[i]) - 2);
obstack_grow (&env_obstack, ".ltrans.o", sizeof (".ltrans.o"));
output_name = XOBFINISH (&env_obstack, char *);
if (ltrans_output_list_stream)
{
len = strlen (output_name);
 
if (fwrite (output_name, 1, len, ltrans_output_list_stream) < len
|| fwrite ("\n", 1, 1, ltrans_output_list_stream) < 1)
error ("writing to LTRANS output list %s: %m",
ltrans_output_list);
}
 
argv_ptr[0] = "-o";
argv_ptr[1] = output_name;
argv_ptr[2] = files[i];
argv_ptr[3] = NULL;
 
/* Append a sequence number to -dumpbase for LTRANS. */
if (dumpbase_suffix)
snprintf (dumpbase_suffix, sizeof (DUMPBASE_SUFFIX) - 7,
"%lu", (unsigned long) i);
 
/* Execute the driver. */
pex = pex_init (0, "lto1", NULL);
if (pex == NULL)
fatal_error ("pex_init failed: %s", xstrerror (errno));
 
errmsg = pex_run (pex, PEX_LAST | PEX_SEARCH, argv[0],
CONST_CAST (char **, argv), NULL, NULL, &err);
if (errmsg)
fatal_error ("%s: %s", errmsg, xstrerror (err));
 
if (!pex_get_status (pex, 1, &status))
fatal_error ("can't get program status: %s", xstrerror (errno));
 
if (status)
{
if (WIFSIGNALED (status))
{
int sig = WTERMSIG (status);
fatal_error ("%s terminated with signal %d [%s]%s",
argv[0], sig, strsignal (sig),
WCOREDUMP (status) ? ", core dumped" : "");
}
else
fatal_error ("%s terminated with status %d", argv[0], status);
}
 
pex_free (pex);
}
}
 
/* Close the LTRANS output list. */
if (ltrans_output_list_stream && fclose (ltrans_output_list_stream))
error ("closing LTRANS output list %s: %m", ltrans_output_list);
 
obstack_free (&env_obstack, NULL);
free (argv);
 
timevar_pop (TV_WHOPR_WPA_LTRANS_EXEC);
}
 
 
typedef struct {
struct pointer_set_t *seen;
} lto_fixup_data_t;
 
#define LTO_FIXUP_SUBTREE(t) \
do \
walk_tree (&(t), lto_fixup_tree, data, NULL); \
while (0)
 
#define LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE(t) \
do \
{ \
if (t) \
(t) = gimple_register_type (t); \
walk_tree (&(t), lto_fixup_tree, data, NULL); \
} \
while (0)
 
static tree lto_fixup_tree (tree *, int *, void *);
 
/* Return true if T does not need to be fixed up recursively. */
 
static inline bool
no_fixup_p (tree t)
{
return (t == NULL
|| CONSTANT_CLASS_P (t)
|| TREE_CODE (t) == IDENTIFIER_NODE);
}
 
/* Fix up fields of a tree_common T. DATA points to fix-up states. */
 
static void
lto_fixup_common (tree t, void *data)
{
/* The following re-creates the TYPE_REFERENCE_TO and TYPE_POINTER_TO
lists. We do not stream TYPE_REFERENCE_TO, TYPE_POINTER_TO or
TYPE_NEXT_PTR_TO and TYPE_NEXT_REF_TO.
First remove us from any pointer list we are on. */
if (TREE_CODE (t) == POINTER_TYPE)
{
if (TYPE_POINTER_TO (TREE_TYPE (t)) == t)
TYPE_POINTER_TO (TREE_TYPE (t)) = TYPE_NEXT_PTR_TO (t);
else
{
tree tem = TYPE_POINTER_TO (TREE_TYPE (t));
while (tem && TYPE_NEXT_PTR_TO (tem) != t)
tem = TYPE_NEXT_PTR_TO (tem);
if (tem)
TYPE_NEXT_PTR_TO (tem) = TYPE_NEXT_PTR_TO (t);
}
TYPE_NEXT_PTR_TO (t) = NULL_TREE;
}
else if (TREE_CODE (t) == REFERENCE_TYPE)
{
if (TYPE_REFERENCE_TO (TREE_TYPE (t)) == t)
TYPE_REFERENCE_TO (TREE_TYPE (t)) = TYPE_NEXT_REF_TO (t);
else
{
tree tem = TYPE_REFERENCE_TO (TREE_TYPE (t));
while (tem && TYPE_NEXT_REF_TO (tem) != t)
tem = TYPE_NEXT_REF_TO (tem);
if (tem)
TYPE_NEXT_REF_TO (tem) = TYPE_NEXT_REF_TO (t);
}
TYPE_NEXT_REF_TO (t) = NULL_TREE;
}
 
/* Fixup our type. */
LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE (TREE_TYPE (t));
 
/* Second put us on the list of pointers of the new pointed-to type
if we are a main variant. This is done in lto_fixup_type after
fixing up our main variant. */
 
/* This is not very efficient because we cannot do tail-recursion with
a long chain of trees. */
LTO_FIXUP_SUBTREE (TREE_CHAIN (t));
}
 
/* Fix up fields of a decl_minimal T. DATA points to fix-up states. */
 
static void
lto_fixup_decl_minimal (tree t, void *data)
{
lto_fixup_common (t, data);
LTO_FIXUP_SUBTREE (DECL_NAME (t));
LTO_FIXUP_SUBTREE (DECL_CONTEXT (t));
}
 
/* Fix up fields of a decl_common T. DATA points to fix-up states. */
 
static void
lto_fixup_decl_common (tree t, void *data)
{
lto_fixup_decl_minimal (t, data);
LTO_FIXUP_SUBTREE (DECL_SIZE (t));
LTO_FIXUP_SUBTREE (DECL_SIZE_UNIT (t));
LTO_FIXUP_SUBTREE (DECL_INITIAL (t));
LTO_FIXUP_SUBTREE (DECL_ATTRIBUTES (t));
LTO_FIXUP_SUBTREE (DECL_ABSTRACT_ORIGIN (t));
}
 
/* Fix up fields of a decl_with_vis T. DATA points to fix-up states. */
 
static void
lto_fixup_decl_with_vis (tree t, void *data)
{
lto_fixup_decl_common (t, data);
 
/* Accessor macro has side-effects, use field-name here. */
LTO_FIXUP_SUBTREE (t->decl_with_vis.assembler_name);
 
gcc_assert (no_fixup_p (DECL_SECTION_NAME (t)));
}
 
/* Fix up fields of a decl_non_common T. DATA points to fix-up states. */
 
static void
lto_fixup_decl_non_common (tree t, void *data)
{
lto_fixup_decl_with_vis (t, data);
LTO_FIXUP_SUBTREE (DECL_ARGUMENT_FLD (t));
LTO_FIXUP_SUBTREE (DECL_RESULT_FLD (t));
LTO_FIXUP_SUBTREE (DECL_VINDEX (t));
 
/* SAVED_TREE should not cleared by now. Also no accessor for base type. */
gcc_assert (no_fixup_p (t->decl_non_common.saved_tree));
}
 
/* Fix up fields of a decl_non_common T. DATA points to fix-up states. */
 
static void
lto_fixup_function (tree t, void *data)
{
lto_fixup_decl_non_common (t, data);
LTO_FIXUP_SUBTREE (DECL_FUNCTION_PERSONALITY (t));
}
 
/* Fix up fields of a field_decl T. DATA points to fix-up states. */
 
static void
lto_fixup_field_decl (tree t, void *data)
{
lto_fixup_decl_common (t, data);
LTO_FIXUP_SUBTREE (DECL_FIELD_OFFSET (t));
LTO_FIXUP_SUBTREE (DECL_BIT_FIELD_TYPE (t));
LTO_FIXUP_SUBTREE (DECL_QUALIFIER (t));
gcc_assert (no_fixup_p (DECL_FIELD_BIT_OFFSET (t)));
LTO_FIXUP_SUBTREE (DECL_FCONTEXT (t));
}
 
/* Fix up fields of a type T. DATA points to fix-up states. */
 
static void
lto_fixup_type (tree t, void *data)
{
tree tem, mv;
 
lto_fixup_common (t, data);
LTO_FIXUP_SUBTREE (TYPE_CACHED_VALUES (t));
LTO_FIXUP_SUBTREE (TYPE_SIZE (t));
LTO_FIXUP_SUBTREE (TYPE_SIZE_UNIT (t));
LTO_FIXUP_SUBTREE (TYPE_ATTRIBUTES (t));
LTO_FIXUP_SUBTREE (TYPE_NAME (t));
 
/* Accessors are for derived node types only. */
if (!POINTER_TYPE_P (t))
LTO_FIXUP_SUBTREE (t->type.minval);
LTO_FIXUP_SUBTREE (t->type.maxval);
 
/* Accessor is for derived node types only. */
LTO_FIXUP_SUBTREE (t->type.binfo);
 
if (TYPE_CONTEXT (t))
{
if (TYPE_P (TYPE_CONTEXT (t)))
LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE (TYPE_CONTEXT (t));
else
LTO_FIXUP_SUBTREE (TYPE_CONTEXT (t));
}
LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE (TYPE_CANONICAL (t));
 
/* The following re-creates proper variant lists while fixing up
the variant leaders. We do not stream TYPE_NEXT_VARIANT so the
variant list state before fixup is broken. */
 
/* Remove us from our main variant list if we are not the variant leader. */
if (TYPE_MAIN_VARIANT (t) != t)
{
tem = TYPE_MAIN_VARIANT (t);
while (tem && TYPE_NEXT_VARIANT (tem) != t)
tem = TYPE_NEXT_VARIANT (tem);
if (tem)
TYPE_NEXT_VARIANT (tem) = TYPE_NEXT_VARIANT (t);
TYPE_NEXT_VARIANT (t) = NULL_TREE;
}
 
/* Query our new main variant. */
mv = gimple_register_type (TYPE_MAIN_VARIANT (t));
 
/* If we were the variant leader and we get replaced ourselves drop
all variants from our list. */
if (TYPE_MAIN_VARIANT (t) == t
&& mv != t)
{
tem = t;
while (tem)
{
tree tem2 = TYPE_NEXT_VARIANT (tem);
TYPE_NEXT_VARIANT (tem) = NULL_TREE;
tem = tem2;
}
}
 
/* If we are not our own variant leader link us into our new leaders
variant list. */
if (mv != t)
{
TYPE_NEXT_VARIANT (t) = TYPE_NEXT_VARIANT (mv);
TYPE_NEXT_VARIANT (mv) = t;
}
 
/* Finally adjust our main variant and fix it up. */
TYPE_MAIN_VARIANT (t) = mv;
LTO_FIXUP_SUBTREE (TYPE_MAIN_VARIANT (t));
 
/* As the second step of reconstructing the pointer chains put us
on the list of pointers of the new pointed-to type
if we are a main variant. See lto_fixup_common for the first step. */
if (TREE_CODE (t) == POINTER_TYPE
&& TYPE_MAIN_VARIANT (t) == t)
{
TYPE_NEXT_PTR_TO (t) = TYPE_POINTER_TO (TREE_TYPE (t));
TYPE_POINTER_TO (TREE_TYPE (t)) = t;
}
else if (TREE_CODE (t) == REFERENCE_TYPE
&& TYPE_MAIN_VARIANT (t) == t)
{
TYPE_NEXT_REF_TO (t) = TYPE_REFERENCE_TO (TREE_TYPE (t));
TYPE_REFERENCE_TO (TREE_TYPE (t)) = t;
}
}
 
/* Fix up fields of a BINFO T. DATA points to fix-up states. */
 
static void
lto_fixup_binfo (tree t, void *data)
{
unsigned HOST_WIDE_INT i, n;
tree base, saved_base;
 
lto_fixup_common (t, data);
gcc_assert (no_fixup_p (BINFO_OFFSET (t)));
LTO_FIXUP_SUBTREE (BINFO_VTABLE (t));
LTO_FIXUP_SUBTREE (BINFO_VIRTUALS (t));
LTO_FIXUP_SUBTREE (BINFO_VPTR_FIELD (t));
n = VEC_length (tree, BINFO_BASE_ACCESSES (t));
for (i = 0; i < n; i++)
{
saved_base = base = BINFO_BASE_ACCESS (t, i);
LTO_FIXUP_SUBTREE (base);
if (base != saved_base)
VEC_replace (tree, BINFO_BASE_ACCESSES (t), i, base);
}
LTO_FIXUP_SUBTREE (BINFO_INHERITANCE_CHAIN (t));
LTO_FIXUP_SUBTREE (BINFO_SUBVTT_INDEX (t));
LTO_FIXUP_SUBTREE (BINFO_VPTR_INDEX (t));
n = BINFO_N_BASE_BINFOS (t);
for (i = 0; i < n; i++)
{
saved_base = base = BINFO_BASE_BINFO (t, i);
LTO_FIXUP_SUBTREE (base);
if (base != saved_base)
VEC_replace (tree, BINFO_BASE_BINFOS (t), i, base);
}
}
 
/* Fix up fields of a CONSTRUCTOR T. DATA points to fix-up states. */
 
static void
lto_fixup_constructor (tree t, void *data)
{
unsigned HOST_WIDE_INT idx;
constructor_elt *ce;
 
LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE (TREE_TYPE (t));
 
for (idx = 0;
VEC_iterate(constructor_elt, CONSTRUCTOR_ELTS (t), idx, ce);
idx++)
{
LTO_FIXUP_SUBTREE (ce->index);
LTO_FIXUP_SUBTREE (ce->value);
}
}
 
/* A walk_tree callback used by lto_fixup_state. TP is the pointer to the
current tree. WALK_SUBTREES indicates if the subtrees will be walked.
DATA is a pointer set to record visited nodes. */
 
static tree
lto_fixup_tree (tree *tp, int *walk_subtrees, void *data)
{
tree t;
lto_fixup_data_t *fixup_data = (lto_fixup_data_t *) data;
tree prevailing;
 
t = *tp;
*walk_subtrees = 0;
if (pointer_set_contains (fixup_data->seen, t))
return NULL;
 
if (TREE_CODE (t) == VAR_DECL || TREE_CODE (t) == FUNCTION_DECL)
{
prevailing = lto_symtab_prevailing_decl (t);
 
if (t != prevailing)
{
if (TREE_CODE (t) == FUNCTION_DECL
&& TREE_NOTHROW (prevailing) != TREE_NOTHROW (t))
{
/* If the prevailing definition does not throw but the
declaration (T) was considered throwing, then we
simply add PREVAILING to the list of throwing
functions. However, if the opposite is true, then
the call to PREVAILING was generated assuming that
the function didn't throw, which means that CFG
cleanup may have removed surrounding try/catch
regions.
 
Note that we currently accept these cases even when
they occur within a single file. It's certainly a
user error, but we silently allow the compiler to
remove surrounding try/catch regions. Perhaps we
could emit a warning here, instead of silently
accepting the conflicting declaration. */
if (TREE_NOTHROW (prevailing))
lto_mark_nothrow_fndecl (prevailing);
}
 
/* Also replace t with prevailing defintion. We don't want to
insert the other defintion in the seen set as we want to
replace all instances of it. */
*tp = prevailing;
t = prevailing;
}
}
else if (TYPE_P (t))
{
/* Replace t with the prevailing type. We don't want to insert the
other type in the seen set as we want to replace all instances of it. */
t = gimple_register_type (t);
*tp = t;
}
 
if (pointer_set_insert (fixup_data->seen, t))
return NULL;
 
/* walk_tree does not visit all reachable nodes that need to be fixed up.
Hence we do special processing here for those kind of nodes. */
switch (TREE_CODE (t))
{
case FIELD_DECL:
lto_fixup_field_decl (t, data);
break;
 
case LABEL_DECL:
case CONST_DECL:
case PARM_DECL:
case RESULT_DECL:
case IMPORTED_DECL:
lto_fixup_decl_common (t, data);
break;
 
case VAR_DECL:
lto_fixup_decl_with_vis (t, data);
break;
 
case TYPE_DECL:
lto_fixup_decl_non_common (t, data);
break;
 
case FUNCTION_DECL:
lto_fixup_function (t, data);
break;
 
case TREE_BINFO:
lto_fixup_binfo (t, data);
break;
 
default:
if (TYPE_P (t))
lto_fixup_type (t, data);
else if (TREE_CODE (t) == CONSTRUCTOR)
lto_fixup_constructor (t, data);
else if (CONSTANT_CLASS_P (t))
LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE (TREE_TYPE (t));
else if (EXPR_P (t))
{
/* walk_tree only handles TREE_OPERANDs. Do the rest here. */
lto_fixup_common (t, data);
LTO_FIXUP_SUBTREE (t->exp.block);
*walk_subtrees = 1;
}
else
{
/* Let walk_tree handle sub-trees. */
*walk_subtrees = 1;
}
}
 
return NULL;
}
 
/* Helper function of lto_fixup_decls. Walks the var and fn streams in STATE,
replaces var and function decls with the corresponding prevailing def and
records the old decl in the free-list in DATA. We also record visted nodes
in the seen-set in DATA to avoid multiple visit for nodes that need not
to be replaced. */
 
static void
lto_fixup_state (struct lto_in_decl_state *state, lto_fixup_data_t *data)
{
unsigned i, si;
struct lto_tree_ref_table *table;
 
/* Although we only want to replace FUNCTION_DECLs and VAR_DECLs,
we still need to walk from all DECLs to find the reachable
FUNCTION_DECLs and VAR_DECLs. */
for (si = 0; si < LTO_N_DECL_STREAMS; si++)
{
table = &state->streams[si];
for (i = 0; i < table->size; i++)
walk_tree (table->trees + i, lto_fixup_tree, data, NULL);
}
}
 
/* A callback of htab_traverse. Just extract a state from SLOT and the
lto_fixup_data_t object from AUX and calls lto_fixup_state. */
 
static int
lto_fixup_state_aux (void **slot, void *aux)
{
struct lto_in_decl_state *state = (struct lto_in_decl_state *) *slot;
lto_fixup_state (state, (lto_fixup_data_t *) aux);
return 1;
}
 
/* Fix the decls from all FILES. Replaces each decl with the corresponding
prevailing one. */
 
static void
lto_fixup_decls (struct lto_file_decl_data **files)
{
unsigned int i;
tree decl;
struct pointer_set_t *seen = pointer_set_create ();
lto_fixup_data_t data;
 
data.seen = seen;
for (i = 0; files[i]; i++)
{
struct lto_file_decl_data *file = files[i];
struct lto_in_decl_state *state = file->global_decl_state;
lto_fixup_state (state, &data);
 
htab_traverse (file->function_decl_states, lto_fixup_state_aux, &data);
}
 
for (i = 0; VEC_iterate (tree, lto_global_var_decls, i, decl); i++)
{
tree saved_decl = decl;
walk_tree (&decl, lto_fixup_tree, &data, NULL);
if (decl != saved_decl)
VEC_replace (tree, lto_global_var_decls, i, decl);
}
 
pointer_set_destroy (seen);
}
 
/* Unlink a temporary LTRANS file unless requested otherwise. */
 
static void
lto_maybe_unlink (const char *file)
{
if (!getenv ("WPA_SAVE_LTRANS"))
{
if (unlink_if_ordinary (file))
error ("deleting LTRANS input file %s: %m", file);
}
else
fprintf (stderr, "[Leaving LTRANS input file %s]\n", file);
}
 
/* Read the options saved from each file in the command line. Called
from lang_hooks.post_options which is called by process_options
right before all the options are used to initialize the compiler.
This assumes that decode_options has already run, so the
num_in_fnames and in_fnames are properly set.
 
Note that this assumes that all the files had been compiled with
the same options, which is not a good assumption. In general,
options ought to be read from all the files in the set and merged.
However, it is still unclear what the merge rules should be. */
 
void
lto_read_all_file_options (void)
{
size_t i;
 
/* Clear any file options currently saved. */
lto_clear_file_options ();
 
/* Set the hooks to read ELF sections. */
lto_set_in_hooks (NULL, get_section_data, free_section_data);
 
for (i = 0; i < num_in_fnames; i++)
{
struct lto_file_decl_data *file_data;
lto_file *file = lto_obj_file_open (in_fnames[i], false);
if (!file)
break;
 
file_data = XCNEW (struct lto_file_decl_data);
file_data->file_name = file->filename;
file_data->section_hash_table = lto_obj_build_section_table (file);
 
lto_read_file_options (file_data);
 
lto_obj_file_close (file);
htab_delete (file_data->section_hash_table);
free (file_data);
}
 
/* Apply globally the options read from all the files. */
lto_reissue_options ();
}
 
 
/* Read all the symbols from the input files FNAMES. NFILES is the
number of files requested in the command line. Instantiate a
global call graph by aggregating all the sub-graphs found in each
file. */
 
static void
read_cgraph_and_symbols (unsigned nfiles, const char **fnames)
{
unsigned int i, last_file_ix;
struct lto_file_decl_data **all_file_decl_data;
FILE *resolution;
struct cgraph_node *node;
 
lto_stats.num_input_files = nfiles;
 
timevar_push (TV_IPA_LTO_DECL_IO);
 
/* Set the hooks so that all of the ipa passes can read in their data. */
all_file_decl_data = XNEWVEC (struct lto_file_decl_data *, nfiles + 1);
lto_set_in_hooks (all_file_decl_data, get_section_data, free_section_data);
 
/* Read the resolution file. */
resolution = NULL;
if (resolution_file_name)
{
int t;
unsigned num_objects;
 
resolution = fopen (resolution_file_name, "r");
if (resolution == NULL)
fatal_error ("could not open symbol resolution file: %s",
xstrerror (errno));
 
t = fscanf (resolution, "%u", &num_objects);
gcc_assert (t == 1);
 
/* True, since the plugin splits the archives. */
gcc_assert (num_objects == nfiles);
}
 
/* Read all of the object files specified on the command line. */
for (i = 0, last_file_ix = 0; i < nfiles; ++i)
{
struct lto_file_decl_data *file_data = NULL;
 
current_lto_file = lto_obj_file_open (fnames[i], false);
if (!current_lto_file)
break;
 
file_data = lto_file_read (current_lto_file, resolution);
if (!file_data)
break;
 
all_file_decl_data[last_file_ix++] = file_data;
 
lto_obj_file_close (current_lto_file);
current_lto_file = NULL;
}
 
if (resolution_file_name)
fclose (resolution);
 
all_file_decl_data[last_file_ix] = NULL;
 
/* Set the hooks so that all of the ipa passes can read in their data. */
lto_set_in_hooks (all_file_decl_data, get_section_data, free_section_data);
 
/* Each pass will set the appropriate timer. */
timevar_pop (TV_IPA_LTO_DECL_IO);
 
/* Read the callgraph. */
input_cgraph ();
 
/* Merge global decls. */
lto_symtab_merge_decls ();
 
/* Fixup all decls and types and free the type hash tables. */
lto_fixup_decls (all_file_decl_data);
free_gimple_type_tables ();
 
/* Read the IPA summary data. */
ipa_read_summaries ();
 
/* Finally merge the cgraph according to the decl merging decisions. */
lto_symtab_merge_cgraph_nodes ();
 
/* Mark cgraph nodes needed in the merged cgraph
This normally happens in whole-program pass, but for
ltrans the pass was already run at WPA phase.
FIXME: This is not valid way to do so; nodes can be needed
for non-obvious reasons. We should stream the flags from WPA
phase. */
if (flag_ltrans)
for (node = cgraph_nodes; node; node = node->next)
{
if (!node->global.inlined_to
&& cgraph_decide_is_function_needed (node, node->decl))
cgraph_mark_needed_node (node);
/* FIXME: ipa_transforms_to_apply holds list of passes that have optimization
summaries computed and needs to apply changes. At the moment WHOPR only
supports inlining, so we can push it here by hand. In future we need to stream
this field into ltrans compilation. */
if (node->analyzed)
VEC_safe_push (ipa_opt_pass, heap,
node->ipa_transforms_to_apply,
(ipa_opt_pass)&pass_ipa_inline);
}
 
timevar_push (TV_IPA_LTO_DECL_IO);
 
/* FIXME lto. This loop needs to be changed to use the pass manager to
call the ipa passes directly. */
if (!errorcount)
for (i = 0; i < last_file_ix; i++)
{
struct lto_file_decl_data *file_data = all_file_decl_data [i];
lto_materialize_constructors_and_inits (file_data);
}
 
/* Indicate that the cgraph is built and ready. */
cgraph_function_flags_ready = true;
 
timevar_pop (TV_IPA_LTO_DECL_IO);
}
 
 
/* Materialize all the bodies for all the nodes in the callgraph. */
 
static void
materialize_cgraph (void)
{
tree decl;
struct cgraph_node *node;
unsigned i;
timevar_id_t lto_timer;
 
/* Now that we have input the cgraph, we need to clear all of the aux
nodes and read the functions if we are not running in WPA mode. */
timevar_push (TV_IPA_LTO_GIMPLE_IO);
 
for (node = cgraph_nodes; node; node = node->next)
{
/* Some cgraph nodes get created on the fly, and they don't need
to be materialized. For instance, nodes for nested functions
where the parent function was not streamed out or builtin
functions. Additionally, builtin functions should not be
materialized and may, in fact, cause confusion because there
may be a regular function in the file whose assembler name
matches that of the function.
See gcc.c-torture/execute/20030125-1.c and
gcc.c-torture/execute/921215-1.c. */
if (node->local.lto_file_data
&& !DECL_IS_BUILTIN (node->decl))
{
lto_materialize_function (node);
lto_stats.num_input_cgraph_nodes++;
}
}
 
timevar_pop (TV_IPA_LTO_GIMPLE_IO);
 
/* Start the appropriate timer depending on the mode that we are
operating in. */
lto_timer = (flag_wpa) ? TV_WHOPR_WPA
: (flag_ltrans) ? TV_WHOPR_LTRANS
: TV_LTO;
timevar_push (lto_timer);
 
current_function_decl = NULL;
set_cfun (NULL);
 
/* Inform the middle end about the global variables we have seen. */
for (i = 0; VEC_iterate (tree, lto_global_var_decls, i, decl); i++)
rest_of_decl_compilation (decl, 1, 0);
 
/* Fix up any calls to DECLs that have become not exception throwing. */
lto_fixup_nothrow_decls ();
 
timevar_pop (lto_timer);
}
 
 
/* Perform whole program analysis (WPA) on the callgraph and write out the
optimization plan. */
 
static void
do_whole_program_analysis (void)
{
char **output_files;
size_t i;
struct cgraph_node *node;
 
lto_1_to_1_map ();
 
/* Note that since we are in WPA mode, materialize_cgraph will not
actually read in all the function bodies. It only materializes
the decls and cgraph nodes so that analysis can be performed. */
materialize_cgraph ();
 
/* Reading in the cgraph uses different timers, start timing WPA now. */
timevar_push (TV_WHOPR_WPA);
 
/* FIXME lto. Hack. We should use the IPA passes. There are a
number of issues with this now. 1. There is no convenient way to
do this. 2. Some passes may depend on properties that requires
the function bodies to compute. */
cgraph_function_flags_ready = true;
bitmap_obstack_initialize (NULL);
ipa_register_cgraph_hooks ();
 
/* Reset inlining information before running IPA inliner. */
for (node = cgraph_nodes; node; node = node->next)
reset_inline_failed (node);
 
/* FIXME lto. We should not call this function directly. */
pass_ipa_inline.pass.execute ();
 
verify_cgraph ();
bitmap_obstack_release (NULL);
 
/* We are about to launch the final LTRANS phase, stop the WPA timer. */
timevar_pop (TV_WHOPR_WPA);
 
output_files = lto_wpa_write_files ();
 
/* Show the LTO report before launching LTRANS. */
if (flag_lto_report)
print_lto_report ();
 
lto_execute_ltrans (output_files);
 
for (i = 0; output_files[i]; ++i)
{
if (output_files[i][0] != '*')
lto_maybe_unlink (output_files[i]);
 
free (output_files[i]);
}
 
XDELETEVEC (output_files);
}
 
 
/* Main entry point for the GIMPLE front end. This front end has
three main personalities:
 
- LTO (-flto). All the object files on the command line are
loaded in memory and processed as a single translation unit.
This is the traditional link-time optimization behavior.
 
- WPA (-fwpa). Only the callgraph and summary information for
files in the command file are loaded. A single callgraph
(without function bodies) is instantiated for the whole set of
files. IPA passes are only allowed to analyze the call graph
and make transformation decisions. The callgraph is
partitioned, each partition is written to a new object file
together with the transformation decisions.
 
- LTRANS (-fltrans). Similar to -flto but it prevents the IPA
summary files from running again. Since WPA computed summary
information and decided what transformations to apply, LTRANS
simply applies them. */
 
void
lto_main (int debug_p ATTRIBUTE_UNUSED)
{
lto_init_reader ();
 
/* Read all the symbols and call graph from all the files in the
command line. */
read_cgraph_and_symbols (num_in_fnames, in_fnames);
 
if (!errorcount)
{
/* If WPA is enabled analyze the whole call graph and create an
optimization plan. Otherwise, read in all the function
bodies and continue with optimization. */
if (flag_wpa)
do_whole_program_analysis ();
else
{
materialize_cgraph ();
 
/* Let the middle end know that we have read and merged all of
the input files. */
cgraph_optimize ();
 
/* FIXME lto, if the processes spawned by WPA fail, we miss
the chance to print WPA's report, so WPA will call
print_lto_report before launching LTRANS. If LTRANS was
launched directly by the driver we would not need to do
this. */
if (flag_lto_report)
print_lto_report ();
}
}
}
 
#include "gt-lto-lto.h"
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-endian.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-endian.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-endian.h (revision 288) @@ -0,0 +1,205 @@ +/* Very simple endian-ness layer for LTO object file handling + Copyright 2010 Free Software Foundation, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +/* This header file provides a simple way to handle object files in + another endian-ness than the host machine. This is necesarry to + enable cross-compilation with LTO enabled. Targets that use the + ELF binary object format do not need this (libelf already handles + endian-ness) but for COFF and Mach-O the functions in this header + are used in the minimal binary object reader/writer. + + For all functions in this header, the user is responsible for + making sure that the memory accesses are valid. */ + +#ifndef GCC_LTO_ENDIAN_H +#define GCC_LTO_ENDIAN_H + +#include +#include + +static inline uint16_t +get_uint16_le (const unsigned char *ptr) +{ + return ptr[0] | (ptr[1] << 8); +} + +static inline uint32_t +get_uint32_le (const unsigned char *ptr) +{ + return ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24); +} + +static inline uint64_t +get_uint64_le (const unsigned char *ptr_) +{ +#define ptr (uint64_t) ptr_ + return ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24) + | (ptr[4] << 32) | (ptr[5] << 40) | (ptr[6] << 48) | (ptr[7] << 56); +#undef ptr +} + +static inline uint16_t +get_uint16_be (const unsigned char *ptr) +{ + return ptr[1] | (ptr[2] << 8); +} + +static inline uint32_t +get_uint32_be (const unsigned char *ptr) +{ + return ptr[3] | (ptr[2] << 8) | (ptr[1] << 16) | (ptr[0] << 24); +} + +static inline uint64_t +get_uint64_be (const unsigned char *ptr_) +{ +#define ptr (uint64_t) ptr_ + return ptr[7] | (ptr[6] << 8) | (ptr[5] << 16) | (ptr[4] << 24) + | (ptr[3] << 32) | (ptr[2] << 40) | (ptr[1] << 48) | (ptr[0] << 56); +#undef ptr +} + +static inline void +put_uint16_le (unsigned char *ptr, uint16_t data) +{ + ptr[0] = data & 0xff; + ptr[1] = (data >> 8) & 0xff; +} + +static inline void +put_uint32_le (unsigned char *ptr, uint32_t data) +{ + ptr[0] = data & 0xff; + ptr[1] = (data >> 8) & 0xff; + ptr[2] = (data >> 16) & 0xff; + ptr[3] = (data >> 24) & 0xff; +} + +static inline void +put_uint64_le (unsigned char *ptr, uint64_t data) +{ + ptr[0] = data & 0xff; + ptr[1] = (data >> 8) & 0xff; + ptr[2] = (data >> 16) & 0xff; + ptr[3] = (data >> 24) & 0xff; + ptr[4] = (data >> 32) & 0xff; + ptr[5] = (data >> 40) & 0xff; + ptr[6] = (data >> 48) & 0xff; + ptr[7] = (data >> 56) & 0xff; +} + +static inline void +put_uint16_be (unsigned char *ptr, uint16_t data) +{ + ptr[1] = data & 0xff; + ptr[0] = (data >> 8) & 0xff; +} + +static inline void +put_uint32_be (unsigned char *ptr, uint32_t data) +{ + ptr[3] = data & 0xff; + ptr[2] = (data >> 8) & 0xff; + ptr[1] = (data >> 16) & 0xff; + ptr[0] = (data >> 24) & 0xff; +} + +static inline void +put_uint64_be (unsigned char *ptr, uint64_t data) +{ + ptr[7] = data & 0xff; + ptr[6] = (data >> 8) & 0xff; + ptr[5] = (data >> 16) & 0xff; + ptr[4] = (data >> 24) & 0xff; + ptr[3] = (data >> 32) & 0xff; + ptr[2] = (data >> 40) & 0xff; + ptr[1] = (data >> 48) & 0xff; + ptr[0] = (data >> 56) & 0xff; +} + +static inline void +get_string (unsigned char *ptr, char *dest, size_t len) +{ + memcpy (dest, ptr, len); +} + +static inline void +put_string (unsigned char *ptr, char *src, size_t len) +{ + memcpy (ptr, src, len); +} + +/* Use the target macro BYTES_BIG_ENDIAN to choose. */ + +static inline uint16_t +get_uint16 (const unsigned char *ptr) +{ + if (BYTES_BIG_ENDIAN) + return get_uint16_be (ptr); + else + return get_uint16_le (ptr); +} + +static inline uint32_t +get_uint32 (const unsigned char *ptr) +{ + if (BYTES_BIG_ENDIAN) + return get_uint32_be (ptr); + else + return get_uint32_le (ptr); +} + +static inline uint64_t +get_uint64 (const unsigned char *ptr) +{ + if (BYTES_BIG_ENDIAN) + return get_uint64_be (ptr); + else + return get_uint64_le (ptr); +} + +static inline void +put_uint16 (unsigned char *ptr, uint16_t data) +{ + if (BYTES_BIG_ENDIAN) + put_uint16_be (ptr, data); + else + put_uint16_le (ptr, data); +} + +static inline void +put_uint32 (unsigned char *ptr, uint32_t data) +{ + if (BYTES_BIG_ENDIAN) + put_uint32_be (ptr, data); + else + put_uint32_le (ptr, data); +} + +static inline void +put_uint64 (unsigned char *ptr, uint64_t data) +{ + if (BYTES_BIG_ENDIAN) + put_uint64_be (ptr, data); + else + put_uint64_le (ptr, data); +} + +#endif /* GCC_LTO_ENDIAN_H */ +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-endian.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-tree.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-tree.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-tree.h (revision 288) @@ -0,0 +1,61 @@ +/* Language-dependent trees for LTO. + Copyright 2009 Free Software Foundation, Inc. + Contributed by CodeSourcery, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#ifndef GCC_LTO_TREE_H +#define GCC_LTO_TREE_H + +#include "plugin-api.h" + +struct GTY(()) lang_identifier +{ + struct tree_identifier base; +}; + +struct GTY(()) lang_decl +{ + int dummy; /* Added because ggc does not like empty structs. */ +}; + +struct GTY(()) lang_type +{ + int dummy; /* Added because ggc does not like empty structs. */ +}; + +struct GTY(()) language_function +{ + int dummy; /* Added because ggc does not like empty structs. */ +}; + +enum lto_tree_node_structure_enum { + TS_LTO_GENERIC +}; + +union GTY((desc ("lto_tree_node_structure (&%h)"), + chain_next ("(union lang_tree_node *)TREE_CHAIN (&%h.generic)"))) + lang_tree_node +{ + union tree_node GTY ((tag ("TS_LTO_GENERIC"), + desc ("tree_node_structure (&%h)"))) generic; +}; + +/* Vector to keep track of external variables we've seen so far. */ +extern GTY(()) VEC(tree,gc) *lto_global_var_decls; + +#endif /* GCC_LTO_TREE_H */
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-tree.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang-specs.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang-specs.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang-specs.h (revision 288) @@ -0,0 +1,24 @@ +/* LTO driver specs. + Copyright 2009 Free Software Foundation, Inc. + Contributed by CodeSourcery, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +/* LTO contributions to the "compilers" array in gcc.c. */ + + {"@lto", "lto1 %(cc1_options) %i %{!fsyntax-only:%(invoke_as)}", + /*cpp_spec=*/NULL, /*combinable=*/1, /*needs_preprocessing=*/0},
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang-specs.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang.opt =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang.opt (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lang.opt (revision 288) @@ -0,0 +1,43 @@ +; Options for the LTO front end. +; Copyright (C) 2008 Free Software Foundation, Inc. +; +; This file is part of GCC. +; +; GCC is free software; you can redistribute it and/or modify it under +; the terms of the GNU General Public License as published by the Free +; Software Foundation; either version 3, or (at your option) any later +; version. +; +; GCC is distributed in the hope that it will be useful, but WITHOUT ANY +; WARRANTY; without even the implied warranty of MERCHANTABILITY or +; FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +; for more details. +; +; You should have received a copy of the GNU General Public License +; along with GCC; see the file COPYING3. If not see +; . + +; See the GCC internals manual for a description of this file's format. + +; Please try to keep this file in ASCII collating order. + +Language +LTO + +fltrans +LTO Report Var(flag_ltrans) Optimization +Run the link-time optimizer in local transformation (LTRANS) mode. + +fltrans-output-list= +LTO Joined Var(ltrans_output_list) +Specify a file to which a list of files output by LTRANS is written. + +fwpa +LTO Report Var(flag_wpa) Optimization +Run the link-time optimizer in whole program analysis (WPA) mode. + +fresolution +LTO Separate +The resolution file + +; This comment is to ensure we retain the blank line above. Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-elf.c =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-elf.c (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-elf.c (revision 288) @@ -0,0 +1,809 @@ +/* LTO routines for ELF object files. + Copyright 2009, 2010 Free Software Foundation, Inc. + Contributed by CodeSourcery, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#include "config.h" +#include "system.h" +#include "coretypes.h" +#include "toplev.h" +#include +#include "lto.h" +#include "tm.h" +#include "libiberty.h" +#include "ggc.h" +#include "lto-streamer.h" + +/* Cater to hosts with half-backed file like HP-UX. */ +#ifndef EM_SPARC +# define EM_SPARC 2 +#endif + +#ifndef EM_SPARC32PLUS +# define EM_SPARC32PLUS 18 +#endif + + +/* Handle opening elf files on hosts, such as Windows, that may use + text file handling that will break binary access. */ +#ifndef O_BINARY +# define O_BINARY 0 +#endif + + +/* Initialize FILE, an LTO file object for FILENAME. */ +static void +lto_file_init (lto_file *file, const char *filename, off_t offset) +{ + file->filename = filename; + file->offset = offset; +} + +/* An ELF file. */ +struct lto_elf_file +{ + /* The base information. */ + lto_file base; + + /* The system file descriptor for the file. */ + int fd; + + /* The libelf descriptor for the file. */ + Elf *elf; + + /* Section number of string table used for section names. */ + size_t sec_strtab; + + /* Writable file members. */ + + /* The currently active section. */ + Elf_Scn *scn; + + /* The output stream for section header names. */ + struct lto_output_stream *shstrtab_stream; + + /* Linked list of data which must be freed *after* the file has been + closed. This is an annoying limitation of libelf. */ + struct lto_char_ptr_base *data; +}; +typedef struct lto_elf_file lto_elf_file; + +/* Stores executable header attributes which must be shared by all ELF files. + This is used for validating input files and populating output files. */ +static struct { + bool initialized; + /* 32 or 64 bits? */ + size_t bits; + unsigned char elf_ident[EI_NIDENT]; + Elf64_Half elf_machine; +} cached_file_attrs; + + +/* Return the section header for SECTION. The return value is never + NULL. Call lto_elf_free_shdr to release the memory allocated. */ + +static Elf64_Shdr * +lto_elf_get_shdr (Elf_Scn *section) +{ + Elf64_Shdr *shdr; + + switch (cached_file_attrs.bits) + { + case 32: + { + Elf32_Shdr *shdr32; + + /* Read the 32-bit section header. */ + shdr32 = elf32_getshdr (section); + if (!shdr32) + fatal_error ("could not read section header: %s", elf_errmsg (0)); + + /* Transform it into a 64-bit section header. */ + shdr = XNEW (Elf64_Shdr); + shdr->sh_name = shdr32->sh_name; + shdr->sh_type = shdr32->sh_type; + shdr->sh_flags = shdr32->sh_flags; + shdr->sh_addr = shdr32->sh_addr; + shdr->sh_offset = shdr32->sh_offset; + shdr->sh_size = shdr32->sh_size; + shdr->sh_link = shdr32->sh_link; + shdr->sh_info = shdr32->sh_info; + shdr->sh_addralign = shdr32->sh_addralign; + shdr->sh_entsize = shdr32->sh_entsize; + break; + } + break; + + case 64: + shdr = elf64_getshdr (section); + if (!shdr) + fatal_error ("could not read section header: %s", elf_errmsg (0)); + break; + + default: + gcc_unreachable (); + } + + return shdr; +} + +/* Free SHDR, previously allocated by lto_elf_get_shdr. */ +static void +lto_elf_free_shdr (Elf64_Shdr *shdr) +{ + if (cached_file_attrs.bits != 64) + free (shdr); +} + + +/* Returns a hash code for P. */ + +static hashval_t +hash_name (const void *p) +{ + const struct lto_section_slot *ds = (const struct lto_section_slot *) p; + return (hashval_t) htab_hash_string (ds->name); +} + + +/* Returns nonzero if P1 and P2 are equal. */ + +static int +eq_name (const void *p1, const void *p2) +{ + const struct lto_section_slot *s1 = + (const struct lto_section_slot *) p1; + const struct lto_section_slot *s2 = + (const struct lto_section_slot *) p2; + + return strcmp (s1->name, s2->name) == 0; +} + + +/* Build a hash table whose key is the section names and whose data is + the start and size of each section in the .o file. */ + +htab_t +lto_obj_build_section_table (lto_file *lto_file) +{ + lto_elf_file *elf_file = (lto_elf_file *)lto_file; + htab_t section_hash_table; + Elf_Scn *section; + size_t base_offset; + + section_hash_table = htab_create (37, hash_name, eq_name, free); + + base_offset = elf_getbase (elf_file->elf); + /* We are reasonably sure that elf_getbase does not fail at this + point. So assume that we run into the incompatibility with + the FreeBSD libelf implementation that has a non-working + elf_getbase for non-archive members in which case the offset + should be zero. */ + if (base_offset == (size_t)-1) + base_offset = 0; + for (section = elf_getscn (elf_file->elf, 0); + section; + section = elf_nextscn (elf_file->elf, section)) + { + Elf64_Shdr *shdr; + const char *name; + size_t offset; + char *new_name; + void **slot; + struct lto_section_slot s_slot; + + /* Get the name of this section. */ + shdr = lto_elf_get_shdr (section); + offset = shdr->sh_name; + name = elf_strptr (elf_file->elf, + elf_file->sec_strtab, + offset); + + /* Only put lto stuff into the symtab. */ + if (strncmp (name, LTO_SECTION_NAME_PREFIX, + strlen (LTO_SECTION_NAME_PREFIX)) != 0) + { + lto_elf_free_shdr (shdr); + continue; + } + + new_name = XNEWVEC (char, strlen (name) + 1); + strcpy (new_name, name); + s_slot.name = new_name; + slot = htab_find_slot (section_hash_table, &s_slot, INSERT); + if (*slot == NULL) + { + struct lto_section_slot *new_slot = XNEW (struct lto_section_slot); + + new_slot->name = new_name; + /* The offset into the file for this section. */ + new_slot->start = base_offset + shdr->sh_offset; + new_slot->len = shdr->sh_size; + *slot = new_slot; + } + else + { + error ("two or more sections for %s:", new_name); + return NULL; + } + + lto_elf_free_shdr (shdr); + } + + return section_hash_table; +} + + +/* Initialize the section header of section SCN. SH_NAME is the section name + as an index into the section header string table. SH_TYPE is the section + type, an SHT_* macro from libelf headers. */ + +#define DEFINE_INIT_SHDR(BITS) \ +static void \ +init_shdr##BITS (Elf_Scn *scn, size_t sh_name, size_t sh_type) \ +{ \ + Elf##BITS##_Shdr *shdr; \ + \ + shdr = elf##BITS##_getshdr (scn); \ + if (!shdr) \ + { \ + if (BITS == 32) \ + fatal_error ("elf32_getshdr() failed: %s", elf_errmsg (-1)); \ + else \ + fatal_error ("elf64_getshdr() failed: %s", elf_errmsg (-1)); \ + } \ + \ + shdr->sh_name = sh_name; \ + shdr->sh_type = sh_type; \ + shdr->sh_addralign = POINTER_SIZE / BITS_PER_UNIT; \ + shdr->sh_flags = 0; \ + shdr->sh_entsize = 0; \ +} + +DEFINE_INIT_SHDR (32) +DEFINE_INIT_SHDR (64) + +static bool first_data_block; + +/* Begin a new ELF section named NAME with type TYPE in the current output + file. TYPE is an SHT_* macro from the libelf headers. */ + +static void +lto_elf_begin_section_with_type (const char *name, size_t type) +{ + lto_elf_file *file; + Elf_Scn *scn; + size_t sh_name; + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_elf_file *) lto_get_current_out_file (), + gcc_assert (file); + gcc_assert (file->elf); + gcc_assert (!file->scn); + + /* Create a new section. */ + scn = elf_newscn (file->elf); + if (!scn) + fatal_error ("could not create a new ELF section: %s", elf_errmsg (-1)); + file->scn = scn; + + /* Add a string table entry and record the offset. */ + gcc_assert (file->shstrtab_stream); + sh_name = file->shstrtab_stream->total_size; + lto_output_data_stream (file->shstrtab_stream, name, strlen (name) + 1); + + /* Initialize the section header. */ + switch (cached_file_attrs.bits) + { + case 32: + init_shdr32 (scn, sh_name, type); + break; + + case 64: + init_shdr64 (scn, sh_name, type); + break; + + default: + gcc_unreachable (); + } + + first_data_block = true; +} + + +/* Begin a new ELF section named NAME in the current output file. */ + +void +lto_obj_begin_section (const char *name) +{ + lto_elf_begin_section_with_type (name, SHT_PROGBITS); +} + + +/* Append DATA of length LEN to the current output section. BASE is a pointer + to the output page containing DATA. It is freed once the output file has + been written. */ + +void +lto_obj_append_data (const void *data, size_t len, void *block) +{ + lto_elf_file *file; + Elf_Data *elf_data; + struct lto_char_ptr_base *base = (struct lto_char_ptr_base *) block; + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_elf_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + elf_data = elf_newdata (file->scn); + if (!elf_data) + fatal_error ("could not append data to ELF section: %s", elf_errmsg (-1)); + + if (first_data_block) + { + elf_data->d_align = POINTER_SIZE / BITS_PER_UNIT; + first_data_block = false; + } + else + elf_data->d_align = 1; + elf_data->d_buf = CONST_CAST (void *, data); + elf_data->d_off = 0LL; + elf_data->d_size = len; + elf_data->d_type = ELF_T_BYTE; + elf_data->d_version = EV_CURRENT; + + base->ptr = (char *)file->data; + file->data = base; +} + + +/* End the current output section. This just does some assertion checking + and sets the current output file's scn member to NULL. */ + +void +lto_obj_end_section (void) +{ + lto_elf_file *file; + + /* Grab the current output file and validate some basic assertions. */ + file = (lto_elf_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + file->scn = NULL; +} + + +/* Return true if ELF_MACHINE is compatible with the cached value of the + architecture and possibly update the latter. Return false otherwise. + + Note: if you want to add more EM_* cases, you'll need to provide the + corresponding definitions at the beginning of the file. */ + +static bool +is_compatible_architecture (Elf64_Half elf_machine) +{ + if (cached_file_attrs.elf_machine == elf_machine) + return true; + + switch (cached_file_attrs.elf_machine) + { + case EM_SPARC: + if (elf_machine == EM_SPARC32PLUS) + { + cached_file_attrs.elf_machine = elf_machine; + return true; + } + break; + + case EM_SPARC32PLUS: + if (elf_machine == EM_SPARC) + return true; + break; + + default: + break; + } + + return false; +} + + +/* Validate's ELF_FILE's executable header and, if cached_file_attrs is + uninitialized, caches the architecture. */ + +#define DEFINE_VALIDATE_EHDR(BITS) \ +static bool \ +validate_ehdr##BITS (lto_elf_file *elf_file) \ +{ \ + Elf##BITS##_Ehdr *elf_header; \ + \ + elf_header = elf##BITS##_getehdr (elf_file->elf); \ + if (!elf_header) \ + { \ + error ("could not read ELF header: %s", elf_errmsg (0)); \ + return false; \ + } \ + \ + if (elf_header->e_type != ET_REL) \ + { \ + error ("not a relocatable ELF object file"); \ + return false; \ + } \ + \ + if (!cached_file_attrs.initialized) \ + cached_file_attrs.elf_machine = elf_header->e_machine; \ + else if (!is_compatible_architecture (elf_header->e_machine)) \ + { \ + error ("inconsistent file architecture detected"); \ + return false; \ + } \ + \ + return true; \ +} + +DEFINE_VALIDATE_EHDR (32) +DEFINE_VALIDATE_EHDR (64) + + +#ifndef HAVE_ELF_GETSHDRSTRNDX +/* elf_getshdrstrndx replacement for systems that lack it, but provide + either the gABI conformant or Solaris 2 variant of elf_getshstrndx + instead. */ + +static int +elf_getshdrstrndx (Elf *elf, size_t *dst) +{ +#ifdef HAVE_ELF_GETSHSTRNDX_GABI + return elf_getshstrndx (elf, dst); +#else + return elf_getshstrndx (elf, dst) ? 0 : -1; +#endif +} +#endif + +/* Validate's ELF_FILE's executable header and, if cached_file_attrs is + uninitialized, caches the results. Also records the section header string + table's section index. Returns true on success or false on failure. */ + +static bool +validate_file (lto_elf_file *elf_file) +{ + const char *elf_ident; + + /* Some aspects of the libelf API are dependent on whether the + object file is a 32-bit or 64-bit file. Determine which kind of + file this is now. */ + elf_ident = elf_getident (elf_file->elf, NULL); + if (!elf_ident) + { + error ("could not read ELF identification information: %s", + elf_errmsg (0)); + return false; + + } + + if (!cached_file_attrs.initialized) + { + switch (elf_ident[EI_CLASS]) + { + case ELFCLASS32: + cached_file_attrs.bits = 32; + break; + + case ELFCLASS64: + cached_file_attrs.bits = 64; + break; + + default: + error ("unsupported ELF file class"); + return false; + } + + memcpy (cached_file_attrs.elf_ident, elf_ident, + sizeof cached_file_attrs.elf_ident); + } + + if (memcmp (elf_ident, cached_file_attrs.elf_ident, + sizeof cached_file_attrs.elf_ident)) + return false; + + /* Check that the input file is a relocatable object file with the correct + architecture. */ + switch (cached_file_attrs.bits) + { + case 32: + if (!validate_ehdr32 (elf_file)) + return false; + break; + + case 64: + if (!validate_ehdr64 (elf_file)) + return false; + break; + + default: + gcc_unreachable (); + } + + /* Read the string table used for section header names. */ + if (elf_getshdrstrndx (elf_file->elf, &elf_file->sec_strtab) == -1) + { + error ("could not locate ELF string table: %s", elf_errmsg (0)); + return false; + } + + cached_file_attrs.initialized = true; + return true; +} + + +/* Helper functions used by init_ehdr. Initialize ELF_FILE's executable + header using cached data from previously read files. */ + +#define DEFINE_INIT_EHDR(BITS) \ +static void \ +init_ehdr##BITS (lto_elf_file *elf_file) \ +{ \ + Elf##BITS##_Ehdr *ehdr; \ + \ + gcc_assert (cached_file_attrs.bits); \ + \ + ehdr = elf##BITS##_newehdr (elf_file->elf); \ + if (!ehdr) \ + { \ + if (BITS == 32) \ + fatal_error ("elf32_newehdr() failed: %s", elf_errmsg (-1)); \ + else \ + fatal_error ("elf64_newehdr() failed: %s", elf_errmsg (-1)); \ + } \ + \ + memcpy (ehdr->e_ident, cached_file_attrs.elf_ident, \ + sizeof cached_file_attrs.elf_ident); \ + ehdr->e_type = ET_REL; \ + ehdr->e_version = EV_CURRENT; \ + ehdr->e_machine = cached_file_attrs.elf_machine; \ +} + +DEFINE_INIT_EHDR (32) +DEFINE_INIT_EHDR (64) + + +/* Initialize ELF_FILE's executable header using cached data from previously + read files. */ + +static void +init_ehdr (lto_elf_file *elf_file) +{ + switch (cached_file_attrs.bits) + { + case 32: + init_ehdr32 (elf_file); + break; + + case 64: + init_ehdr64 (elf_file); + break; + + default: + gcc_unreachable (); + } +} + +/* Open ELF file FILENAME. If WRITABLE is true, the file is opened for write + and, if necessary, created. Otherwise, the file is opened for reading. + Returns the opened file. */ + +lto_file * +lto_obj_file_open (const char *filename, bool writable) +{ + lto_elf_file *elf_file; + lto_file *result = NULL; + off_t offset; + long loffset; + off_t header_offset; + const char *offset_p; + char *fname; + int consumed; + + offset_p = strrchr (filename, '@'); + if (offset_p + && offset_p != filename + && sscanf (offset_p, "@%li%n", &loffset, &consumed) >= 1 + && strlen (offset_p) == (unsigned int)consumed) + { + fname = (char *) xmalloc (offset_p - filename + 1); + memcpy (fname, filename, offset_p - filename); + fname[offset_p - filename] = '\0'; + offset = (off_t)loffset; + /* elf_rand expects the offset to point to the ar header, not the + object itself. Subtract the size of the ar header (60 bytes). + We don't uses sizeof (struct ar_hd) to avoid including ar.h */ + header_offset = offset - 60; + } + else + { + fname = xstrdup (filename); + offset = 0; + header_offset = 0; + } + + /* Set up. */ + elf_file = XCNEW (lto_elf_file); + result = (lto_file *) elf_file; + lto_file_init (result, fname, offset); + elf_file->fd = -1; + + /* Open the file. */ + elf_file->fd = open (fname, writable ? O_WRONLY|O_CREAT|O_BINARY + : O_RDONLY|O_BINARY, 0666); + if (elf_file->fd == -1) + { + error ("could not open file %s", fname); + goto fail; + } + + /* Initialize the ELF library. */ + if (elf_version (EV_CURRENT) == EV_NONE) + { + error ("ELF library is older than that used when building GCC"); + goto fail; + } + + /* Open the ELF file descriptor. */ + elf_file->elf = elf_begin (elf_file->fd, writable ? ELF_C_WRITE : ELF_C_READ, + NULL); + if (!elf_file->elf) + { + error ("could not open ELF file: %s", elf_errmsg (0)); + goto fail; + } + + if (offset != 0) + { + Elf *e; + off_t t = elf_rand (elf_file->elf, header_offset); + if (t != header_offset) + { + error ("could not seek in archive"); + goto fail; + } + + e = elf_begin (elf_file->fd, ELF_C_READ, elf_file->elf); + if (e == NULL) + { + error("could not find archive member"); + goto fail; + } + elf_end (elf_file->elf); + elf_file->elf = e; + } + + if (writable) + { + init_ehdr (elf_file); + elf_file->shstrtab_stream = XCNEW (struct lto_output_stream); + /* Output an empty string to the section header table. This becomes the + name of the initial NULL section. */ + lto_output_1_stream (elf_file->shstrtab_stream, '\0'); + } + else + if (!validate_file (elf_file)) + goto fail; + + return result; + + fail: + if (result) + lto_obj_file_close (result); + return NULL; +} + + +/* Close ELF file FILE and clean up any associated data structures. If FILE + was opened for writing, the file's ELF data is written at this time, and + any cached data buffers are freed. */ + +void +lto_obj_file_close (lto_file *file) +{ + lto_elf_file *elf_file = (lto_elf_file *) file; + struct lto_char_ptr_base *cur, *tmp; + + /* Write the ELF section header string table. */ + if (elf_file->shstrtab_stream) + { + size_t strtab; + GElf_Ehdr *ehdr_p, ehdr_buf; + lto_file *old_file = lto_set_current_out_file (file); + + lto_elf_begin_section_with_type (".shstrtab", SHT_STRTAB); + ehdr_p = gelf_getehdr (elf_file->elf, &ehdr_buf); + if (ehdr_p == NULL) + fatal_error ("gelf_getehdr() failed: %s", elf_errmsg (-1)); + strtab = elf_ndxscn (elf_file->scn); + if (strtab < SHN_LORESERVE) + ehdr_p->e_shstrndx = strtab; + else + { + GElf_Shdr *shdr_p, shdr_buf; + Elf_Scn *scn_p = elf_getscn (elf_file->elf, 0); + if (scn_p == NULL) + fatal_error ("elf_getscn() failed: %s", elf_errmsg (-1)); + shdr_p = gelf_getshdr (scn_p, &shdr_buf); + if (shdr_p == NULL) + fatal_error ("gelf_getshdr() failed: %s", elf_errmsg (-1)); + shdr_p->sh_link = strtab; + if (gelf_update_shdr (scn_p, shdr_p) == 0) + fatal_error ("gelf_update_shdr() failed: %s", elf_errmsg (-1)); + ehdr_p->e_shstrndx = SHN_XINDEX; + } + if (gelf_update_ehdr (elf_file->elf, ehdr_p) == 0) + fatal_error ("gelf_update_ehdr() failed: %s", elf_errmsg (-1)); + lto_write_stream (elf_file->shstrtab_stream); + lto_obj_end_section (); + + lto_set_current_out_file (old_file); + free (elf_file->shstrtab_stream); + + if (elf_update (elf_file->elf, ELF_C_WRITE) < 0) + fatal_error ("elf_update() failed: %s", elf_errmsg (-1)); + } + + if (elf_file->elf) + elf_end (elf_file->elf); + if (elf_file->fd != -1) + close (elf_file->fd); + + /* Free any ELF data buffers. */ + cur = elf_file->data; + while (cur) + { + tmp = cur; + cur = (struct lto_char_ptr_base *) cur->ptr; + free (tmp); + } + + free (file); +} + + +/* The current output file. */ +static lto_file *current_out_file; + + +/* Sets the current output file to FILE. Returns the old output file or + NULL. */ + +lto_file * +lto_set_current_out_file (lto_file *file) +{ + lto_file *old_file = current_out_file; + current_out_file = file; + return old_file; +} + + +/* Returns the current output file. */ + +lto_file * +lto_get_current_out_file (void) +{ + return current_out_file; +}
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-elf.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.h (revision 288) @@ -0,0 +1,63 @@ +/* LTO declarations. + Copyright 2009, 2010 Free Software Foundation, Inc. + Contributed by CodeSourcery, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#ifndef LTO_H +#define LTO_H + +#include "hashtab.h" + +/* A file. */ +typedef struct lto_file_struct +{ + /* The name of the file. */ + const char *filename; + /* The offset for the object inside an ar archive file (or zero). */ + off_t offset; +} lto_file; + +/* In lto-lang.c */ +extern const char *resolution_file_name; + +/* In lto.c */ +extern void lto_main (int); +extern void lto_read_all_file_options (void); + +/* In lto-elf.c or lto-coff.c */ +extern lto_file *lto_obj_file_open (const char *filename, bool writable); +extern void lto_obj_file_close (lto_file *file); +extern htab_t lto_obj_build_section_table (lto_file *file); +extern void lto_obj_begin_section (const char *name); +extern void lto_obj_append_data (const void *data, size_t len, void *block); +extern void lto_obj_end_section (void); +extern lto_file *lto_set_current_out_file (lto_file *file); +extern lto_file *lto_get_current_out_file (void); + +/* Hash table entry to hold the start offset and length of an LTO + section in a .o file. */ +struct lto_section_slot +{ + const char *name; + intptr_t start; + size_t len; +}; + +int64_t lto_parse_hex (const char *p); + +#endif /* LTO_H */
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/ChangeLog =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/ChangeLog (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/ChangeLog (revision 288) @@ -0,0 +1,2817 @@ +2010-07-31 Release Manager + + * GCC 4.5.1 released. + +2010-07-10 Iain Sandoe + + Added the following, missed from the previous commit. + * lto-endian.h: New file. + * lto-macho.h: New file. + * lto-macho.c: New file. + +2010-06-24 Steven Bosscher + + Backport from mainline: + 2010-05-18 Steven Bosscher + + * lto.h (struct lto_file_struct): Document offset member. + * lto-endian.h: New file. + * lto-macho.h: New file. + * lto-macho.c: New file. + * Make-lang.in: Add rule for lto-macho.o. + +2010-06-14 Dave Korn + + Backport from mainline: + 2010-04-27 Dave Korn + + PR lto/42776 + * Make-lang.in (LTO_OBJS): Use LTO_BINARY_READER instead of + hardcoding 'lto-elf.o'. + ($(LTO_EXE)): Use LTO_USE_LIBELF instead of hardcoding '-lelf'. + * lto-coff.h: New file. + * lto-coff.c: Likewise. + + 2010-04-26 Dave Korn + + * lto.h (lto_elf_file_open): Rename prototype from this ... + (lto_obj_file_open): ... to this. + (lto_elf_file_close): Likewise ... + (lto_obj_file_close): ... and likewise. + (lto_elf_build_section_table): Likewise ... + (lto_obj_build_section_table): ... and likewise. + (lto_elf_begin_section): Likewise ... + (lto_obj_begin_section): ... and likewise. + (lto_elf_append_data): Likewise ... + (lto_obj_append_data): ... and likewise. + (lto_elf_end_section): Likewise ... + (lto_obj_end_section): ... and likewise. + * lto.c (lto_file_read): Update references to the above. + (lto_wpa_write_files): Likewise. + (lto_read_all_file_options): Likewise. + (read_cgraph_and_symbols): Likewise. + * lto-lang.c (LANG_HOOKS_BEGIN_SECTION): Likewise. + (LANG_HOOKS_APPEND_DATA): Likewise. + (LANG_HOOKS_END_SECTION): Likewise. + * lto-elf.c (lto_elf_file_open): Rename from this ... + (lto_obj_file_open): ... to this, updating any references. + (lto_elf_file_close): Likewise ... + (lto_obj_file_close): ... and likewise. + (lto_elf_build_section_table): Likewise ... + (lto_obj_build_section_table): ... and likewise. + (lto_elf_begin_section): Likewise ... + (lto_obj_begin_section): ... and likewise. + (lto_elf_append_data): Likewise ... + (lto_obj_append_data): ... and likewise. + (lto_elf_end_section): Likewise ... + (lto_obj_end_section): ... and likewise. + +2010-06-04 Rainer Orth + + Backport from mainline: + 2010-04-16 Rainer Orth + + * lto-elf.c [!HAVE_ELF_GETSHDRSTRNDX] (elf_getshdrstrndx): New + function. + +2010-05-24 Richard Guenther + + PR lto/44256 + * lto.c (lto_fixup_type): Deal with non-type TYPE_CONTEXT. + +2010-05-24 Richard Guenther + + * lto-elf.c (lto_obj_build_section_table): Work around + FreeBSD libelf issue. + +2010-04-14 Release Manager + + * GCC 4.5.0 released. + +2010-03-09 Eric Botcazou + + PR bootstrap/43276 + * lto-elf.c: Define EM_* constants if not already defined. + +2010-03-03 Eric Botcazou + + * lto-elf.c (is_compatible_architecture): New static function. + (DEFINE_VALIDATE_EHDR): Use it to validate the architecture. + +2010-02-11 Richard Guenther + + PR driver/43021 + * lto-elf.c (lto_elf_file_open): Handle file@offset case more + appropriately. + +2010-01-11 Andy Hutchinson + + * lto.c (O_BINARY): Define. + (lto_read_section_data): Open file in binary mode. + * lto-elf.c (O_BINARY): Define. + (lto_elf_file_open): Open file in binary mode. + +2010-01-08 Richard Guenther + + PR lto/42528 + * lto-lang.c (lto_handle_option): Handle -f[un]signed-char. + (lto_init): Do not init char_type_node in a standard way + but according to flag_signed_char. + +2010-01-03 H.J. Lu + + PR lto/41564 + * lto.c (DUMPBASE_SUFFIX): New. + (lto_execute_ltrans): Append a sequence number to -dumpbase + for LTRANS. + +2010-01-02 H.J. Lu + + PR lto/42580 + * lto-elf.c (lto_elf_file_open): Stop if the command line + option file is missing. + +2009-12-15 Richard Guenther + + * lto.c (lto_fixup_field_decl): Fixup DECL_FIELD_OFFSET. + (lto_post_options): Do not disable debuginfo. + +2009-12-14 Dmitry Gorbachev + + * Make-lang.in ($(LTO_EXE)): Use $(LINKER). + +2009-12-11 Richard Guenther + + PR lto/42037 + * lto.c (lto_resolution_read): Properly grow the vector. + +2009-12-11 Richard Guenther + + PR lto/41915 + * lto-lang.c (lto_init_options): Initialize flag_complex_method + to the C99 default. Do not set flag_unit_at_a_time. + +2009-11-28 Jakub Jelinek + + * lto-lang.c (handle_nonnull_attribute): Remove unused attr_arg_num + variable. + +2009-11-19 Rafael Avila de Espindola + + PR bootstrap/42096 + * lto-elf.c (lto_elf_file_open): Use lto_parse_hex. + * lto.c (lto_parse_hex): New. + (lto_resolution_read): Use lto_parse_hex. + * lto.h (lto_parse_hex): New. + +2009-11-17 Rafael Avila de Espindola + + * lto-elf.c (lto_file_init): Add offset argument. + (lto_elf_file_open): Record the offset. + * lto.c (lto_resolution_read): Change file_name into a lto_file + argument. Check offsets. + (lto_file_read): Update call to lto_resolution_read. + * lto.h (lto_file_struct): Add the offset field. + +2009-11-16 Rafael Avila de Espindola + + * lto-elf.c (lto_elf_file_open): Use strtoll to parse the offset. + +2009-11-14 Jan Hubicka + + * lto.c (read_cgraph_and_symbols): Set also ipa_transforms_to_apply. + +2009-11-12 Rafael Avila de Espindola + + * lang.opt (fresolution): Renamed from resolution. + * lto-lang.c (lto_handle_option): Handle new option name. + * lto.c (lto_resolution_read): Add more checks. Discard rest of line. + +2009-11-04 Richard Guenther + Rafael Avila de Espindola + + * lto-elf.c (lto_elf_build_section_table): Add the base offset. + (lto_elf_file_open): Handle offsets in arguments name@offest. + +2009-10-30 Richard Guenther + + PR lto/41858 + * lto.c (lto_file_read): Do not set file_data->fd. + (lto_read_section_data): Use a single-entry file-descriptor cache. + Do not check the result of xmalloc. + (free_section_data): Do not use file_data->fd. + (lto_read_all_file_options): Likewise. + +2009-10-22 Richard Guenther + + * lto.c (lto_fixup_jump_functions): Remove. + (lto_fixup_decls): Do not fixup jump functions. + (read_cgraph_and_symbols): Schedule cgraph merging after + summary reading. Schedule type and decl fixup before + summary reading. + +2009-10-22 Richard Guenther + + * lto.c (lto_fixup_data_t): Remove free_list member. + (lto_fixup_tree): Do not insert into free_list. + (free_decl): Remove. + (lto_fixup_decls): Remove free-list handling. + +2009-10-22 Jan Hubicka + + * lto.c (lto_fixup_jump_functions): New function. + (lto_fixup_decls): Use it. + +2009-10-16 Richard Guenther + + PR lto/41715 + * lto.c (lto_fixup_tree): Revert last change. + +2009-10-14 Richard Guenther + + * lto.c (lto_fixup_tree): In case the prevailing decl is not + compatible with the one we replace wrap it around a + VIEW_CONVERT_EXPR. + +2009-10-09 Richard Guenther + + PR lto/41635 + PR lto/41636 + * lto.c (read_cgraph_and_symbols): Do not assert we can open + a file. + * lto-elf.c (init_shdr##BITS): Fix i18n problems. + (init_ehdr##BITS): Likewise. + +2009-10-08 Joseph Myers + + * lto-elf.c (init_shdr##BITS, lto_elf_begin_section_with_type, + init_ehdr##BITS, lto_elf_file_close): Remove trailing "." from + diagnostics. + * lto-lang.c (lto_post_options): Remove trailing "." from + diagnostics. + +2009-10-08 Richard Guenther + + * lto.c (read_cgraph_and_symbols): Free the gimple type merging + hash tables. + +2009-10-07 Joseph Myers + + * lto.c: Only include if HAVE_MMAP_FILE. + +2009-10-07 Jan Hubicka + + * lto.c (read_cgraph_and_symbols): Mark functions neccesary only at + ltrans stage; explain why this is needed and should not. + +2009-10-05 Richard Guenther + + PR lto/41552 + PR lto/41487 + * lto.c (lto_read_decls): Do not register deferred decls. + (read_cgraph_and_symbols): Delay symbol and cgraph merging + until after reading the IPA summaries. + +2009-10-02 Rafael Avila de Espindola + + * Make-lang.in (lto/lto-lang.o): Don't depend on lto/common.h. + (lto-lang.c): Don't include lto/common.h. + +2009-10-02 Rafael Avila de Espindola + + * Make-lang.in (LTO_OBJS): Remove lto/common.o. + (lto/common.o): Remove. + * common.c: Remove. + * common.h (lto_kind_str): Remove. + (lto_visibility_str): Remove. + (lto_resolution_str): Make it static. + +2009-10-01 Diego Novillo + + * lto.c (lto_read_decls): Add comment. + Call internal_error instead of gcc_assert. + (lto_resolution_read): Likewise. + (lto_add_all_inlinees): Tidy. + * Make-lang.in: Fix copyright boilerplate. + (lto.pdf): New. + (lto.install-pdf): New. + * lto-tree.h: Fix copyright boilerplate. + * lang-specs.h: Likewise. + Remove ".lto" entry from compilers fragment. + * lto-elf.c: Move inclusion of gelf.h after config.h. + Tidy formatting everywhere. + * lto.h: Fix copyright boilerplate. + Tidy formatting everywhere. + * common.c: Likewise. + * config-lang.in: Likewise. + * common.h: Likewise. + * lto-lang.c: Likewise. + +2009-10-01 Richard Guenther + + * lto.c (lto_read_section_data): Use plain lseek/read. + +2009-10-01 Richard Guenther + + * lto.c (LTO_MMAP_IO): Define if we can mmap files and + use sysconf to query the system page size. + (lto_file_read): Implement fallback using stdio. + (free_section_data): Likewise. + +2009-09-29 Diego Novillo + + * lto-lang.c (lto_init): Really fix call to + build_common_builtin_nodes. + +2009-09-29 Diego Novillo + + * lto-lang.c (lto_init): Fix call to + build_common_builtin_nodes. + +2009-09-29 Richard Guenther + + PR lto/40754 + * lto-elf.c (init_shdr##BITS): Properly specify alignment + in bytes. + (first_data_block): New static variable. + (lto_elf_append_data): Align the first data block in each + section. + +2009-09-28 Diego Novillo + + * lto-lang.c: Tidy. Remove stale FIXME lto markers. + * lto.c (strip_extension): New. + (get_filename_for_set): Call it. Do not call make_cwd_temp_file. + (lto_execute_ltrans): Tidy. + Do not pass -fwpa nor -fltrans-* to LTRANS. + * opts.c: Tidy formatting and remove stale FIXME lto markers. + * tree.c (need_assembler_name_p): Call + lang_hooks.decls.may_need_assembler_name_p if set. + * varasm.c (default_binds_local_p_1): Remove check for + flag_ltrans. + * varpool.c (decide_is_variable_needed): Do not test for + in_lto_p. + +2009-09-22 Richard Guenther + + PR lto/39276 + * lto.c (lto_execute_ltrans): Perform ltrans phase manually. + * Make-lang.in: Remove ltrans-driver stuff. + * config-lang.in: Likewise. + * lang.opt (fltrans-driver): Remove. + * lto-lang.c (lto_init_options): Remove code initializing + ltrans_driver. + * ltrans-driver: Remove. + +2009-09-21 Diego Novillo + + * lto-lang.c (lto_define_builtins): Remove superfluous + calls to targetm.init_builtins and build_common_builtin_nodes. + (lto_init): Add targetm.arm_eabi_unwinder as parameter to + build_common_builtin_nodes. + * lto.c (lto_materialize_function): Do nothing if NODE is a + clone. + +2009-09-03 Diego Novillo + + * lto-elf.c (validate_file): Replace call to + elf_getshstrndx with call to elf_getshdrstrndx. + +2009-08-19 Richard Guenther + + * lto-lang.c (lto_init): Merge char_type_node with the + appropriately signed variant. + +2009-08-19 Richard Guenther + + PR lto/41071 + * lto.c (lto_fixup_common): Re-build the pointer-to chain part one. + (lto_fixup_type): Re-build the pointer-to chain part two. + +2009-08-19 Richard Guenther + + PR lto/41071 + * lto.c (lto_fixup_type): Re-build the type variant chain. + +2009-08-19 Richard Guenther + + PR lto/41071 + * lto.c (lto_fixup_constructor): New function. + (lto_fixup_tree): Replace all types. Properly fixup + constructors and constants. + +2009-08-14 Richard Guenther + + * lto.c (read_cgraph_and_symbols): Exchange TREE_CHAIN use + for DECL_LANG_SPECIFIC. + +2009-08-13 Richard Guenther + + PR lto/41032 + * lto-lang.c (LANG_HOOKS_TYPES_COMPATIBLE_P): Define to NULL. + +2009-07-30 Richard Guenther + + PR lto/40903 + * lto.c (read_cgraph_and_symbols): After fixing up decls choose + the largest decl for output and free TREE_CHAIN for further + use. + +2009-07-24 Diego Novillo + + * Make-lang.in: Add empty lto.install-plugin target. + +2009-07-13 Diego Novillo + + * lto.c (lto_fixup_tree): Handle IMPORTED_DECL. + +2009-07-11 Richard Guenther + + * lto-lang.c (lto_write_globals): Wrapup global decls. + +2009-07-10 Richard Guenther + + * lto-lang.c (lto_init): Allocate one more location to make + BUILTINS_LOCATION correct. + +2009-07-09 Rainer Orth + + * lto.c (free_section_data): Cast computed_offset to caddr_t. + +2009-07-06 Diego Novillo + + * lto.c (lto_fixup_type): Fixup TYPE_SIZE and + TYPE_SIZE_UNIT. + +2009-07-06 Diego Novillo + + * lto.c (preload_common_nodes): Remove. + (lto_read_in_decl_state): Call lto_streamer_cache_get. + (lto_read_decls): Call lto_data_in_create and + lto_data_in_delete. + (free_decl): Do not call ggc_free. + (lto_main): Call lto_init_reader. + * lto-lang.c (lto_type_for_size): Handle intTI_type_node. + (lto_init): Initialize main_identifier_node if needed. + Make ptrdiff_type_node be integer_type_node. + +2009-06-19 Diego Novillo + + * lto.c: Remove code guarded by #ifdef LTO_STREAM_DEBUGGING. + Remove code guarded by #ifdef GLOBAL_STREAMER_TRACE. + Remove code guarded by #ifdef LOCAL_TRACE. + +2009-06-18 Diego Novillo + + * lto.c: Update license to GPLv3. + * lto-elf.c: Likewise. + * common.c: Likewise. + * lto-lang.c: Likewise. + * lto.h: Remove superfluous include files. Update all + users. + +2009-06-17 Diego Novillo + + * lto.c (read_cgraph_and_symbols): Call input_cgraph. + +2009-06-02 Diego Novillo + + * lto.c (lto_1_to_1_map): Ignore nodes that have not been + read in. + (materialize_cgraph): Only materialize nodes that have a + representation on file. + +2009-06-01 Diego Novillo + + * lto-lang.c (lto_handle_option): Hanlde OPT_Wabi. + +2009-05-31 Diego Novillo + + * lto-lang.c (lto_type_for_mode): Handle all the modes + handled in c_common_type_for_mode. + +2009-05-21 Diego Novillo + + * lto-elf.c: Always include . + * config-lang.in (target_libs): Remove. + (build_by_default): Set to no. + +2009-05-15 Diego Novillo + + * lto.c (lto_materialize_function): Assert that DECL is + not a builtin. + (materialize_cgraph): Don't try to materialize builtin + functions. + * lto-section-out.c (write_symbol_vec): Do not write + builtin functions. + +2009-05-13 Diego Novillo + + * lto-lang.c (LANG_HOOKS_GET_ALIAS_SET): Define. + +2009-05-07 Diego Novillo + + * lto.c (lto_resolution_read): Add type casts for C++ warnings. + (LTO_REGISTER_TYPE_AND_FIXUP_SUBTREE): Define. + (lto_fixup_type): Call it for TYPE_POINTER_TO, + TYPE_REFERENCE_TO, TYPE_CONTEXT and TYPE_CANONICAL. + (lto_fixup_tree): Call gimple_register_type when *TP is a + type. + (lto_main): Call bitmap_obstack_initialize. + +2009-04-22 Diego Novillo + + * lto.c (free_section_data): Tidy. + (lto_1_to_1_map): Tidy. + (lto_add_all_inlinees): Tidy. + (prefix_name_with_star): New. + (get_filename_for_set): New. + (lto_wpa_write_files): Call cgraph_node_set_needs_ltrans_p + to determine what cgraph node sets to write. + Call get_filename_for_set to compute temporary file + names. + (lto_execute_ltrans): Do not execute LTRANS on files with + names that start with '*'. + Move logic to execute LTRANS together so that LTRANS is + invoked only if there are any files to compile. + (do_whole_program_analysis): Only remove output files + that do not start with '*'. + +2009-04-06 Diego Novillo + + * lto-lang.c (lto_post_options): Set flag_excess_precision_cmdline. + * lto.c (read_cgraph_and_symbols): Set cgraph_function_flags_ready. + (lto_add_all_inlinees): Tidy. + +2009-03-26 Diego Novillo + + * lto.c: Include gimple.h. + (lto_read_in_decl_state): Call gimple_register_type for + every type in every stream. + (lto_fixup_common): Call gimple_register_type if T has a + type. + (do_whole_program_analysis): Call print_lto_report. + (lto_main): Call print_lto_report after cgraph_optimize. + * Make-lang.in (lto.o): Add dependency on GIMPLE_H. + +2009-03-24 Diego Novillo + + * Make-lang.in (lto-lang.o): Add dependency on TARGET_H and EXPR_H. + (lto.o): Add dependency on GIMPLE_H. + +2009-03-10 Simon Baldwin + + * lto.c (lto_read_all_file_options): Close any open file descriptor + contained in file_data before freeing. + +2009-02-24 Rafael Avila de Espindola + + * lto.c (lto_add_inline_clones): Don't add the master clone. Check + for a decl in the original bitmap, not a node. + (lto_add_all_inlinees): Remove original nodes that are not needed. + (lto_scan_statics_in_cgraph_node): Don't care if the node is the master. + +2009-02-24 Diego Novillo + + * lto.c (lto_materialize_function): Update + lto_stats.num_function_bodies. + (get_section_data): Initialize *LEN to 0. + (lto_1_to_1_map): Update lto_stats.num_cgraph_partitions. + (lto_wpa_write_files): Update lto_stats.num_cgraph_nodes. + Update lto_stats.num_output_files. + (read_cgraph_and_symbols): Update lto_stats.num_input_files. + (materialize_cgraph): Update lto_stats.num_input_cgraph_nodes. + (lto_main): Initialize lto_stats. + If flag_lto_report is set, call print_lto_report. + +2009-02-19 Diego Novillo + + Revert + + 2009-02-19 Rafael Avila de Espindola + + * lto.c (lto_add_inline_clones): Don't add the + master clone. Check for a decl in the original + bitmap, not a node. + (lto_add_all_inlinees): Remove original nodes + that are not needed. + (lto_scan_statics_in_cgraph_node): Don't care if + the node is the master. + (lto_promote_cross_file_statics): Use a new + context.seen_node_decls for each set + +2009-02-19 Rafael Avila de Espindola + + * lto.c (lto_add_inline_clones): Don't add the master clone. Check + for a decl in the original bitmap, not a node. + (lto_add_all_inlinees): Remove original nodes that are not needed. + (lto_scan_statics_in_cgraph_node): Don't care if the node is the master. + (lto_promote_cross_file_statics): Use a new context.seen_node_decls + for each set + +2009-02-18 Diego Novillo + + * lto.c (lto_wpa_write_files): Use timers TV_WHOPR_WPA + and TV_WHOPR_WPA_IO. + (lto_execute_ltrans): Use timer TV_WHOPR_WPA_LTRANS_EXEC. + (read_cgraph_and_symbols): Use timer TV_IPA_LTO_DECL_IO. + (materialize_cgraph): Use timer TV_IPA_LTO_GIMPLE_IO. + Use timer TV_WHOPR_WPA or TV_WHOPR_LTRANS or TV_LTO + depending on command line flags. + (do_whole_program_analysis): Use timer TV_WHOPR_WPA. + (lto_main): Remove timer uses. + +2009-02-18 Rafael Avila de Espindola + + * lto.c (lto_materialize_function): Don't set DECL_EXTERN to 0. + (lto_wpa_write_files): Update calls to renamed functions. + +2009-02-17 Diego Novillo + + PR 39203 + * lto-lang.c (lto_post_options): Disable -fwhole-program + when running LTRANS. + +2009-02-10 Diego Novillo + + * lto.c (read_cgraph_and_symbols): Fix comment. + +2009-02-10 Diego Novillo + + * lto.c (read_cgraph_and_symbols): Read options from all + IL files. + +2009-02-10 Diego Novillo + + * lto.c (read_cgraph_and_symbols): Factor out of lto_main. + (materialize_cgraph): Likewise. + (do_whole_program_analysis): Likewise. + (lto_main): Call read_cgraph_and_symbols, + materialize_cgraph and do_whole_program_analysis. + +2009-02-10 Simon Baldwin + + * lto.c: Include lto-opts.h. + * (lto_main): Clear file options at loop start, read any saved + options from the first file handled, and re-issue options. + * Makefile.in (lto.o): Add dependency on lto-opts.h. + +2009-02-02 Diego Novillo + + * lto.c (lto_main): Stop LTO_TIMER and use + TV_WHOPR_WPA_LTRANS_EXEC when launching LTRANS. + +2009-01-30 H.J. Lu + + PR lto/38995 + * lto-elf.c (init_shdr##BITS): Set the sh_addralign field + to POINTER_SIZE. + +2009-01-29 Ramana Radhakrishnan + + * Make-lang.in (LTO_EXE): Link with all + BACKENDLIBS and not only GMPLIBS + +2009-01-28 H.J. Lu + + PR bootstrap/38992 + * lto-elf.c: Include gelf.h instead of libelf.h. + (lto_elf_file_close): Replace elfx_update_shstrndx with + gelf_getehdr, elf_getscn, gelf_getshdr, gelf_update_shdr and + gelf_update_ehdr. + +2009-01-28 H.J. Lu + + PR middle-end/38996 + * lto-elf.c (DEFINE_INIT_EHDR): Initialize e_version. + +2009-01-26 Diego Novillo + + * lto-lang.c (LANG_HOOKS_TYPES_COMPATIBLE_P): Update. + +2009-01-26 Diego Novillo + + * lto-lang.c (lto_types_compatible_p): Move to gimple.c + and rename into gimple_types_compatible_p. + +2009-01-12 Rafael Avila de Espindola + + * lto-lang.c (lang_hooks): Remove the const qualifier. + +2009-01-06 Diego Novillo + + * ltrans-driver: Mark 'all' target as phony. + +2008-12-31 Diego Novillo + + * ltrans-driver: Execute a NOP action for target 'all'. + +2008-12-19 Diego Novillo + + * lto.c (lto_1_to_1_map): Tidy. + +2008-12-19 Diego Novillo + + * lto-elf.c (lto_elf_file_open): When FILENAME cannot + be opened, show its name. + * ltrans-driver: If $verbose is set, do not use parallelism. + +2008-12-17 Rafael Avila de Espindola + + * lto.c (lto_fixup_function): New. + (lto_fixup_tree): Call lto_fixup_function. + +2008-12-14 Doug Kwan + + * lto.c (lto_1_to_1_map): Create a cgraph node set for any global + variables if there is no function. + +2008-12-10 Simon Baldwin + + * ltrans-driver: Always run make in silent mode, to avoid make's + trace on stdout interfering with lto-wrapper output. + +2008-12-10 Doug Kwan + + * lto.c (lto_add_inline_clones): Do not force master clones of + inlined functions already in SET to be static inline. + +2008-12-04 Doug Kwan + + * lto.c (globalize_context_t): New type to store states in + globalization of cross-file statics. + (globalize_cross_file_statics): New. + (lto_scan_statics_in_ref_table): Walk tree to look for reachable + static decls that need to be fixed up. + (lto_scan_statics_in_cgraph_node): Change call interface to use + a globalize_context_t CONTEXT for all states used. + (lto_scan_statics_in_remaining_global_vars): New. + (lto_promote_cross_file_statics): Use new call interface of + LTO_SCAN_STATICS_IN_CGRAPH_NODE. Handle remaining externally + visible vars in the last set. + +2008-12-03 Diego Novillo + + * lto.c (lto_fixup_tree): Do not emit an error when + PREVAILING throw but T doesn't. + +2008-12-02 Doug Kwan + + * lto.c (lto_scan_statics_in_ref_table): New function factored out + from code in ... + (lto_scan_statics_in_cgraph_node): Handle both file-scope static + variables and functions. + (lto_promote_cross_file_statics): Rename bitmaps to SEEN_DECLS + and GLOBAL_DECLS from SEEN_VARS and GLOBAL_VARS. + +2008-11-29 Diego Novillo + + * lto.c: Include timevar.h. + (lto_materialize_function): Tidy. Add comments. + (lto_wpa_write_files): Tidy. + (lto_execute_ltrans): Tidy. + (lto_main): Add local variable LTO_TIMER. Initialize it + to one of TV_WHOPR_WPA, TV_WHOPR_LTRANS or TV_LTO. + Start and stop the timer. + Tidy comments. + * Make-lang.in (lto.o): Add dependency on timevar.h. + * ltrans-driver: React to -v and -save-temps. + Use simple heuristic to determine how much parallelism to + use when executing make. + +2008-11-12 Doug Kwan + + * lto.c (lto_bitmap_obstack): Remove var. + (lto_materialize_function): Do nothing instead of marking function + body in file if flag_wpa is true. + (lto_add_all_inlinees): Use bitmap functions in lto-utils.c. + (lto_scan_statics_in_cgraph_node): New function. + (lto_promote_cross_file_statics): Same. + (lto_wpa_write_files): Call lto_promote_cross_file_statics. + Use bitmap functions in lto-utils.c. Remove unsued label OUT. + * Make-lang.in (lto/lto.o): Add lto-utils.h to dependency list. + +2008-11-09 Diego Novillo + + * lto/lto.c (lto_fixup_tree): Change error message locus + information to include location of mismatching + declaration. + Use TREE_NO_WARNING to avoid repeated messages. + (lto_main): If lto_fixup_decls emitted any errors, exit. + * lto/lto-lang.c: Don't include libfuncs.h and except.h + (lto_init_options): Don't enable exceptions by default. + (lto_eh_runtime_type): Move to lto-function-in.c + (lto_init_eh): Likewise. + (lto_init): Don't call lto_init_eh. + * lto/Make-lang.in (lto-lang.o): Remove dependency on + libfuncs.h and except.h. + +2008-10-30 Diego Novillo + + * lto.c (lto_read_decls): Declare debug_main only if + LTO_STREAM_DEBUGGING is enabled. + +2008-10-30 Simon Baldwin + + * lto.c (lto_wpa_write_files): Create intermediate files with + make_cwd_temp_file(). + (lto_maybe_unlink): New. Delete intermediate WPA files unless + WPA_SAVE_LTRANS is set. + (lto_main): Call lto_maybe_unlink() for intermediate WPA files. + * ltrans-driver: Do not strip directory from output files. + +2008-10-29 Doug Kwan + + * lto.c (free_decl): Call lto_symtab_clear_resolution when freeing + DECL. + * Make-lang.in (LTO_OBJS): Remove lto/lto-symtab.o + (lto/lto-symtab.o): Remove rule. + * lto-tree.h (struct lang_identifier): Remove LTO specific fields. + (struct lang_decl): Remove RESOLUTION and add DUMMY in struct. + (LANG_IDENTIFIER_CAST, LTO_IDENTIFIER_DECL, LTO_DECL_RESOLUTION): + Remove macros. + lto-symtab.c (File): Move up one level. + lto-lang.c (cgraph.h): Remove include. + (input_overwrite_node, input_node, input_edge, input_cgraph_1, + input_cgraph): Move to lto-cgraph.c in gcc directory above. + (LANG_HOOKS_INPUT_CGRAPH): Remove use of macro. + +2008-10-24 Rafael Espindola + + * lto-function-in.c (get_resolution): Return LDPR_PREEMPTED_IR for + non prevailing weak symbols. + +2008-10-24 Rafael Espindola + + * lto-lang.c (input_cgraph_1): Iterate over nodes, not cgraph_nodes. + +2008-10-24 Rafael Espindola + + * lto-lang.c (input_node): Avoid casts from pointers to ints of + different types. + +2008-10-23 Simon Baldwin + + * lto-lang.c (input_node): Save the node reference, rather than the + node pointer, in node->inlined_to. + (input_cgraph_1): Convert node references into node pointers. + +2008-10-22 Diego Novillo + Rafael Espindola + + * lto.c (lto_resolution_read): Tidy. + * lto-symtab.c (lto_symtab_prevailing_decl): Do not + abort if RET is NULL. + +2008-10-22 Doug Kwan + + * lto.c (lto_fixup_tree): Check for NOTHROW conflict only if + exceptions flag is given. + * lto-lang.c: (lto_init_options) Set default exceptions flag. + (lto_init_eh): Remove exceptions flag initialization. + (lto_init): Only call lto_init_eh if exceptions flag is set. + +2008-10-21 Diego Novillo + + * lto.c: Tidy some formatting. + * lto.h: Likewise. + +2008-10-21 Simon Baldwin + + * lto-symtab.c: (lto_same_type_p): Types cannot be equal if one of + them is NULL (but not the other). + +2008-10-17 Diego Novillo + + * ltrans-driver: Divert output from make to a temporary file. + Show it if the call to make failed. + +2008-10-15 Diego Novillo + + * lto.c (lto_wpa_write_files): Reformat do-while loop. + Do not print TEMP_FILENAME + * ltrans-driver: Call make with -s. + +2008-10-15 Diego Novillo + + * lto-symtab.c (lto_symtab_merge_decl): Do not force + TREE_STATIC on global symbols. + +2008-10-14 Ollie Wild + + * Make-lang.in (LTRANS_DRIVER_INSTALL_NAME): Remove. + (LTRANS_DRIVER_EXE): Add. + (lto.all.cross): Add LTRANS_DRIVER_EXE. + (lto.all.encap): Add LTRANS_DRIVER_EXE. + (lto.install.common): Remove ltrans-driver. + (lto.mostlyclean): Add LTRANS_DRIVER_EXE. + (LTRANS_DRIVER_EXE): New build rule. + * config-lang.in (compilers): Add ltrans-driver. + +2008-10-14 Diego Novillo + + * Make-lang.in (LTRANS_DRIVER_INSTALL_NAME): Disable transformation + of program name. + +2008-10-13 Ollie Wild + + * lang-spec.h (@lto): Replace lto1_options with cc1_options. + * lto.c (lto_execute_ltrans): Add "-fno-wpa -fltrans -xlto" to CFLAGS. + * ltrans-driver (LTRANS_FLAGS): Remove. + +2008-10-08 Doug Kwan + + * lto.c (lto_fixup_tree): Remove ATTRIBUTE_UNUSED from DATA. + Handle new tree codes RECORD_TYPE, UNION_TYPE, QUAL_UNION_TYPE + and TREE_BINFO. Also move code handling FUNCTION_DECL and VAR_DECL + from lto_fixup_state to here. + (lto_fixup_state): Take an lto_fixup_data_t object DATA instead of + just a free-list. Fix up types also. Move decl merging code to + lto_fixup_tree. + (lto_fixup_state_aux): Change AUX to point to an lto_fixup_data_t + object. + (lto_fixup_decls): Use another pointer set to avoid multiple + walking of nodes except for DECLs to be replaced. Pass an + lto_fixup_data_t object to tree-walker. + +2008-10-08 Rafael Espindola + + * lto-symtab.c (lto_symtab_set_resolution): New. + (lto_symtab_merge_decl): Use lto_symtab_set_resolution and + lto_symtab_get_resolution. + (lto_symtab_prevailing_decl): Return decl for non public decls. + (lto_symtab_get_resolution): New. + * lto.c (lto_fixup_tree, lto_fixup_state): Remove unecessary checks. + +2008-10-06 Rafael Espindola + + * lto-lang.c: Include cgraph.h. + (input_overwrite_node, input_node, input_edge, input_cgraph_1, + input_cgraph): Moved from lto-cgraph.c. + (LANG_HOOKS_INPUT_CGRAPH): New. + +2008-10-03 Rafael Espindola + + * lto.c (lto_fixup_tree, lto_fixup_state): Fix the FIXME. + +2008-10-03 Rafael Espindola + + * lto-symtab.c (lto_symtab_overwrite_decl): Remove. Remove all calls. + (lto_symtab_merge_decl): Update LTO_IDENTIFIER_DECL the reflect the + prevailing definition. Don't mark TREE_NOTHROW differences. + * lto.c (lto_fixup_tree): New. + (lto_fixup_state): New. + (lto_fixup_state_aux): New. + (free_decl): New. + (lto_fixup_decls): New. + (lto_main): Call lto_fixup_decls. + +2008-10-02 Ollie Wild + + * lang.opt (fltrans): Moved from common.opt. Remove RejectNegative + and Init. + (fwpa): Moved from common.opt. Remove RejectNegative and Init. + * lto-lang.c (lto_post_options): Add validation and fixups for + -fltrans and -fwpa. + +2008-10-02 Rafael Espindola + + * lto-symtab.c (lto_symtab_merge_var, lto_symtab_merge_fn, + lto_symtab_merge_decl): Return void. + (lto_symtab_prevailing_decl): New. + +2008-09-30 Rafael Espindola + + * lto-symtab.c (lto_symtab_compatible): Remove the check for already + defined symbols. + (lto_symtab_overwrite_decl): Copy LTO_DECL_RESOLUTION. + (lto_symtab_merge_decl): Store symbol resolution in LTO_DECL_RESOLUTION. + Check for already defined symbols. + * lto-tree.h (lang_decl): Remove dummy and add resolution fields. + (LTO_IDENTIFIER_RESOLUTION): Remove. + (LTO_DECL_RESOLUTION): New. + +2008-09-30 Rafael Espindola + + * lto.c (lto_read_decls): Use new input_tree signature. + +2008-09-26 Doug Kwan + + * lto.c (lto_main): Call lto_fixup_nothrow_decls to fix up function + bodies affected by exception attribute merging of DECLs. + * lto-symtab.c (lto_symtab_merge_decl): Handle exception attribute + merging. + +2008-09-25 Rafael Espindola + + * Make-lang.in (PLUGIN_API_H, LTO_TREE_H): New. + (lto/lto-lang.o, lto/lto.o, lto/lto-symtab.o) Use LTO_TREE_H. + * lto-symtab.c (lto_symtab_compatible): New. + (lto_symtab_overwrite_decl): New. + (lto_symtab_merge_decl): Refactor to use the above functions + and the resolution from lang_identifier. + * lto-tree.h: Include plugin-api.h. + (lang_identifier): Add resolution. + (LTO_IDENTIFIER_RESOLUTION): New. + +2008-09-25 Ollie Wild + + * lang.opt (fltrans-output-list=): New option. + * lto.c (lto_execute_ltrans): Output file names to ltrans_output_list. + +2008-09-25 Rafael Espindola + + * lto.c (lto_resolution_read): Initialize ret; + +2008-09-24 Ollie Wild + + * lto.c (sys/mman.h): Move include. + (lto_wpa_write_files): Return a list of written files. + (lto_execute_ltrans): New function. + (lto_main): Call lto_execute_ltrans. + (ltrans-driver): New file. + * lto-lang.c (DEFAULT_LTRANS_DRIVER): New macro. + (DEAULT_LTRANS_DRIVER_LEN): New macro. + (lto_init_options): Initialize ltrans_driver. + (lto_handle_option): Fix incorrect default output value. + * lang.opt (fltrans-driver=): New option. + * Make-lang.in (LTRANS_DRIVER_INSTALL_NAME): New variable. + (lto.install-common): Add lto/ltrans-driver. + +2008-09-24 Rafael Espindola + + * Make-lang.in (LTO_OBJS): Add lto/common.o. + (lto/lto.o): Depend on lto/common.h. + (lto/common.o): New. + * lang.opt (resolution): New. + * lto-lang.c (resolution_file_name): New. + (lto_handle_option): Handle OPT_resolution. + * lto-symtab.c (lto_symtab_merge_decl): Add a resolution argument. + (lto_symtab_merge_var,lto_symtab_merge_fn): Add a resolution argument. + pass it to lto_symtab_merge_decl. + * lto.c: Include common.h. + (lto_read_decls): Add resolutions and resolutions_size arguments. + Initialize data_in.globals_resolution and + data_in.globals_resolution_size. + (index_and_symbol_resolution): New. + (lto_resolution_read): New. + (lto_file_read): Add argument resolution_file. + Read resolution. + * lto.h (resolution_file_name): New. + +2008-09-23 Rafael Espindola + + * common.c: Update description. + * common.h: Update description. + +2008-09-23 Rafael Espindola + + * common.c: Moved from lto-plugin. + * common.h: Moved from lto-plugin. + +2008-09-22 Doug Kwan + + * lto.c (VEC(bitmap,heap)): Declare. + (lto_materialize_function): Handle WAP mode specially. + (lto_add_inline_clones): New. + (lto_add_all_inlinees): Changle algorithm and to use bitmaps. Also + return a bitmap of inlined decls. + (lto_wpa_write_files): Handle all DECLs brought in by inlining. + (lto_main): Call reset_inline_failed to reset inlining states. + Check call-graph after WPA inlining. + * lto-lang.c (lto_init): Do not clear flag_generate_lto + unconditionally. + +2008-09-19 Doug Kwan + + lto.c (lto_main): Remove unsued wrapper code. + lang-specs.h (@lto): Use lto1_options instead of cc1_options. + +2008-09-19 Rafael Espindola + + * lto-symtab.c: Include lto-tree-in.h. + * lto-tree.h (lto_symtab_merge_var, lto_symtab_merge_fn): Remove. + * lto.h (lto_symtab_merge_var, lto_symtab_merge_fn): Remove + * Make-lang.in (lto/lto-symtab.o): Add lto-tree-in.h. + +2008-09-17 Paolo Bonzini + Rafael Avila de Espindola + + * lto-lang.c (COMPOUND_LITERAL_EXPR_DECL_STMT, + COMPOUND_LITERAL_EXPR_DECL): Remove. + (emit_local_var): Remove. + (lto_expand_expr): Remove. + (lto_staticp): Remove. + (LANG_HOOKS_EXPAND_EXPR): Remove. + (LANG_HOOKS_STATICP): Remove. + +2008-09-11 Diego Novillo + + * lto-lang.c: Include except.h and libfuncs.h. + (lto_init_eh): New. + (lto_init): Call it. + Set flag_generate_lto to 0. + * Make-lang.in (lto-lang.o): Add dependency on except.h + and libfuncs.h. + +2008-09-09 Bill Maddox + + * lto-lang.c: Include header file expr.h. + (COMPOUND_LITERAL_EXPR_DECL_STMT, + COMPOUND_LITERAL_EXPR_DECL): Copied from c-common.h. + (emit_local_var): Copied from c-semantics.c. + (lto_expand_expr, lto_staticp): Copied from c_expand_expr + and c_staticp in c-common.c. + (LANG_HOOKS_EXPAND_EXPR,LANG_HOOKS_STATICP): Redefined. + +2008-09-08 Diego Novillo + + * lto-lang.c (lto_global_bindings_p): Return 1 during + IPA passes. + +2008-09-07 Diego Novillo + + * lto.c: Tidy formatting. + +2008-08-04 Bill Maddox + + * lto-symtab.c (lto_symtab_merge_decl): Add comment. + +2008-09-03 Doug Kwan + + lto.c (lto_add_all_inlinees): Reset FAILED_REASON of edges to + CIF_OK instead of NULL. + +2008-09-02 Diego Novillo + Simon Baldwin + + * lto-lang.c (lto_type_for_size): Rewrite. Adapt from + c_common_type_for_size. + (lto_type_for_mode): Remove ATTRIBUTE_UNUSED markers. + (lto_init): Call linemap_add. + (signed_and_unsigned_types): Remove. + +2008-08-29 Diego Novillo + + * lto-lang.c (handle_noreturn_attribute): New local function. + (handle_const_attribute): New local function. + (handle_malloc_attribute): New local function. + (handle_pure_attribute): New local function. + (handle_novops_attribute): New local function. + (handle_nonnull_attribute): New local function. + (handle_nothrow_attribute): New local function. + (handle_sentinel_attribute): New local function. + (handle_type_generic_attribute): New local function. + (handle_format_attribute): New local function. + (handle_format_arg_attribute): New local function. + (lto_attribute_table): Declare. + (lto_format_attribute_table): Declare. + (lto_init_attributes): New local function. + (lto_define_builtins): Call it. + Call targetm.init_builtins and build_common_builtin_nodes. + (LANG_HOOKS_COMMON_ATTRIBUTE_TABLE): Define. + (LANG_HOOKS_FORMAT_ATTRIBUTE_TABLE): Define. + +2008-08-28 Diego Novillo + + * Make-lang.in (lto-lang.o): Replace tree-gimple.h with + $(GIMPLE_H). + (lto-symtab.o): Add dependency on $(GIMPLE_H). + * lto-lang.c: Include gimple.h instead of tree-gimple.h. + * lto-symtab.c: Include gimple.h. + * lto-tree.h (chain_next): Replace GENERIC_NEXT with + TREE_CHAIN. + +2008-08-27 Doug Kwan + + * lto.c (vec.h, bitmap.h, pointer-set.h, ipa-prop.h, ggc.h, + gt-lto-lto.h): New includes. + (lto_materialize_function): Do not read in function body in WPA mode. + Format a line to fit in 80 columns. + (lto_cgraph_node_sets): New garbage collected variable. + (lto_1_to_1_map, lto_add_all_inlinees, lto_wpa_write_files): + New functions. + (lto_main): Initialize bitmap obstack. Add code to handle WPA mode. + * Make-lang.in (LTO_H): Replace filename lto-section-in.h with + variable LTO_SECTION_IN_H. + (lto/lto.o): Include gt-lto-lto-c.h ggc.h ,VEC_H, BITMAP_H, + pointer-set.h and IPA_PROP_H. Also replace filename lto-section-in.h + with variable LTO_SECTION_IN_H. + * config-lang.in (gtfiles): Add lto/lto.c. + * lto-symtab.c (lto_symtab_merge_decl): Set DECL_CONTEXT of + merged DECL_RESULT correctly. + +2008-08-26 Bill Maddox + + * lto-lang.c Include tree-gimple.h. + (lto_mark_addressable): Call mark_addressable rather than + asserting. + (lto_post_options): Suppress debug info generation. + * Make-lang.in: Add dependency of lto-lang.o on tree-gimple.h. + +2008-08-25 Bill Maddox + + * lto-symtab.c (lto_symtab_merge_decl): Remove a suspect + assertion and leave an explanatory comment in its place. + +2008-08-21 Doug Kwan + + * lto.c (preload_common_nodes): Call lto_get_common_nodes to get a list + of common nodes instead of computing locallly. + (lto_read_in_decl_state): New. + (lto_read_decls): Change code for udpate in struct lto_decl_header. + Read global and per-function in-decl states. + * Make-lang.in (LTO_H): Update dependency. + (lto/lto.o): Same. + (lto-symtab.c): Merge (revision 139039) + * lto-symtab.c (lto_symtab_merge_decl): Merge DECL_RESULT. + +2008-08-21 Rafael Espindola + + * config-lang.in (target_libs): New. + +2008-08-20 Bill Maddox + + * lto.c (current_lto_file): Remove GTY marker from static + variable. Remove include of file gt-lto-lto.h. + * Make-lang.in: Remove dependency of lto/lto.o on + gt-lto-lto.h. + * lto-elf.c (lto_file_close): Removed. + (lto_elf_file_open): Use XCNEW instead of GGC_CNEW to + allocate lto_elf_file object. + (lto_elf_file_close): Free lto_elf_file object after close. + * lto.h (struct lto_file_struct): Remove GTY marker. + * config-lang.in: Remove lto/lto.h and lto/lto.c from + gtfiles. + +2008-08-20 Bill Maddox + + * lto.c (lto_read_decls): Provide dummy argument to input_tree + to conform to its new signature. + * lto-symtab.c (lto_symtab_merge_decl): Do not invoke ggc_free + on discarded node here, now called in global_vector_fixup. + +2008-08-09 Bill Maddox + + * lto.c (preload_common_nodes): Verify that fileptr_type_node + has not been set to a front-end-specific value. + +2008-08-05 Doug Kwan + + * Make-lang.in (lto-symtab.o): Add missing dependencies to fix + build breakage. + +2008-07-30 Bill Maddox + + * lto.c (lto_materialize_function): Call lto_original_decl_name. + Remove obsolete comments. + (lto_read_decls): Remove initialization of deleted field data_in.global. + Tidy up comments. + (lto_main): Remove redundant initialization of section_hash_table. + * lto-elf.c: Removed obsolete comments. + * lto.h: Tidy up comments. + * lto-symtab.c (lto_least_common_multiple): New function. + (lto_symtab_merge_decl): Merge variable alignments in some cases. + Tidy up comments. + +2008-07-25 Diego Novillo + Bill Maddox + + * lto.c: Re-order include files. + Include lto-section-out.h. + (preload_common_nodes): Add debugging output. + Add new local INDEX_TABLE. + Call preload_common_node. + * Make-lang.in (lto.o): Add dependency on lto-section-out.h + +2008-07-13 Bill Maddox + + * lto.c (lto_read_decls): Cast pointer to const char * to avoid + unwanted scaling during pointer addition. + +2008-07-11 Bill Maddox + Diego Novillo + + * lto.c (lto_read_decls): Fix C++ compatibility warnings. + Make code const-correct. + (lto_file_read): Fix C++ compatibility warnings. + (lto_read_section_data): Fix C++ compatibility warnings. + (lto_get_section_data): Use CONST_CAST to avoid warning when + const pointer passed to free. + * lto-elf.c (lto_elf_build_section_table): Fix C++ + compatibility warnings. + (lto_elf_append_data): Fix C++ compatibility warnings. Use CONST_CAST + to avoid warning assigning const pointer to d_buf field of Elf_Data. + (lto_get_current_out_file): Fix C++ compatibility warnings. + +2008-07-11 Diego Novillo + + * Make-lang.in (lto-warn): Define. + +2008-07-03 Simon Baldwin + + * lto.c (lto_read_decls): Wrapped debug-only data items within #ifdef + LTO_STREAM_DEBUGGING. + +2008-06-27 Ollie Wild + + * lto-elf.c (lto-section-out.h): New include. + (struct lto_elf_file): Remove bits member. Add scn, shstrtab_stream, + and data members. + (cached_file_attrs): New static variable. + (lto_elf_get_shdr, lto_elf_free_shdr): Remove elf_file parameter. + Use cached_file_attrs for checking bits. + (lto_elf_build_section_table): Remove elf_file argument from + lto_elf_get_shdr and lto_elf_free_shdr calls. + (DEFINE_INIT_SHDR): New macro. + (init_shdr32, init_shdr64): New functions defined via the + DEFINE_INIT_SHDR macro. + (lto_elf_begin_section_with_type): New function. + (lto_elf_begin_section): New function. + (lto_elf_append_data): New function. + (lto_elf_end_section): New function. + (DEFINE_VALIDATE_EHDR): New macro. + (validate_ehdr32, validate_ehdr64): New functions defined via the + DEFINE_VALIDATE_EHDR macro. + (validate_file): New function. + (DEFINE_INIT_EHDR): New macro. + (init_ehdr32, init_ehdr64): New functions defined via the + DEFINE_INIT_EHDR macro. + (init_ehdr): New function. + (lto_elf_file_open): Add support for writable files. Move some + validation logic to validate_file. + (lto_elf_file_close): Add support for writable files. Write file data + and free data blocks. + (current_out_file): New static variable. + (lto_set_current_out_file): New function. + (lto_get_current_out_file): New function. + * lto.c (lto_main): Add writable argument to lto_elf_file_open calls. + Add temporary initialization for testing ELF serialization. + * lto.h (lto-section-out.h): New include. + (struct lto_file_struct): Slight modification to comment. + (lto_elf_file_open): Add writable parameter. + (lto_elf_begin_section): New function declaration. + (lto_elf_append_data): New function declaration. + (lto_elf_end_section): New function declaration. + (lto_set_current_out_file, lto_get_current_out_file): New function + declarations. + * lto-lang.c (LANG_HOOKS_BEGIN_SECTION): Set as lto_elf_begin_section. + (LANG_HOOKS_APPEND_DATA): Set as lto_elf_append_data. + (LANG_HOOKS_END_SECTION): Set as lto_elf_end_section. + * Make-lang.in (LTO_H): Add lto-section-out.h. + +2008-06-12 Ollie Wild + + * lto.h (struct lto_file_vtable_struct): Remove. + (struct lto_file_struct): Remove vtable member. + * lto-elf.c (lto_file_init): Remove vtable argument. + (lto_elf_map_optional_lto_section): Remove. + (lto_elf_unmap_section): Remove. + (lto_elf_file_vtable): Remove. + (lto_elf_file_open): Remove lto_elf_file_vtable argument from + lto_file_init call. + (lto_elf_find_section_data): Remove. + +2008-06-11 Ollie Wild + + * lto.c (lto_file_read): Add const qualifier to data variable. + +2008-06-11 Diego Novillo + + Merge from lto-streamber sub-branch. + + 2008-06-04 Ollie Wild + + * lto.c: Remove inclusion of dwarf2.h and dwarf2out.h. + * Make-lang.in (lto.o): Remove dependency on dwarf2.h. + + 2008-05-28 Bill Maddox + + Replace the DWARF reader in the LTO front-end. + + * lto.c: Include lto-tree-in.h, lto-tags.h. + (enum DWARF2_class, DW_cl_constant, struct + DWARF2_form_data, struct lto_context, + lto_fd_init, lto_info_fd_init, + lto_abbrev_fd_init, lto_info_fd_close, + lto_file_init, lto_file_close, + lto_file_corrupt_error, lto_abi_mismatch_error, + LTO_CHECK_INT_VAL, LTO_READ_TYPE, + lto_read_uleb128, lto_read_sleb128, + lto_read_initial_length, lto_abbrev_read_attrs, + lto_abbrev_read, lto_abbrev_read_lookup, + lto_read_section_offset, + lto_read_comp_unit_header, find_cu_for_offset, + lto_get_file_name, + lto_resolve_reference,lto_read_form, + attribute_value_as_int, + make_signed_host_wide_int, + attribute_value_as_constant, lto_cache_hash, + lto_cache_eq, lto_cache_store_DIE, + lto_cache_lookup_DIE, lto_find_integral_type, + lto_find_integral_type_1, + LTO_BEGIN_READ_ATTRS_UNCHECKED, + LTO_BEGIN_READ_ATTRS, LTO_END_READ_ATTRS, + lto_unsupported_attr_error, lto_get_identifier, + lto_read_referenced_type_DIE, + lto_read_compile_unit_DIE, + lto_read_array_type_DIE, + lto_read_structure_union_class_type_DIE, + lto_read_enumerator_DIE, + lto_read_enumeration_type_DIE, + lto_read_only_for_child_DIEs, + lto_read_only_for_child_DIEs, + lto_read_member_DIE, lto_read_abbrev, + lto_read_variable_formal_parameter_constant_DIE, + lto_get_body): Removed. + (preload_common_nodes): New function. + (lto_read_decls): Convert for new global streamer. + (lto_materialze_file_data, + lto_read_subroutine_type_subprogram_die, + lto_read_unspecified_parameters_DIE, + lto_read_typedef_DIE, + lto_read_pointer_reference_type_DIE, + lto_read_subrange_type_DIE, + lto_read_base_type_DIE, + lto_read_const_volatile_restrict_type_DIE, + lto_read_namespace_DIE, + lto_read_unspecified_type_DIE, lto_read_DIE, + lto_read_child_DIEs, lto_collect_child_DIEs): + Removed. + (lto_info_read, lto_set_cu_context): Removed. + (lto_file_read): Convert for new global streamer. + (lto_resolve_type_ref, lto_read_DIE_at_ptr, + lto_resolve_var_ref, lto_resolve_fn_ref, + lto_resolve_field_ref, lto_resolve_typedecl_ref, + lto_resolve_namespacedecl_ref): Removed. + (lto_file_init, lto_file_close): Moved to lto-elf.c. + * lto-tree.h (lto_symtab_merge_var, + lto_symtab_mergee_fun): Declare here. + * lto-elf.c (lto_file_init, lto_file_close): Moved from lto.c. + (lto_elf_file_open): Removed code to read DWARF debug sections. + * lto.h (lto_context, DWARF2_attr, DWARF2_abbrev, + DWARF2_CompUnit, lto_die_ptr, + lto_die_cache_entry, lto_fd, lto_info_fd, + lto_abbrev_fd): Removed. + (lto_file): Removed debug_info and debug_abbrev fields. + (lto_ref): Removed. + (lto_file_init, lto_file_close, + lto_resolve_type_ref, lto_resolve_var_ref, + lto_resolve_fn_ref, lto_resolve_field_ref, + lto_resolve_typedecl_ref, + lto_resolve_namespacedecl_ref, + lto_get_file_name): Removed declarations. + (lto_symtab_merge_var, lto_symtab_merge_fn): + Declarations moved to lto-tree.h. + * lto-symtab.c (lto_compatible_attributes_p): + Lobotomize this, as it barfs on "Hello, world!". + * lto-section-out.c: Include lto-tree-out.h. + (lto_hash_global_slot_node, + lto_eq_global_slot_node, preload_common_nodes, + write_global_stream, write_global_references): + New functions. + (produce_asm_for_decls): Convert for new global streamer. + * lto-section-out.h (lto_hash_global_slot_node, + lto_eq_global_slot_node): Declare. + +2008-06-07 Kenneth Zadeck + Jan Hubicka + + * lto.c (sys/mman.h, tree-pass.h): New includes. + (lto_materialize_constructors_and_inits, + lto_materialize_function): Keeps length of section. + (lto_materialize_cgraph): Removed. + (lto_read_decls): Initialize fd field. + (lto_file_read): Different return type and removed much code to + lto_main. + (page_mask): New variable. + (lto_read_section_data, get_section_data, free_section_data): New + functions. + (lto_main): Now calls pass manager, sets the hooks so that the ipa + passes can get the section data. + +2008-05-27 Kenneth Zadeck + + * lto.h (lto_read_decls): Made local. + (lto_input_function_body, lto_input_constructors_and_inits, + lto_input_cgraph): Declarations moved to lto-section-in.h. + * lto-section-in.c: Moved to .. . + * lto-cgraph-in.c: Ditto. + * lto-section-in.h: Ditto. + * lto-function-in.c: Ditto. + * lto-lang.c (lto_handle_option): Added ATTRIBUTE_UNUSED to parms. + (lto_insert_block): Removed. + (LANG_HOOKS_INSERT_BLOCK): Removed. + * Make-lang.in (lto-cgraph-in.o, lto-function-in.o, + lto-section-in.o): Rules moved to lto/Makefile.in. + + +2008-05-16 Ollie Wild + + * lto-lang.c (tree-inline.h): Include. + (lto_post_options): New function. + (LANG_HOOKS_POST_OPTIONS): Define. + * lto-cgraph-in.c (overwrite_node): Set node->global.insns. + * lto-function-in.c (input_bb): Set TREE_BLOCK (stmt). + +2008-05-13 Diego Novillo + + * lto-function-in.c (input_ssa_names): Call + make_ssa_name_fn instead of make_ssa_name. + +2008-05-12 Diego Novillo + + * lto-cgraph-in.c (overwrite_node): Update references to + inline summary fields. + * lto-function-in.c (input_expr_operand): Do not handle + STRUCT_FIELD_TAG. + +2008-05-09 Ollie Wild + + * lang.opt: New file. + * lto-lang.c (lto_init_options): New function. + (lto_handle_option): New function. + (lto_init): Move initialization of flag_unit_at_a_time to + lto_init_options. + (LANG_HOOKS_INIT_OPTIONS): Define. + (LANG_HOOKS_HANDLE_OPTION): Define. + +2008-04-29 Ollie Wild + + * lto.c (lto_read_namespace_DIE): New function. + (lto_read_DIE): Add lto_read_namespace_DIE callback. Cache + NAMESPACE_DECL DIE's. + (lto_resolve_namespacedecl_ref): New function. + * lto.h (lto_resolve_namespacedecl_ref): New function. + * lto-section-in.c (lto_read_decls): Read namespace declarations. + * lto-section-in.h (struct lto_file_decl_data): Add namespace_decls + and num_namespace_decls. + * lto-function-in.c (input_expr_operand): Add NAMESPACE_DECL case. + * lto-lang.c (lto_init_ts): New function. + (LANG_HOOKS_INIT_TS): Set as lto_init_ts. + +2008-04-16 Ollie Wild + + * lto-function-in.c (input_type_ref): Updated function description. + +2008-04-16 Ollie Wild + + * lto-function-in.c (input_type_ref_1): New function. + (input_type_ref): Split into two functions. + (input_function): Add support for type contexts. + +2008-04-16 Ollie Wild + + * lto.c (lto_materialize_function): Use DECL_ASSEMBLER_NAME to compute + section name + +2008-04-16 Ollie Wild + + * lto.c (lto_read_compile_unit_DIE): Add DW_LANG_C_plus_plus to the + list of supported languages. + +2008-03-25 Kenneth Zadeck + + Merge with mainline @133491. + +2008-03-05 Kenneth Zadeck + Jan Hubicka + + * lto.c (lto_info_fd_init, lto_info_fd_close): Get rid of + fd->unmaterialized_fndecls. + (lto_get_file_name, lto_materialize_cgraph): New function. + (lto_materialize_constructors_and_inits, + lto_materialize_function): Read info directly from elf file. + (lto_file_read): Made local and initialize dictionary so that + other lto sections can be read without reprocessing the elf file. + (lto_main): Read all functions after all files have been processed + for their types, globals and cgraph. + * Make-lang.in (lto.o, lto-cgraph-in.c, lto-section-in): Changed + dependencies. + * lto-elf.c (lto_elf_file): Removed strtab, symtab fields. + (hash_name, eq_name, lto_elf_build_section_table): New functions. + (lto_elf_read_symtab): Removed function. + (lto_elf_file_open): Removed call to lto_elf_read_symtab. + * lto.h (lto_info_fd_struct): Removed unmaterialized_fndecls. + (lto_file_read): Made local. + (lto_get_file_name, lto_elf_build_section_table, + lto_input_cgraph): + New function. + * lto-section-in.c (lto_read_section_data, lto_get_section_data): + New functions. + (lto_read_decls): Get the file name. + * lto-cgraph-in.c: New file. + * lto-function-in.c (tag_to_expr): Stops at LTO_tree_last_tag. + (input_expr_operand, lto_read_body): Set lto_debug_context.tag_names. + (input_labels): Fixed latent sizeof issue. + (input_function): Build stmt array to set call sites into cgraph + edges. + (lto_read_body): Reset cfun->curr_properties. + * lto_section_in.h (lto_section_slot): New structure. + (section_hash_table.lto_file_decl_data): New field. + + +2008-02-09 Kenneth Zadeck + + * lto.c (lto_read_variable_formal_parameter_const): Remove code to + reconstruct static initializers. + (lto_get_body, lto_materialize_function): Add lto_section_type as + a parameter. + (lto_materialize_constructors_and_inits, + lto_materialize_file_data): New function. + (lto_materialize_function, + lto_read_subroutine_type_subprogram_DIE): Renamed unmap_fn_body to + unmap_section and map_fn_body to map_section. + (lto_set_cu_context): Process functions and static inits + differently. + * Make-lang.in (LTO_H, lto/lto-function-in.o, + lto/lto-section-in.o): Update dependencies. + * lto/lto-elf.c (lto_elf_map_optional_lto_section): Add + lto_section_type parameter. + (lto_elf_unmap_fn_body): Renamed to lto_elf_unmap_section. + * lto.h (lto_file_vtable_struct): Removed two of the fields and + renamed the other two so that there is only one map function and + one unmap function and each takes a section type parameter. + (lto_read_function_body): Renamed to lto_input_function_body and + added file_data parameter. + (lto_read_var_init): Removed. + (lto_input_constructors_and_inits): New function. + * lto-section-in.c (lto_read_decls): New function. + * lto-function-in.c (data_in): Moved fields field_decls, fn_decls, + var_decls, type_decls, types to lto_file_decl_data. + (input_type_ref, input_expr_operand, lto_read_body): Get + field_decls, fn_decls, var_decls, type_decls, types from different + structure. + (input_globals, input_constructor, lto_read_var_init): Removed. + (input_constructors_or_inits): New function. + (lto_read_function_body, lto_input_constructors_and_inits): + Renamed to lto_input_function_body and takes file_data parameter. + * lto-section-in.h (lto_file_decl_data): New structure. + +2008-01-28 Kenneth Zadeck + + * lto-function-in.c (input_globals.c): Changed input type to + lto_function_header. + (input_bb): Removed code to deserialize the stmt number. + (input_function): Renumber all stmts after they are input. + (lto_read_body, lto_read_function_body, lto_read_var_init): + Changed to used new header format and enum section_type. + *lto-lang.c (success): Removed. + +2008-01-28 Nathan Froyd + + * lto-elf.c (lto_elf_lookup_sym): Remove unused function. + (lto_elf_free_sym): Likewise. + + * lto-elf.c (lto_elf_read_var_init): Remove unused function. + (lto_elf_build_init): Likewise. + +2008-01-14 Kenneth Zadeck + + * lto-read.c: Renamed to lto-function-in.c. + (input_1_unsigned): Moved to lto-section-in.c and renamed + lto_input_1_unsigned. + (input_uleb128): Moved to lto-section-in.c and renamed + lto_input_uleb128. + (input_widest_uint_uleb128): Moved to lto-section-in.c and renamed + lto_input_widest_uint_uleb128. + (input_sleb128): Moved to lto-section-in.c and renamed + lto_input_sleb128. + (input_integer): Moved to lto-section-in.c and renamed + lto_input_integer. + (debug_in_fun): Moved to lto-section-in.c and renamed + lto_debug_in_fun. + (input_block): Moved to lto-section-in.h and renamed + lto_input_block. + (input_expr_operand): Fixed to allow lists with more than one + element. + * lto-section-in.h: New file. + * lto-section-in.c: New file with changes from above. + * Make-lang.in (lto-read.o): Renamed lto-function-in.c. + (lto-section-in.o): New rule. + +2007-12-29 Nathan Froyd + + * lto-read.c (input_expr_operand): Mark static and external + VAR_DECLs as needed. + +2007-12-29 Nathan Froyd + + * lto-read.c (input_integer): Use the correct shift amount. + +2007-12-29 Nathan Froyd + + * lto-lang.c (lto_pushdecl): Do nothing instead of aborting. + (LANG_HOOKS_NAME): Define. + +2007-12-27 Nathan Froyd + + * lto.c (lto_find_integral_type): Define as a macro. Rename the + original function to... + (lto_find_integral_type_1): ...this. Consult UNSIGNEDP if we + don't have a base type. + (lto_read_enumeration_type_DIE): Examine the values of the + enumeration to determine whether we can use an unsigned type for + the base type of the enumeration. + +2007-12-24 Nathan Froyd + + * lto.c (lto_read_structure_union_class_type_DIE): Set TYPE_MODE + and TYPE_ALIGN on UNION_TYPEs as soon as possible. + +2007-12-22 Nathan Froyd + + * lto-lang.c (lto_types_compatible_p): New function. + (LANG_HOOKS_TYPES_COMPATIBLE_P): Define. + +2007-12-22 Nathan Froyd + Kenneth Zadeck + + * lto-read.c (input_expr_operand): Fixed uninitialize var warning. + (input_local_vars): Read in DECL_INITIAL and context for local + statics that need to be put in unexpanded_vars_list. + +2007-12-21 Nathan Froyd + + * lto-read.c (input_real): Use a separate null-terminated buffer + for calling real_from_string. + (input_expr_operand): If we take the address of a FUNCTION_DECL, + tell cgraph that it's needed. + +2007-12-19 Doug Kwan + + * lto.c (lto_read_base_type_DIE): Handle complex integer types. + +2007-12-18 Nathan Froyd + + * lto.c (lto_read_DIE): Call lto_read_only_for_child_DIEs instead. + (lto_file_read): Reset the members of 'context' every time we read + a toplevel DIE, with special attention to last_param_type. + +2007-12-18 Nathan Froyd + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Initialize + 'declaration'. Set the assembler name for non-public functions. + +2007-12-17 Kenneth Zadeck + + * lto_read.c (data_in.unexpanded_indexes): New array. + (input_local_var): Added code to read in unexpanded_var_list + indexes for variables. Only read in DECL_CHAIN field for + parameters. + (input_local_vars): Added code to rebuild unexpanded_var_list in + order using unexpanded_indexes. + (input_function): Added code to set DECL_CONTEXT for functions. + +2007-12-13 Doug Kwan + + * lto.c (lto_read_pointer_reference_type_DIE): Handle optional name + in pointer and reference types. + +2007-12-13 Nathan Froyd + + * lto-read.c (input_expr_operand): Use DECL_RESULT when reading a + RESULT_DECL. + +2007-12-13 Nathan Froyd + + * lto.c (lto_read_array_type_DIE): Return the cached DIE if we've + already read the DIE. + (lto_get_body): New function, split out from... + (lto_materialize_function): ...here. Call it. + (lto_read_subroutine_type_subprogram_DIE): Call lto_get_body to + determine DECL_EXTERNAL. + * lto-symtab.c (lto_symtab_merge_decl): Merge the DECL_RESULTs of + FUNCTION_DECLs when necessary. Use the type of the actual + function definition if we are unable to easily merge types. Ignore + spurious DECL_MODE mismatches on VAR_DECLs. Merge DECL_MODEs when + necessary. + +2007-12-13 Nathan Froyd + + * lto-lang.c (LANG_HOOKS_REDUCE_BIT_FIELD_OPERATIONS): Define. + +2007-12-12 Bill Maddox + + Revert + 2007-12-07 Bill Maddox + + * lto.c (lto_str_fd_init): New function. + (lto_str_fd_close): New function. + (lto_file_init): Call lto_str_fd_init. + (lto_file_close): Call lto_str_fd_close. + (lto_str_read): New function. Read debug string table. + (lto_str_lookup): New function. Get string for debug + string table offset. + (lto_read_form): Recognize DW_FORM_strp. + (lto_file_read): Invoke lto_str_read. + + * lto-elf.c (lto_elf_file_open): Read raw section data + for the .debug_str section, if present. + + * lto.h (struct lto_str_fd_struct): New struct. + (struct lto_file_struct): Added new field DEBUG_STR + to hold the file descriptor for the debug string table. + +2007-12-07 Bill Maddox + + * lto.c (lto_str_fd_init): New function. + (lto_str_fd_close): New function. + (lto_file_init): Call lto_str_fd_init. + (lto_file_close): Call lto_str_fd_close. + (lto_str_read): New function. Read debug string table. + (lto_str_lookup): New function. Get string for debug + string table offset. + (lto_read_form): Recognize DW_FORM_strp. + (lto_file_read): Invoke lto_str_read. + + * lto-elf.c (lto_elf_file_open): Read raw section data + for the .debug_str section, if present. + + * lto.h (struct lto_str_fd_struct): New struct. + (struct lto_file_struct): Added new field DEBUG_STR + to hold the file descriptor for the debug string table. + +2007-12-07 Nathan Froyd + + * lto-read.c (input_cfg): Call init_empty_tree_cfg_for_function. + Grow the basic_block_info and label_to_block_map vectors if + necessary. Read in the block chain. + +2007-12-06 Nathan Froyd + + * lto.c (lto_read_DIE): Set TYPE_ALIAS_SET where necessary. + +2007-12-06 Nathan Froyd + + * lto.c (lto_read_form): Add DW_cl_address for DW_AT_const_value. + +2007-12-06 Nathan Froyd + + * lto-read.c (input_expr_operand): Don't check for MTAGs. + (lto_read_body): Don't declare PROP_alias. + +2007-12-06 Nathan Froyd + + * lto-symtab.c (lto_symtab_merge_decl): Handle FUNCTION_DECLs without + argument type information. + +2007-12-03 Nathan Froyd + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Set + TREE_THIS_VOLATILE if the associated type is a volatile type. + (lto_materialize_function): Remove call to init_ssa_operands. + * lto-read.c (input_expr_operand): Add SSA_NAME_VAR as a referenced + variable when reading an SSA_NAME. Do the same when reading a + RESULT_DECL, a RETURN_EXPR, or an MTAG. + (input_cfg): Call init_ssa_operands. + (input_ssa_names): Set the default def of an SSA_NAME if necessary. + Move call to init_tree_ssa... + (lto_read_body): ...here. Use push_cfun and pop_cfun. Call + add_referenced_var on any variables referenced from the body of the + function. Inform the rest of the compiler we are in SSA form and + inform later passes about the current properties. + +2007-11-30 Nathan Froyd + + * lto.c (lto_materialize_function): Add FIXME. + +2007-11-29 Nathan Froyd + + * lto-lang.c (enum built_in_attribute): New enum. + (flag_no_builtin, flag_no_nonansi_builtin, flag_isoc94, flag_isoc99, + built_in_attributes): New variables. + (def_builtin_1): New function. + (lto_define_builtins): #define DEF_BUILTIN and include builtins.def. + +2007-11-28 Nathan Froyd + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Set + DECL_SOURCE_LOCATION for debugging purposes. + (lto_read_member_DIE): Set DECL_SOURCE_LOCATION. If we have read a + bitfield, use the type specified by the DIE for TREE_TYPE and defer + laying out the decl until later. + (lto_read_subroutine_type_subprogram_DIE): Compare the function's + name with DECL_ASSEMBLER_NAME. Set DECL_SOURCE_LOCATION and + TREE_ADDRESSABLE. + * lto-read.c (input_expr_operand): Set TREE_ADDRESSABLE on the + operand of an ADDR_EXPR. + * lto-lang.c (enum lto_builtin_type): New enum. + (builtin_type): New typedef. + (builtin_types, string_type_node, const_string_type_node, + wint_type_node, intmax_type_node, uintmax_type_node, + signed_size_type_node): New variables. + (def_fn_type, builtin_type_for_size, lto_define_builtins, + lto_build_c_type_nodes): New functions. + (lto_init): Initialize builtin types. + (lto_set_decl_assembler_name): Let the target machine mangle the + name if the decl is TREE_PUBLIC, otherwise uniquify it. + +2007-11-21 Nathan Froyd + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Don't + set TREE_ADDRESSABLE. Do set DECL_COMDAT. Set TREE_READONLY if + the type is a constant type. Set the assembler name and inform + the rest of the compiler about the new decl if the decl is not + public. + (lto_read_subroutine_type_subprogram_DIE): Don't check for equivalency + of DECL_ASSEMBLER_NAME when determining if we have a builtin. Don't + try to read in function bodies for functions that already have bodies. + * lto-symtab.c (lto_same_type_p): Check for unbounded array + equivalency. + (lto_symtab_merge_decl): Don't merge decls that aren't TREE_PUBLIC. + Check for whether we matched a builtin function type before calling + lto_same_type_p on the generated type. Permit cases where the + declaration of an array is unbounded, but the definition is bounded. + Don't combine TREE_PUBLIC flags. Copy over DECL_SIZE and + DECL_SIZE_UNIT if necessary. + +2007-11-16 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Get types right + for COMPLEX_CST. + +2007-11-16 Kenneth Zadeck + + * lto-read.c (make_new_block, input_cfg): Properly set + n_basic_blocks. + +2007-11-16 Nathan Froyd + + * lto.c (lto_read_array_type_DIE): Handle DIEs with DW_AT_GNU_vector + set properly by building a VECTOR_TYPE instead of an ARRAY_TYPE. + +2007-11-16 Nathan Froyd + + * lto.c (lto_read_base_type_DIE): Use make_bitfield_integer_type to + construct the integer type for bitfields. + +2007-11-16 Kenneth Zadeck + + * lto-read.c (data_in.current_node_has_loc): Removed. + (input_line_info): Returns true if node needs line set. + (set_line_info): Always sets line if called. + (clear_line_info): Removed reference to current_node_needs_loc. + (input_expr_operand): Keeps track locally if current node needs a loc. + (input_local_var): Added code to handle DECL_INITIAL for + static local vars. Only set loc if necessary. + +2007-11-15 Nathan Froyd + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Fix thinko'd + DECL_CONTEXT. + +2007-11-15 Nathan Froyd + + * lto.c: Include langhooks.h. + (lto_find_integral_type): Rework logic to handle the case where + got_byte_size is true, but the bitsize requested and that of the + base_type doesn't match. + (lto_read_variable_formal_parameter_constant_DIE): Only check for + asm_name if we are creating a VAR_DECL. + (lto_materialize_function): Set DECL_EXTERNAL if we can't find a + definition. + (lto_read_subroutine_type_subprogram_DIE): Check for a builtin + function reference and use the builtin's decl if so. Set the + DECL_CONTEXT of the RESULT_DECL for the function. + * lto-lang.c (registered_builtin_fndecls): New variable. + (lto_getdecls): Return it. + (lto_builtin_function): Chain the new decl onto + registered_builtin_fndecls. + +2007-11-15 Kenneth Zadeck + + * lto-read.c (process_tree_flags, lto_static_init_local): + Renamed to ADD_CLASS_EXPR_FLAG. ADD_CLASS_DECL_FLAG New Macro. + (input_line_info, clear_line_info): Fixed new line number code. + (input_expr_operand): Added type to SWITCH_EXPR. + (lto_read_body): Properly initialized data_in. + Clear line info when leaving. + +2007-11-13 Diego Novillo + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): + Initialize ARTIFICIAL. + (lto_read_subroutine_type_subprogram_DIE): Initialize + SAVED_SCOPE. + * lto-read.c (set_line_info): Remove ; from calls to + LINEMAP_POSITION_FOR_COLUMN. + +2007-11-13 Kenneth Zadeck + + * lto-read.c (input_type_ref): Renamed from get_type_ref. + (input_expr_operand, input_local_var): Renamed get_type_ref to + input_type_ref. + (input_expr_operand): Get the types correct for + vector-cst. Get SSA_NAME_DEF_STMT correct for return_exprs. + +2007-11-13 Doug Kwan + + * lto-read.c (input_widest_uint_uleb128): New function. + (input_tree_flags, process_tree_flags, input_line_info, + input_expr_operand, input_local_var, input_phi, input_ssa_names): + Change to use lto_flags_type and BITS_PER_LTO_FLAGS_TYPES instead of + unsigned HOST_WIDE_INT and HOST_BITS_PER_WIDE_INT. + (lto_static_init_local): Add code to assert that lto_flags_type is + wide enough. + +2007-11-13 Nathan Froyd + + * lto.c (lto_read_array_type_DIE): Handle DW_AT_GNU_vector. + (lto_read_subroutine_type_subprogram_DIE): Handle + DW_AT_static_link and DW_AT_specification. Return the + specification if present. + (lto_read_base_type_DIE): Handle DW_ATE_complex_float. + +2007-11-13 Nathan Froyd + + * lto-lang.c: Include target.h. + (registered_builtin_types): New variable. + (lto_type_for_mode): Increase number of modes handled. + (lto_builtin_function): Fix argument list and return the decl. + (lto_register_builtin_type): New function. + (lto_init): Initialize target builtins and language-independent + nodes. + (LANG_HOOKS_REGISTER_BUILTIN_TYPE): Define. + +2007-11-13 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Added code to properly handle + index filed. Added new RANGE_EXPR case. + +2007-11-11 Kenneth Zadeck + + * lto-read.c (ADD_FUNC_FLAG): Deleted macro. + (data_in): Added current_node_has_loc field. + (input_line_info, set_line_info, clear_line_info): Added a support + for USE_MAPPED_LOCATION and not adding line numbers to nodes that + did not have on on the source side. + (input_expr_operand): Make sure that GIMPLE_MODIFY_STMTS get line + numbers too. + +2007-11-09 Doug Kwan + + * lto-read.c (input_expr_operand): Change type of operand 2 of + BIT_FIELD_REF expression to be bitsizetype instead of sizetype. + +2007-11-09 Nathan Froyd + + * lto.c: Include lto-tree.h. Effect small spaces->tabs cleanup. + (lto_read_variable_formal_parameter_constant_DIE): Transfer bits + from a DW_AT_specification or DW_AT_abstract_origin attribute to + the new decl we are creating. Move informing the middle end about + the new decl to... + (lto_main): ...here. Inform the middle end about global variables + after we have read in all the input files. + * lto-symtab.c (lto_symtab_merge_decl): We really do need to merge + variables with internal linkage, so delete the check for internal + linkage. Combine TREE_PUBLIC flags. + +2007-11-08 Nathan Froyd + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Handle + DW_AT_decl_line. + * lto-symtab.c (lto_symtab_merge_decl): Handle redefinition of a + builtin specially. Move check for attribute compatibility + earlier. + +2007-11-07 Nathan Froyd + + * Make-lang.in (lto/lto.o): Depend on gt-lto-lto.h. + * config-lang.in (gtfiles): Add lto.h and lto.c. + * lto-elf.c: Include ggc.h. + (lto_elf_file_open): Allocate elf_file from GC memory. + * lto.c: Include tree-ssa-operands.h and gt-lto-lto.h + (lto_info_fd_init): Allocate the die_cache and unmaterialized_fndecls + in GC memory. + (lto_info_fd_close): Free unmaterialized_fndecls from GC memory. + (lto_file_close): Free file from GC memory. + (lto_cache_store_DIE): Allocate the new entry in GC memory. + (lto_read_member_DIE): Fix declaration. + (lto_read_subroutine_type_subprogram_DIE): unmaterialized_fndecls lives + in GC memory. + (current_lto_file): New variable. + (lto_main): Use it. + (DWARF2_attr, DWARF2_abbrev, lto_die_ptr, DWARF2_CompUnit, + lto_die_cache_entry): Move to... + * lto.h: ...here and add GTY markers as appropriate. Delete forward + declarations accordingly. + (struct lto_file_struct): Declare. + (lto_file_vtable): Use it instead of lto_file. + +2007-11-06 Alon Dayan + Kenneth Zadeck + + * lto-read.c (process_flags, lto_static_init_local): + read flags of VAR_DECL and FUNCTION_DECL of size>1. + change global array num_flags_for_code to flags_length_for_code. + (set_line_info): Make decls work in USE_MAPPED_LOCATION mode. + +2007-11-05 Nathan Froyd + + * lto.c (lto_read_structure_union_class_type_DIE): Use proper record + layout functions to compute information about the newly constructed + type. + +2007-11-02 Nathan Froyd + + * lto-read.c (input_expr_operand): Change the LTO_return_expr1 + case to use DECL_RESULT if necessary. + +2007-11-01 Kenneth Zadeck + + * lto-read.c (input_tree_list): Removed. + (input_tree_flags): Added parameter to force flags no matter what + tree code. + (input_expr_operand): Added parameter to input_tree_flags. + Added case for IDENTIFIER_NODE and TREE_LIST. Changed ASM to call + input_expr_operand rather than input_tree_lists. + (input_local_var): Use input_expr_operand to read attributes + rather then input_tree_list. + (input_phi, input_ssa_names): Added parameter to input_tree_flags. + +2007-10-31 Nathan Froyd + + * lto.c (lto_read_typedef_DIE): Fix comment typo. + (lto_resolve_typedecl_ref): Fetch the referred-to type and build a fake + TYPE_DECL for it. + * lto-read.c (lto_read_body): Use correct sizes for calculating + type_decls_offset and types_offset. + +2007-10-30 Nathan Froyd + + * lto-tree.h (union lang_tree_node): Change GTY description to chain + with GENERIC_NEXT. + * config-lang.in (gtfiles): Add lto-lang.c. + * lto-lang.c: Include gt-lto-lto-lang.h. + * Make-lang.in (lto/lto-lang.o): Add dependency on gt-lto-lto-lang.h + (lto/lto-symtab.o): Depend on LTO_H instead of TREE_H. + (lto/lto-read.o): Likewise. + +2007-10-29 Kenneth Zadeck + + * lto-read.c (data_in): Added type_decls and current_col fields. + (string_slot): New type to hold canonized file name. + (hash_string_slot_node, eq_string_slot_node, canon_file_name, + input_line_info, set_line_info, clear_line_info): New functions. + (file_name_hash_table): New hash table. + (input_local_var, input_labels, input_local_vars_index, + input_local_var, input_local_vars, input_ssa_names): Reorganized parameters. + (input_uleb128): Changed type of byte var. + (input_expr_operand): Large number of changes to get line numbers + correct. Added TYPE_DECL case. + (input_globals): Added code to get TYPE_DECLs processed. + (input_local_var): Added code to process line numbers and + TREE_CHAIN and DECL_CONTEXT. + (input_function, input_constructor): Added call to + clear_line_number. + (lto_static_init_local): Added code to get line numbers correct. + (lto_read_body): Added code to get TYPE_DECLS read and to change + parameters to the calls above that had their parms reorganized. + + +2007-10-29 Nathan Froyd + + * lto.h (lto_resolve_typedecl_ref): Declare. + * lto.c (lto_resolve_typedecl_ref): New function. + +2007-10-29 Mark Mitchell + Nathan Froyd + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Read the child + DIEs even if we find an abstract origin for this DIE. + +2007-10-29 Nathan Froyd + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Build the + RESULT_DECL slightly earlier. Only remember the decl for later + if we successfully merge declarations. + +2007-10-24 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Give label_values the proper + context and provide switch statements with a default type. + +2007-10-23 Nathan Froyd + + * lto-read.c (lto_read_body): Move call to init_ssa_operands... + * lto.c (lto_materialize_function): ...to here. + +2007-10-22 Nathan Froyd + + * lto.h (struct lto_info_fd): Add field unmaterialized_fndecls. + * lto.c (lto_info_fd_init): Initialize it. + (lto_info_fd_close): Free it. + (lto_materialize_function): New function. + (lto_read_subroutine_type_subprogram_DIE): Save the result decl on + unmaterialized_fndecls. + (lto_file_read): Read in all the function bodies after we have read + all of the DWARF info. + * lto-read.c (lto_read_body): Call init_ssa_operands if we are + reading a function body. + +2007-10-20 Kenneth Zadeck + + * lto-read.c (input_tree_flags): Renamed from input_flags to be + semetric with output_tree_flags. Added call to log flags. + (process_tree_flags): Renamed from process_flags. Fixed a lot of + type issues to make everything consistent with flags being + unsigned HOST_WIDE_INTS. + (input_expr_operand): Added call to + recompute_tree_invariant_for_addr_expr. + (input_local_var): Added debugging for tree_chains. Now calls + input_tree_flags. + (input_phi): Made flags unsigned HOST_WIDE_INT. + (input_ssa_names): Now calls input_tree_flags. + (lto_read_body): Now sets cfun. + (lto_read_function_body): Now sets current_function_pointer. + +2007-10-19 Nathan Froyd + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Check + definitively whether SPECIFICATION or ABSTRACT_ORIGIN exist before + inspecting fields within. + (lto_read_DIE_at_ptr): Delete check for null result; let callers + handle this appropriately. + +2007-10-19 Nathan Froyd + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Handle + DW_AT_abstract_origin properly. Ensure that we're not dealing with + both DW_AT_abstract_origin and DW_AT_specification. + (lto_read_subroutine_type_subprogram_DIE): Handle + DW_AT_abstract_origin. + (lto_read_DIE): Use lto_read_only_for_child_DIEs for labels. + (lto_read_DIE_at_ptr): Define as static to match declaration. + Lookup the PTR in the cache before reading it from the file. + (lto_resolve_var_ref): Adjust accordingly. + (lto_resolve_fn_ref): Adjust accordingly. Tweak comment. + (lto_resolve_field_ref): Adjust accordingly. Tweak comment. + +2007-10-19 Nathan Froyd + + * lto.c (lto_read_DIE_at_ptr): New function. + (lto_resolve_var_ref): Use it. + (lto_resolve_fn_ref): Use it. + (lto_resolve_field_ref): Use it. + (lto_read_variable_formal_parameter_constant_DIE): Follow + DW_AT_specification and return the associated decl when appropriate. + +2007-10-18 Nathan Froyd + + * lto-lang.c (lto_type_for_mode): Move after lto_type_for_size. + Implement for scalar integer modes. + (lto_init): Initialize size_type_node. + +2007-10-18 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Remove ssa name asserts. + (input_local_var): Add chaining for params. + (input_ssa_names): Add cfun parameter. + (input_function): Remove unnecessary else. + +2007-10-17 Nathan Froyd + + * lto.c (lto_read_only_for_child_DIEs): Mark die parameter as unused. + (lto_resolve_var_ref): Use proper types. + (lto_resolve_fn_ref): Likewise. + (lto_resolve_field_ref): Likewise. + +2007-10-17 Nathan Froyd + + * lto-read.c (input_expr_operand): Remove case. + +2007-10-17 Nathan Froyd + + * lto.c (lto_read_only_for_child_DIEs): New function. + (lto_read_DIE): Use it for lexical_block and inlined_subroutine DIEs. + * lto-elf.c (lto_elf_map_lto_section): Remove. + (lto_elf_file_vtable): Use lto_elf_map_optional_lto_section instead. + * lto-read.c (input_expr_operand): Assert that we never read a NULL + SSA_NAME. Add missing case for mechanical codes. + (input_cfg): Use basic_block_info_for_function instead of + basic_block_info. + +2007-10-16 Kenneth Zadeck + + * lto-read.c (input_sleb128, input_integer): Use proper casts. + (input_list): Renamed input_tree_list and modified to follow same + protocol as lto-function-out.c:output_tree_list. + (input_expr_operand): Make asm operands use input_tree_list. + (input_local_var): Now uses input_tree_list. + (lto_read_body): Change placement for setting context of debug_labels. + + +2007-10-16 Kenneth Zadeck + + * lto-read.c (input_real): Output debugging in proper order. + (input_integer): Compute bit lengths properly. + (input_list): Clean up declaration. + (input_expr_operand): Change calls to input_real to match fix. + Make reading of LTO_bit_field_ref1 match output. + (input_local_var): Make reading of attributes match what is being + written. + (dump_debug_stream): Also print char in hex. + (debug_out_fun): Fix signed unsigned mismatch. + +2007-10-10 Nathan Froyd + + * lto.c (lto_read_form): Handle DW_AT_MIPS_linkage_name and + DW_AT_GNU_vector specially, as they are not contiguous with the + specified set of attribute names. Use class_mask to check for + errors at the end of the function + (lto_resolve_var_ref): Read the DIE if it is not cached. + (lto_resolve_fn_ref): Likewise. + (lto_resolve_field_ref): Likewise. + +2007-10-05 Nathan Froyd + + * lto.c: Include dwarf2out.h. + (lto_cache_store_DIE): Assert that we never change the value. + (LTO_BEGIN_READ_ATTRS): Print an informative error message. + (lto_read_compile_unit_DIE): Handle DW_AT_entry_pc. + (lto_read_array_type_DIE): Don't error on ndims == 0; build a + sensible type instead. + (lto_read_structure_union_class_type_DIE): Store the newly + created type prior to reading the members of the structure to + avoid infinite recursion. Avoid computing types and alignments + for structures whose sizes are unknown. + (lto_read_variable_formal_parameter_const): Handle DW_AT_artificial + and set DECL_ARTIFICIAL accordingly. Ignore DW_AT_abstract_origin, + DW_AT_const_value, and DW_AT_specification. + (lto_read_subroutine_type_subprogram_DIE): Handle DW_AT_declaration. + Return early if we have already constructed the function type. + (lto_read_typedef_DIE): Check to see if the type has been cached + already. Cache the type before reading any children. + (lto_read_const_volatile_restrict_type_DIE): Handle DW_AT_name. + (lto_read_DIE): Unset context->skip_non_parameters around reading + the DIE. + (lto_resolve_fn_ref): Delete trailing whitespace. + +2007-09-11 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Added type for STRING_CST. + +2007-09-10 Nathan Froyd + + * lto-read.c (lto_read): Set the type of the newly created CALL_EXPR. + +2007-09-07 Nathan Froyd + + * lto-lang.c (signed_and_unsigned_types): New variable. + (lto_type_for_size): Consult signed_and_unsigned_types to find + an approprite type, creating it if necessary. + (lto_set_decl_assembler_name): Add actual method body. + +2007-09-06 Jim Blandy + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): If we + can't find a var init for this variable, leave its DECL_INITIAL. + * lto-elf.c (lto_elf_map_optional_lto_section): Renamed from + lto_elf_map_fn_body. + (lto_map_lto_section): New function. + (lto_elf_file_vtable): Use lto_elf_map_lto_section for function + bodies, and lto_elf_map_optional_lto_section for variable + initializers. + (lto_elf_find_section_data): Quietly return NULL if the section is + missing. + (lto_elf_file_open): Check for a NULL from lto_elf_find_section_data. + + * lto-elf.c (lto_elf_find_section_data): Remove dead code. + + * lto-read.c (lto_read_body): Doc fix. + +2007-08-29 Kenneth Zadeck + + * lto-read.c (fun_in): Renamed to data_in. + (input_expr_operand, input_local_var, input_string_internal, + input_string, input_real, input_list, get_label_decl, + get_type_ref, input_expr_operand, input_globals, input_labels, + input_local_vars_index, input_local_var, input_local_vars, + input_cfg, input_phi, input_ssa_names, input_bb, ): Renamed fun_in to data_in. + (input_constructor): New function. + (lto_read_function_body): Renamed to lto_read_body and generalized + to handle both functions and constructors. + (lto_read_function_body, lto_read_var_init): New function. + + +2007-08-28 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Assert that there really is a + FUNCTION_DECL. + (input_globals): Removed checks on 0 section. + +2007-08-28 Kenneth Zadeck + + * lto-read.c (fun_in): Added local_decls_index and + local_decls_index_d. + (input_expr_operand): Changed inputting of PARM_DECLs and VAR_DECLs. + (input_globals): Enabled code to handle FIELD_DECLs. + (input_local_vars_index, input_local_vars): New function. + (input_local_var): Changed to allow locals to be input randomly. + (lto_read_function_body): Added code to input the + local_decls_index and to free various structures. + +2007-08-17 Jim Blandy + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Remove + ATTRIBUTE_UNUSED from 'die' formal. + + Use enum LTO_tags where appropriate, instead of 'unsigned int'. + * lto-read.c (input_record_start): Fix return type, type of 'tag'. + (input_list): Fix type of 'tag'. + (input_expr_operand): Fix type of 'tag' argument. Update + declaration. Fix type of 'ctag'. Add default case to switch, + since the type of the switched value is now an enum. + (input_local_vars): Fix type of 'tag'. + (input_bb): Fix type of 'tag' argument. + (input_function): Fix type of 'tag' argument. + +2007-08-16 Jim Blandy + + * lto.c (lto_read_member_DIE): Record the tree we create in + fd->die_cache. (Our 'die' argument is no longer unused.) + (lto_resolve_field_ref): New function. + * lto.h (lto_resolve_field_ref): New declaration. + +2007-08-15 Jim Blandy + + * lto-read.c (lto_read_var_init): Mark arguments as unused. + +2007-08-07 Jim Blandy + + * lto.c (lto_read_form): Complete attr_classes table. + (DWARF2_form_data): Doc fix. + +2007-08-05 Mark Mitchell + + * lto.h (lto_file_vtable): Remove read_var_init. Add map_var_init + and unmap_var_init. + (lto_read_var_init): Declare. + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Use new + interface for reading variable initializers. + * lto-elf.c (lto_elf_read_var_init): Remove. + (lto_elf_file_vtable): Update initializer. + (lto_elf_read_var_init): Add comment about unused-ness. + * lto-read.c (lto_read_var_init): New. + + * lto.c (lto_read_form): Add entry for DW_AT_inline. + +2007-08-02 Kenneth Zadeck + + * lto-read.c (lto_read_function_body): Moved declaration of fn + outside of ifdef. + +2007-08-01 Kenneth Zadeck + + * lto-read.c (input_uleb128, input_string_internal, input_real, + input_integer, input_record_start, input_list, get_type_ref, + input_flags, input_expr_operand, input_expr_operand, + input_expr_operand, input_local_vars, input_cfg, input_phi, + input_ssa_names, input_bb, input_function): Added semicolons. + + +2007-07-31 Kenneth Zadeck + + * lto-read.c (input_globals): Remove debugging. + (input_function): Set DECL_ARGUMENTS. + + +2007-07-31 Kenneth Zadeck + + * lto-read.c (input_expr_operand): Fixed code for COND_EXEC, + RETURN_EXPR, MODIFY_EXPR and processing of flags. + (input_phi): Made work with operands other than SSA_NAMES and + fixed processing of flags. + (input_ssa_names): Initialize SSA_NAME_DEF_STMT to empty stmt. + (input_flags): New function. + * lto-lang.c (lto_init): Changed state of in_lto_p. + + +2007-07-24 Mark Mitchell + + * lto-tree.h (lto_varargs_cookie): Remove. + * lto.c (lto_context): Add last_parm_type, varargs_p, skip_all, + skip_non_parameters, skip_parameters. + (lto_varargs_cookie): Remove. + (lto_read_variable_formal_parameter_constant_DIE): Keep track of + parameter types. + (lto_read_abbrev): New function. + (lto_read_subroutine_type_subprogram_DIE): Make two passes over + child DIEs. + (lto_read_unspecified_parameters_DIE): Set context->varargs_p. + (lto_read_DIE): Use lto_read_abbrev. Honor skip_* flags. + (lto_file_read): Initialize new context fields. + * lto-lang.c (lto_type_for_mode): Return NULL_TREE. + (lto_unsigned_type): Remove. + (lto_signed_type): Likewise. + (lto_signed_or_unsigned_type): Likewise. + (lto_init): Do not create lto_varargs_cookie. + (LANG_HOOKS_UNSIGNED_TYPE): Do not define. + (LANG_HOOKS_SIGNED_TYPE): Likewise. + (LANG_HOOKS_SIGNED_OR_UNSIGNED_TYPE): Likewise. + +2007-07-19 Jan Hubicka + + * lto-read.c (lto_read_function_body): Produce empty scope block + to avoid crash. + +2007-07-18 Mark Mitchell + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Do not + process local variables. + (lto_read_subroutine_type_subprogram_DIE): Read child DIEs. + +2007-07-13 Kenneth Zadeck + + * lto-read.c (input_list, input_expr_operand): Added struct + function parameter. + (init_cfg, finalize_cfg): Removed function. + (input_expr_operand): Added SSA_NAME and GIMPLE_MODIFY_STMT cases. + (input_labels, input_local_vars): Now takes input_block parameter rather than + synthsyzing it. + (input_cfg, input_phi, input_ssa_names): New functions. + (input_bb): Now passes in input_blocks. Does not construct cfg + and processes the list of phi functions. + (input_function): Now builds both the cfg and ssa_names table. + (lto_read_function_body): Processes new header fields to construct + streams for the ssa_names and cfg and their debugging. + * lto/lto-lang.c (lto_init): Set in_lto_p. + + +2007-06-28 Mark Mitchell + + * lto.h (lto_file_vtable): Add read_var_init. + * lto.c (lto_read_variable_formal_parameter_constant_DIE): Read + initializers. + (lto_main): Remove bogus asserts. + * lto-elf.c (tm.h): Include it. + (libiberty.y): Likewise. + (lto_elf_file): Add strtab and symtab. Rename + string_table_section_index to sec_strtab. + (lto_elf_file_vtable): Add lto_elf_read_var_init. + (lto_elf_get_shdr): New function. + (lto_elf_free_shdr): Likewise. + (lto_elf_find_section_data): Use them. + (lto_elf_read_symtab): New function. + (lto_elf_lookup_sym): Likewise. + (lto_elf_free_sym): Likewise. + (lto_elf_file_open): Tidy. Call lto_elf_read_symtab. + (lto_elf_built_init): New function. + (lto_elf_read_var_init): Likewise. + * Make-lang.in (lto/lto-elf.o): Depend on $(TM_H). + +2007-06-26 Kenneth Zadeck + + * lto-read (make_new_block): Initialize the stmt_list. + (lto_static_init_local): Add debugging for missing codes. + +2007-06-26 Mark Mitchell + + * lto.c (lto_read_subroutine_type_subprogram_DIE): Handle + unprototyped functions. + +2007-06-23 Mark Mitchell + + * lto.c (lto_read_variable_formal_parameter_constant_DIE): + Handle DW_AT_MIPS_linkage_name. + (lto_read_subroutine_type_subprogram): Likewise. Correct + compilation errors. + (lto_main): Remove incorrect assertions. + * lto-symbtab.c: Build function types out of TREE_LISTs. + + * lto-elf.c (): Check for HAVE_LIBELF_H. + + * Make-lang.in (LTO_OBJS): Depend on attribs.o. + +2007-06-21 Kenneth Zadeck + + * lto/lto-tree.h (lang_decl, lang_type, language_function): Added + dummy since ggc does not like empty structs. + * lto/lto-elf.c (libelf.h): Changed to libelf/libelf.h. + * lto/lto-read.c (ADD_CLASS_FLAG, ADD_EXPR_FLAG): Changed + expr->common to expr->base. + (make_new_block): Moved stmt_list to proper place. + + +2007-03-14 Robert Kennedy + + Eliminate use of lang_hooks.set_decl_assembler_name from LTO + * lto.c (lto_read_subroutine_type_subprogram_DIE) Get DECL + assembler name from DWARF. + * lto-lang.c (lto_set_decl_assembler_name) New function. + +2006-09-10 Mark Mitchell + + * lto.h (lto_file_vtable): New structure. + (lto_file): Add vtable pointer. + (lto_file_init): Add vtable paramter. + (lto_read_function_body): New function. + (lto_symtab_merge_fn): New function. + * lto.c (lto_file_init): Add vtable parameter. + (lto_read_form): Fill in entries for DW_AT_prototyped, + DW_AT_frame_base. + (lto_read_subroutine_type_subprogram_DIE): New function. + (lto_read_DIE): Fill in entries for DW_TAG_subroutine_type and + DW_TAG_subprogram. + * lto-elf.c (lto_elf_vile_vtable): New variable. + (lto_elf_file_open): Pass it to lto_file_init. + (lto_elf_map_fn_body): New function. + (lto_elf_unmap_fn_body): Likewise. + * lto-read.c: New file. + * lto-symtab.c (lto_symtab_merge_fn): New function. + * lto-lang.c (LANG_HOOKS_CALLGRAPH_EXPAND_FUNCTION): Define to + tree_rest_of_compilation. + * Make-lang.in (LTO_OBJS): Add lto-read.o + (lto-read.o): New target. + +2006-09-03 Mark Mitchell + + * lto.c (): Don't include it. + (lto_context): Don't typedef it. + (lto_resolve_reference): New function. + (lto_read_form): Use it. + (lto_resolve_type_ref): New function. + (lto_resolve_var_ref): Likewise. + (lto_resolve_fn_ref): Likewise. + * lto.h (): Include it. + (lto_context): New type. + (lto_ref): New structure. + (lto_resolve_type_ref): Declare. + (lto_resolve_var_ref): Likewise. + (lto_resolve_fn_ref): Likewise. + +2006-08-18 Mark Mitchell + + * lang-specs.h: New file. + +2006-08-14 Mark Mitchell + + * lto.c (lto_info_fd_init): Allocate the DIE cache. + (lto_info_fd_close): Deallocate it. + (lto_die_cache_entry): New structure. + (lto_cache_hash): New function. + (lto_cache_eq): Likewise. + (lto_cache_store_DIE): Likewise. + (lto_cache_lookup_DIE): Likewise. + (lto_read_referenced_type_DIE): Use the cache. + (lto_read_pointer_type_DIE): Robustify. + (lto_read_DIE): Use the cache. + * lto.h (hashtab.h): Include. + (lto_info_fd): Add DIE cache. + * Make-lang.in (LTO_H): New variable. + (lto/lto-lang.o): Use LTO_H. + (lto/lto-elf.o): Likewise. + (lto/lto-symtab.o): Likewise. + +2006-07-09 Mark Mitchell + + * lto.c (lto_abi_mismatch_error): New function. + (lto_abbrev_read): Initialize num_abbrevs. + (lto_read_form): Specify allowed form classes for + DW_AT_declaration. Adjust for change to lto_set_cu_context. + (lto_read_variable_formal_parameter_constant_DIE): Handle + DW_AT_declaration. Call lto_symtab_merge_var. + (lto_read_pointer_type_DIE): New function. + (lto_read_base_type_DIE): Use build_nonstandard_integer_type. Do + not creat TYPE_DECLs for types that already have them. + (lto_read_DIE): Add lto_read_pointer_type_DIE. + (lto_set_cu_context): Make cu_start point to the header, not the + first DIE. + (lto_file_read): Adjust for change to lto_set_cu_context. + * Make-lang.in (LTO_OBJS): Add lto-symtab.o. + (lto/lto-symtab.o): New rule. + * lto-tree.h (lang_identifier): Add decl field. + (LANG_IDENTIFIER_CAST): New macro. + (LTO_IDENTIFIER_DECL): Likewise. + (lto_symtab_merge_var): Declare. + * lto-symtab.c: New file. + +2006-07-02 Daniel Berlin + + * lto.c (lto_context): Add current_cu and info_fd members. + (DWARF2_CompUnit): New structure. + (lto_read_DIE): Take lto_info_fd *. + (lto_read_child_DIEs): Ditto. + (lto_file_corrupt_error): Constify argument. + (lto_set_cu_context): New function + (lto_info_fd_init): Ditto. + (lto_info_fd_close): Ditto. + (lto_file_init): Use lto_info_fd_init. + (lto_file_close): Use lto_info_fd_close. + (lto_read_initial_length): Pass in pointer to header size. + (lto_read_comp_unit_header): Correct cu_length to + real length from beginning of header. Take lto_info_fd * as + argument. + (find_cu_for_offset): New function. + (lto_read_form): Change first argument to lto_info_fd *. + Add FORM_CONTEXT argument. + Handle DW_FORM_ref_addr. + (lto_read_tag_DIE): Change first argument to lto_info_fd *. + (LTO_BEGIN_READ_ATTRS_UNCHECKED): Save old context. + Swap contexts if necessary for form. + (LTO_BEGIN_READ_ATTRS): Cast fd to right type for + lto_file_corrupt_error. + (LTO_END_READ_ATTRS): Swap contexts back if it had changed. + (lto_read_referenced_type_DIE): Change first argument to + lto_info_fd *. Access lto_fd fields through base pointer. + (lto_read_compile_unit_DIE): Change first argument to an + lto_info_fd *. + (lto_read_variable_formal_parameter_constant_DIE): Ditto. + (lto_read_base_type_DIE): Ditto. + (lto_read_child_DIEs): Ditto. + (lto_read_DIE): Ditto. Change type of function pointer. + (lto_info_read): New function. + (lto_set_cu_context): Ditto. + (lto_file_read): Use lto_info_read, walk resulting CU's + (lto_main): Update for lto_info_fd change. + * lto-elf.c (lto_elf_file_open): Cast lto_info_fd to lto_fd where + necessary. + * lto.h (DWARF2_CompUnit): New structure. + (lto_info_fd): Ditto. + (lto_file): Change debug_info to be an lto_info_fd. + +2006-06-25 Mark Mitchell + + * lto.c (toplev.h): Include it. + (dwarf2.h): Likewise. + (tree.h): Likewise. + (tm.h): Likewise. + (cgraph.h): Likewise. + (ggc.h): Likewise. + (inttypes.h): Likewise. + (DWARF2_attr): New type. + (DWARF2_abbrev): Likewise. + (DWARF2_class): Likewise. + (DWARF2_form_data): Likewise. + (lto_context): Likewise. + (lto_fd_init): New function. + (lto_abbrev_fd_init): Likewise. + (lto_abbrev_fd_close): Likewise. + (lto_file_init): Use them. + (lto_file_close): New function. + (lto_file_corrupt_error): Likewise. + (LTO_CHECK_INT_VAL): New macro. + (lto_check_size_t_val): New function. + (lto_check_int_val): Likewise. + (LTO_READ_TYPE): New macro. + (lto_read_ubyte): New function. + (lto_read_uhalf): Likewise. + (lto_read_uword): Likewise. + (lto_read_uleb128): Likewise. + (lto_read_initial_length): Likewise. + (lto_abbrev_read_attrs): Likewise. + (lto_abbrev_read): Likewise. + (lto_abbrev_lookup): Likewise. + (lto_read_section_offset): Likewise. + (lto_read_comp_unit_header): Likewise. + (lto_read_form): Likewise. + (LTO_BEGIN_READ_ATTRS_UNCHECKED): New macro. + (LTO_BEGIN_READ_ATTRS): Likewise. + (LTO_END_READ_ATTRS): Likewise. + (lto_unsupported_attr_error): New function. + (lto_get_identifier): Likewise. + (lto_read_referenced_type_DIE): Likewise. + (lto_read_compile_unit_DIE): Likewise. + (lto_read_variable_formal_parameter_constant_DIE): Likewise. + (lto_read_base_type_DIE): Likewise. + (lto_read_DIE): Likewise. + (lto_read_child_DIEs): Likewise. + (lto_file_read): Read DIEs. + (lto_main): Ask middle end to emit entities. + * lto-tree.h (lang_identifier): Inherit from tree_identifier. + * lto-elf.c (lto_elf_file_open): Adjust for interface changes. + (lto_elf_file_close): Likewise. + * lto.h (lto_file): Declare. + (DWARF2_abbrev): Likewise. + (lto_fd): New type. + (lto_abbrev_fd): Likewise. + (lto_file): Use new types. + (lto_file_close): Declare. + * lto-lang.c (lto_init): Always use unit-at-a-time mode. + +2006-06-18 Mark Mitchell + + * lto.h: New file. + * lto.c: New file. + * lto-elf.c: New file. + * lto-lang.c (flags.h): Include it. + (lto.h): Likewise. + (lto_init): New function. + (lto_write_globals): Remove. + (LANG_HOOKS_WRITE_GLOBALS): Define to lhd_do_nothing. + (LANG_HOOKS_INIT): Define. + (LANG_HOOKS_PARSE_FILE): Likewise. + * Make-lang.in (LTO_OBJS): Add lto.o and lto-elf.o. + (LTO_EXE): Link with libelf. + (lto/lto-lang.o): Update dependencies. + (lto/lto.o): New target. + (lto/lto-elf.o): Likewise. + +2006-06-12 Mark Mitchell + + * config-lang.in: New file. + * Make-lang.in: Likewise. + * lto-tree.h: Likewise. + * lto-lang.c: Likewise. +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/ChangeLog Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/config-lang.in =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/config-lang.in (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/config-lang.in (revision 288) @@ -0,0 +1,32 @@ +# Top level configure fragment for LTO +# Copyright (C) 2009 +# Free Software Foundation, Inc. + +#This file is part of GCC. + +#GCC is free software; you can redistribute it and/or modify +#it under the terms of the GNU General Public License as published by +#the Free Software Foundation; either version 3, or (at your option) +#any later version. + +#GCC is distributed in the hope that it will be useful, +#but WITHOUT ANY WARRANTY; without even the implied warranty of +#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +#GNU General Public License for more details. + +# You should have received a copy of the GNU General Public License +# along with GCC; see the file COPYING3. If not see +# . + +language="lto" +compilers="lto1\$(exeext)" +stagestuff="lto1\$(exeext)" + +gtfiles="\$(srcdir)/lto/lto-tree.h \$(srcdir)/lto/lto-lang.c \$(srcdir)/lto/lto.c" + +# LTO is a special front end. From a user's perspective it is not +# really a language, but a middle end feature. However, the GIMPLE +# reading module is implemented as a front end, so enabling LTO means +# enabling this "language". To enable LTO functionality, use +# --enable-lto when configuring the compiler. +build_by_default=no Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.c =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.c (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.c (revision 288) @@ -0,0 +1,845 @@ +/* LTO routines for COFF object files. + Copyright 2009, 2010 Free Software Foundation, Inc. + Contributed by Dave Korn. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#include "config.h" +#include "system.h" +#include "coretypes.h" +#include "toplev.h" +#include "lto.h" +#include "tm.h" +#include "libiberty.h" +#include "ggc.h" +#include "lto-streamer.h" +#include "lto/lto-coff.h" + + +/* Rather than implementing a libcoff to match libelf, or attempting to + integrate libbfd into GCC, this file is a self-contained (and very + minimal) COFF format object file reader/writer. The generated files + will contain a COFF header, a number of COFF section headers, the + section data itself, and a trailing string table for section names. */ + +/* Handle opening elf files on hosts, such as Windows, that may use + text file handling that will break binary access. */ + +#ifndef O_BINARY +#define O_BINARY 0 +#endif + +/* Known header magics for validation, as an array. */ + +static const unsigned int coff_machine_array[] = COFF_KNOWN_MACHINES; + +/* Number of valid entries (no sentinel) in array. */ + +#define NUM_COFF_KNOWN_MACHINES \ + (sizeof (coff_machine_array) / sizeof (coff_machine_array[0])) + +/* Cached object file header. */ + +static Coff_header cached_coff_hdr; + +/* Flag to indicate if we have read and cached any header yet. */ + +static bool cached_coff_hdr_valid = false; + +/* The current output file. */ + +static lto_file *current_out_file; + + +/* Sets the current output file to FILE. Returns the old output file or + NULL. */ + +lto_file * +lto_set_current_out_file (lto_file *file) +{ + lto_file *old_file = current_out_file; + current_out_file = file; + return old_file; +} + + +/* Returns the current output file. */ + +lto_file * +lto_get_current_out_file (void) +{ + return current_out_file; +} + + +/* COFF section structure constructor. */ + +static lto_coff_section * +coff_newsection (lto_coff_file *file, const char *name, size_t type) +{ + lto_coff_section *ptr, **chain_ptr_ptr; + + ptr = XCNEW (lto_coff_section); + ptr->name = name; + ptr->type = type; + + chain_ptr_ptr = &file->section_chain; + while (*chain_ptr_ptr) + chain_ptr_ptr = &(*chain_ptr_ptr)->next; + *chain_ptr_ptr = ptr; + + return ptr; +} + + +/* COFF section data block structure constructor. */ + +static lto_coff_data * +coff_newdata (lto_coff_section *sec) +{ + lto_coff_data *ptr, **chain_ptr_ptr; + + ptr = XCNEW (lto_coff_data); + + chain_ptr_ptr = &sec->data_chain; + while (*chain_ptr_ptr) + chain_ptr_ptr = &(*chain_ptr_ptr)->next; + *chain_ptr_ptr = ptr; + + return ptr; +} + + +/* Initialize FILE, an LTO file object for FILENAME. */ + +static void +lto_file_init (lto_file *file, const char *filename, off_t offset) +{ + file->filename = filename; + file->offset = offset; +} + +/* Return an error string after an error, or a predetermined one + if ERRCODE is not -1. */ + +static const char * +coff_errmsg (int errcode) +{ + return strerror (errcode == -1 ? errno : errcode); +} + +/* Returns a hash code for P. */ + +static hashval_t +hash_name (const void *p) +{ + const struct lto_section_slot *ds = (const struct lto_section_slot *) p; + return (hashval_t) htab_hash_string (ds->name); +} + +/* Returns nonzero if P1 and P2 are equal. */ + +static int +eq_name (const void *p1, const void *p2) +{ + const struct lto_section_slot *s1 = + (const struct lto_section_slot *) p1; + const struct lto_section_slot *s2 = + (const struct lto_section_slot *) p2; + + return strcmp (s1->name, s2->name) == 0; +} + + +/* Build a hash table whose key is the section names and whose data is + the start and size of each section in the .o file. */ + +htab_t +lto_obj_build_section_table (lto_file *lto_file) +{ + lto_coff_file *coff_file = (lto_coff_file *)lto_file; + lto_coff_section *sec; + htab_t section_hash_table; + ssize_t strtab_size; + char *strtab; + + section_hash_table = htab_create (37, hash_name, eq_name, free); + + /* Seek to start of string table. */ + if (coff_file->strtab_offs != lseek (coff_file->fd, + coff_file->base.offset + coff_file->strtab_offs, SEEK_SET)) + { + error ("altered or invalid COFF object file"); + return section_hash_table; + } + + strtab_size = coff_file->file_size - coff_file->strtab_offs; + strtab = XNEWVEC (char, strtab_size); + if (read (coff_file->fd, strtab, strtab_size) != strtab_size) + { + error ("invalid COFF object file string table"); + return section_hash_table; + } + + /* Scan sections looking at names. */ + COFF_FOR_ALL_SECTIONS(coff_file, sec) + { + struct lto_section_slot s_slot; + void **slot; + char *new_name; + int stringoffset; + char *name = (char *) &sec->coffsec.Name[0]; + + /* Skip dummy string section if by any chance we see it. */ + if (sec->type == 1) + continue; + + if (name[0] == '/') + { + if (1 != sscanf (&name[1], "%d", &stringoffset) + || stringoffset < 0 || stringoffset >= strtab_size) + { + error ("invalid COFF section name string"); + continue; + } + name = strtab + stringoffset; + } + else + { + /* If we cared about the VirtualSize field, we couldn't + crudely trash it like this to guarantee nul-termination + of the Name field. But we don't, so we do. */ + name[8] = 0; + } + if (strncmp (name, LTO_SECTION_NAME_PREFIX, + strlen (LTO_SECTION_NAME_PREFIX)) != 0) + continue; + + new_name = XNEWVEC (char, strlen (name) + 1); + strcpy (new_name, name); + s_slot.name = new_name; + slot = htab_find_slot (section_hash_table, &s_slot, INSERT); + if (*slot == NULL) + { + struct lto_section_slot *new_slot = XNEW (struct lto_section_slot); + + new_slot->name = new_name; + /* The offset into the file for this section. */ + new_slot->start = coff_file->base.offset + + COFF_GET(&sec->coffsec,PointerToRawData); + new_slot->len = COFF_GET(&sec->coffsec,SizeOfRawData); + *slot = new_slot; + } + else + { + error ("two or more sections for %s:", new_name); + return NULL; + } + } + + free (strtab); + return section_hash_table; +} + + +/* Begin a new COFF section named NAME with type TYPE in the current output + file. TYPE is an SHT_* macro from the libelf headers. */ + +static void +lto_coff_begin_section_with_type (const char *name, size_t type) +{ + lto_coff_file *file; + size_t sh_name; + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_coff_file *) lto_get_current_out_file (), + gcc_assert (file); + gcc_assert (!file->scn); + + /* Create a new section. */ + file->scn = coff_newsection (file, name, type); + if (!file->scn) + fatal_error ("could not create a new COFF section: %s", coff_errmsg (-1)); + + /* Add a string table entry and record the offset. */ + gcc_assert (file->shstrtab_stream); + sh_name = file->shstrtab_stream->total_size; + lto_output_data_stream (file->shstrtab_stream, name, strlen (name) + 1); + + /* Initialize the section header. */ + file->scn->strtab_offs = sh_name; +} + + +/* Begin a new COFF section named NAME in the current output file. */ + +void +lto_obj_begin_section (const char *name) +{ + lto_coff_begin_section_with_type (name, 0); +} + + +/* Append DATA of length LEN to the current output section. BASE is a pointer + to the output page containing DATA. It is freed once the output file has + been written. */ + +void +lto_obj_append_data (const void *data, size_t len, void *block) +{ + lto_coff_file *file; + lto_coff_data *coff_data; + struct lto_char_ptr_base *base = (struct lto_char_ptr_base *) block; + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_coff_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + coff_data = coff_newdata (file->scn); + if (!coff_data) + fatal_error ("could not append data to COFF section: %s", coff_errmsg (-1)); + + coff_data->d_buf = CONST_CAST (void *, data); + coff_data->d_size = len; + + /* Chain all data blocks (from all sections) on one singly-linked + list for freeing en masse after the file is closed. */ + base->ptr = (char *)file->data; + file->data = base; +} + + +/* End the current output section. This just does some assertion checking + and sets the current output file's scn member to NULL. */ + +void +lto_obj_end_section (void) +{ + lto_coff_file *file; + + /* Grab the current output file and validate some basic assertions. */ + file = (lto_coff_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + file->scn = NULL; +} + + +/* Validate's COFF_FILE's executable header and, if cached_coff_hdr is + uninitialized, caches the results. Also records the section header string + table's section index. Returns true on success or false on failure. */ + +static bool +validate_file (lto_coff_file *coff_file) +{ + size_t n, secnum; + unsigned int numsections, secheaderssize, numsyms; + off_t sectionsstart, symbolsstart, stringsstart; + unsigned int mach, charact; + + /* Read and sanity check the raw header. */ + n = read (coff_file->fd, &coff_file->coffhdr, sizeof (coff_file->coffhdr)); + if (n != sizeof (coff_file->coffhdr)) + { + error ("not a COFF object file"); + return false; + } + + mach = COFF_GET(&coff_file->coffhdr, Machine); + for (n = 0; n < NUM_COFF_KNOWN_MACHINES; n++) + if (mach == coff_machine_array[n]) + break; + if (n == NUM_COFF_KNOWN_MACHINES) + { + error ("not a recognized COFF object file"); + return false; + } + + charact = COFF_GET(&coff_file->coffhdr, Characteristics); + if (COFF_NOT_CHARACTERISTICS & charact) + { + /* DLL, EXE or SYS file. */ + error ("not a relocatable COFF object file"); + return false; + } + + if (COFF_CHARACTERISTICS != (COFF_CHARACTERISTICS & charact)) + { + /* ECOFF/XCOFF/PE+ support not implemented. */ + error ("not a 32-bit COFF object file"); + return false; + } + + /* It validated OK, so cached it if we don't already have one. */ + if (!cached_coff_hdr_valid) + { + cached_coff_hdr_valid = true; + memcpy (&cached_coff_hdr, &coff_file->coffhdr, sizeof (cached_coff_hdr)); + } + + if (mach != COFF_GET(&cached_coff_hdr, Machine)) + { + error ("inconsistent file architecture detected"); + return false; + } + + /* Read section headers and string table? */ + + numsections = COFF_GET(&coff_file->coffhdr, NumberOfSections); + secheaderssize = numsections * sizeof (Coff_section); + sectionsstart = sizeof (Coff_header) + secheaderssize; + symbolsstart = COFF_GET(&coff_file->coffhdr, PointerToSymbolTable); + numsyms = COFF_GET(&coff_file->coffhdr, NumberOfSymbols); + stringsstart = (symbolsstart + COFF_SYMBOL_SIZE * numsyms); + +#define CVOFFSETTTED(x) (coff_file->base.offset + (x)) + + if (numsections <= 0 || symbolsstart <= 0 || numsyms <= 0 + || (CVOFFSETTTED(sectionsstart) >= coff_file->file_size) + || (CVOFFSETTTED(symbolsstart) >= coff_file->file_size) + || (CVOFFSETTTED(stringsstart) >= coff_file->file_size)) + { + error ("not a valid COFF object file"); + return false; + } + +#undef CVOFFSETTTED + + /* Record start of string table. */ + coff_file->strtab_offs = stringsstart; + + /* Validate section table entries. */ + for (secnum = 0; secnum < numsections; secnum++) + { + Coff_section coffsec; + lto_coff_section *ltosec; + off_t size_raw, offs_raw, offs_relocs, offs_lines; + off_t num_relocs, num_lines; + + n = read (coff_file->fd, &coffsec, sizeof (coffsec)); + if (n != sizeof (coffsec)) + { + error ("short/missing COFF section table"); + return false; + } + + size_raw = COFF_GET(&coffsec, SizeOfRawData); + offs_raw = COFF_GET(&coffsec, PointerToRawData); + offs_relocs = COFF_GET(&coffsec, PointerToRelocations); + offs_lines = COFF_GET(&coffsec, PointerToLinenumbers); + num_relocs = COFF_GET(&coffsec, NumberOfRelocations); + num_lines = COFF_GET(&coffsec, NumberOfLinenumbers); + + if (size_raw < 0 || num_relocs < 0 || num_lines < 0 + || (size_raw + && ((COFF_GET(&coffsec, Characteristics) + & IMAGE_SCN_CNT_UNINITIALIZED_DATA) + ? (offs_raw != 0) + : (offs_raw < sectionsstart || offs_raw >= coff_file->file_size))) + || (num_relocs + && (offs_relocs < sectionsstart + || offs_relocs >= coff_file->file_size)) + || (num_lines + && (offs_lines < sectionsstart + || offs_lines >= coff_file->file_size))) + { + error ("invalid COFF section table"); + return false; + } + + /* Looks ok, so record its details. We don't read the + string table or set up names yet; we'll do that when + we build the hash table. */ + ltosec = coff_newsection (coff_file, NULL, 0); + memcpy (<osec->coffsec, &coffsec, sizeof (ltosec->coffsec)); + } + + return true; +} + +/* Initialize COFF_FILE's executable header using cached data from previously + read files. */ + +static void +init_coffhdr (lto_coff_file *coff_file) +{ + gcc_assert (cached_coff_hdr_valid); + memset (&coff_file->coffhdr, 0, sizeof (coff_file->coffhdr)); + COFF_PUT(&coff_file->coffhdr, Machine, COFF_GET(&cached_coff_hdr, Machine)); + COFF_PUT(&coff_file->coffhdr, Characteristics, COFF_GET(&cached_coff_hdr, Characteristics)); +} + +/* Open COFF file FILENAME. If WRITABLE is true, the file is opened for write + and, if necessary, created. Otherwise, the file is opened for reading. + Returns the opened file. */ + +lto_file * +lto_obj_file_open (const char *filename, bool writable) +{ + lto_coff_file *coff_file; + lto_file *result = NULL; + off_t offset; + const char *offset_p; + char *fname; + struct stat statbuf; + + offset_p = strchr (filename, '@'); + if (!offset_p) + { + fname = xstrdup (filename); + offset = 0; + } + else + { + /* The file started with '@' is a file containing command line + options. Stop if it doesn't exist. */ + if (offset_p == filename) + fatal_error ("command line option file '%s' does not exist", + filename); + + fname = (char *) xmalloc (offset_p - filename + 1); + memcpy (fname, filename, offset_p - filename); + fname[offset_p - filename] = '\0'; + offset_p += 3; /* skip the @0x */ + offset = lto_parse_hex (offset_p); + } + + /* Set up. */ + coff_file = XCNEW (lto_coff_file); + result = (lto_file *) coff_file; + lto_file_init (result, fname, offset); + coff_file->fd = -1; + + /* Open the file. */ + coff_file->fd = open (fname, + O_BINARY | (writable ? O_WRONLY | O_CREAT | O_TRUNC : O_RDONLY), 0666); + + if (coff_file->fd == -1) + { + error ("could not open file %s", fname); + goto fail; + } + + if (stat (fname, &statbuf) < 0) + { + error ("could not stat file %s", fname); + goto fail; + } + + coff_file->file_size = statbuf.st_size; + + if (offset != 0) + { + char ar_tail[12]; + int size; + + /* Surely not? */ + gcc_assert (!writable); + + /* Seek to offset, or error. */ + if (lseek (coff_file->fd, offset, SEEK_SET) != (ssize_t) offset) + { + error ("could not find archive member @0x%lx", (long) offset); + goto fail; + } + + /* Now seek back 12 chars and read the tail of the AR header to + find the length of the member file. */ + if (lseek (coff_file->fd, -12, SEEK_CUR) < 0 + || read (coff_file->fd, ar_tail, 12) != 12 + || lseek (coff_file->fd, 0, SEEK_CUR) != (ssize_t) offset + || ar_tail[10] != '`' || ar_tail[11] != '\n') + { + error ("could not find archive header @0x%lx", (long) offset); + goto fail; + } + + ar_tail[11] = 0; + if (sscanf (ar_tail, "%d", &size) != 1) + { + error ("invalid archive header @0x%lx", (long) offset); + goto fail; + } + coff_file->file_size = size; + } + + if (writable) + { + init_coffhdr (coff_file); + coff_file->shstrtab_stream = XCNEW (struct lto_output_stream); + } + else + if (!validate_file (coff_file)) + goto fail; + + return result; + + fail: + if (result) + lto_obj_file_close (result); + return NULL; +} + + +/* Close COFF file FILE and clean up any associated data structures. If FILE + was opened for writing, the file's COFF data is written at this time, and + any cached data buffers are freed. Return TRUE if there was an error. */ + +static bool +coff_write_object_file (lto_coff_file *coff_file) +{ + lto_coff_section *cursec, *stringsec; + lto_coff_data *data; + size_t fileoffset, numsections, totalsecsize, numsyms, stringssize; + bool write_err = false; + int secnum; + + /* Infer whether this file was opened for reading or writing from the + presence or absense of an initialised stream for the string table; + do nothing if it was opened for reading. */ + if (!coff_file->shstrtab_stream) + return false; + else + { + /* Write the COFF string table into a dummy new section that + we will not write a header for. */ + lto_file *old_file = lto_set_current_out_file (&coff_file->base); + /* This recursively feeds in the data to a new section. */ + lto_coff_begin_section_with_type (".strtab", 1); + lto_write_stream (coff_file->shstrtab_stream); + lto_obj_end_section (); + lto_set_current_out_file (old_file); + free (coff_file->shstrtab_stream); + } + + /* Layout the file. Count sections (not dummy string section) and calculate + data size for all of them. */ + numsections = 0; + totalsecsize = 0; + stringssize = 0; + stringsec = NULL; + COFF_FOR_ALL_SECTIONS(coff_file, cursec) + { + lto_coff_data *data; + size_t cursecsize; + cursecsize = 0; + COFF_FOR_ALL_DATA(cursec,data) + cursecsize += data->d_size; + if (cursec->type == 0) + { + ++numsections; + totalsecsize += COFF_ALIGN(cursecsize); +#if COFF_ALIGNMENT > 1 + cursec->pad_needed = COFF_ALIGN(cursecsize) - cursecsize; +#endif + } + else + { + stringssize = cursecsize; + stringsec = cursec; + } + COFF_PUT(&cursec->coffsec, SizeOfRawData, cursecsize); + } + + /* There is a file symbol and a section symbol per section, + and each of these has a single auxiliary symbol following. */ + numsyms = 2 * (1 + numsections); + + /* Great! Now we have enough info to fill out the file header. */ + COFF_PUT(&coff_file->coffhdr, NumberOfSections, numsections); + COFF_PUT(&coff_file->coffhdr, NumberOfSymbols, numsyms); + COFF_PUT(&coff_file->coffhdr, PointerToSymbolTable, sizeof (Coff_header) + + numsections * sizeof (Coff_section) + totalsecsize); + /* The remaining members were initialised to zero or copied from + a cached header, so we leave them alone here. */ + + /* Now position all the sections, and fill out their headers. */ + fileoffset = sizeof (Coff_header) + numsections * sizeof (Coff_section); + COFF_FOR_ALL_SECTIONS(coff_file, cursec) + { + /* Skip dummy string section. */ + if (cursec->type == 1) + continue; + COFF_PUT(&cursec->coffsec, PointerToRawData, fileoffset); + fileoffset += COFF_ALIGN (COFF_GET(&cursec->coffsec, SizeOfRawData)); + COFF_PUT(&cursec->coffsec, Characteristics, COFF_SECTION_CHARACTERISTICS); + snprintf ((char *)&cursec->coffsec.Name[0], 8, "/%d", cursec->strtab_offs + 4); + } + + /* We can write the data now. As there's no way to indicate an error return + from this hook, error handling is limited to not wasting our time doing + any more writes in the event that any one fails. */ + + /* Write the COFF header. */ + write_err = (write (coff_file->fd, &coff_file->coffhdr, + sizeof (coff_file->coffhdr)) != sizeof (coff_file->coffhdr)); + + /* Write the COFF section headers. */ + COFF_FOR_ALL_SECTIONS(coff_file, cursec) + if (cursec->type == 1) /* Skip dummy string section. */ + continue; + else if (!write_err) + write_err = (write (coff_file->fd, &cursec->coffsec, + sizeof (cursec->coffsec)) != sizeof (cursec->coffsec)); + else + break; + + /* Write the COFF sections. */ + COFF_FOR_ALL_SECTIONS(coff_file, cursec) + { +#if COFF_ALIGNMENT > 1 + static const char padzeros[COFF_ALIGNMENT] = { 0 }; +#endif + /* Skip dummy string section. */ + if (cursec->type == 1) + continue; + COFF_FOR_ALL_DATA(cursec, data) + if (!write_err) + write_err = (write (coff_file->fd, data->d_buf, data->d_size) + != data->d_size); + else + break; +#if COFF_ALIGNMENT > 1 + if (!write_err && cursec->pad_needed) + write_err = (write (coff_file->fd, padzeros, cursec->pad_needed) + != cursec->pad_needed); +#endif + } + + /* Write the COFF symbol table. */ + if (!write_err) + { + union + { + Coff_symbol sym; + Coff_aux_sym_file file; + Coff_aux_sym_section sec; + } symbols[2]; + memset (&symbols[0], 0, sizeof (symbols)); + strcpy ((char *) &symbols[0].sym.Name[0], ".file"); + COFF_PUT(&symbols[0].sym, SectionNumber, IMAGE_SYM_DEBUG); + COFF_PUT(&symbols[0].sym, Type, IMAGE_SYM_TYPE); + symbols[0].sym.StorageClass[0] = IMAGE_SYM_CLASS_FILE; + symbols[0].sym.NumberOfAuxSymbols[0] = 1; + snprintf ((char *)symbols[1].file.FileName, + sizeof (symbols[1].file.FileName), + "%s", lbasename (coff_file->base.filename)); + write_err = (write (coff_file->fd, &symbols[0], sizeof (symbols)) + != (2 * COFF_SYMBOL_SIZE)); + + /* Set up constant parts for section sym loop. */ + memset (&symbols[0], 0, sizeof (symbols)); + COFF_PUT(&symbols[0].sym, Type, IMAGE_SYM_TYPE); + symbols[0].sym.StorageClass[0] = IMAGE_SYM_CLASS_STATIC; + symbols[0].sym.NumberOfAuxSymbols[0] = 1; + + secnum = 1; + if (!write_err) + COFF_FOR_ALL_SECTIONS(coff_file, cursec) + { + /* Skip dummy string section. */ + if (cursec->type == 1) + continue; + /* Reuse section name string for section symbol name. */ + COFF_PUT_NDXSZ(&symbols[0].sym, Name, 0, 0, 4); + COFF_PUT_NDXSZ(&symbols[0].sym, Name, cursec->strtab_offs + 4, 4, 4); + COFF_PUT(&symbols[0].sym, SectionNumber, secnum++); + COFF_PUT(&symbols[1].sec, Length, + COFF_GET(&cursec->coffsec, SizeOfRawData)); + if (!write_err) + write_err = (write (coff_file->fd, &symbols[0], sizeof (symbols)) + != (2 * COFF_SYMBOL_SIZE)); + else + break; + } + } + + /* Write the COFF string table. */ + if (!write_err) + { + unsigned char outlen[4]; + COFF_PUT4(outlen, stringssize + 4); + if (!write_err) + write_err = (write (coff_file->fd, outlen, 4) != 4); + if (stringsec) + COFF_FOR_ALL_DATA(stringsec, data) + if (!write_err) + write_err = (write (coff_file->fd, data->d_buf, data->d_size) + != data->d_size); + else + break; + } + + return write_err; +} + +/* Close COFF file FILE and clean up any associated data structures. If FILE + was opened for writing, the file's COFF data is written at this time, and + any cached data buffers are freed. */ + +void +lto_obj_file_close (lto_file *file) +{ + lto_coff_file *coff_file = (lto_coff_file *) file; + struct lto_char_ptr_base *cur, *tmp; + lto_coff_section *cursec, *nextsec; + bool write_err = false; + + /* Write the COFF string table into a dummy new section that + we will not write a header for. */ + if (coff_file->shstrtab_stream) + coff_write_object_file (coff_file); + + /* Close the file, we're done. */ + if (coff_file->fd != -1) + close (coff_file->fd); + + /* Free any data buffers. */ + cur = coff_file->data; + while (cur) + { + tmp = cur; + cur = (struct lto_char_ptr_base *) cur->ptr; + free (tmp); + } + + /* Free any sections and their data chains. */ + cursec = coff_file->section_chain; + while (cursec) + { + lto_coff_data *curdata, *nextdata; + nextsec = cursec->next; + curdata = cursec->data_chain; + while (curdata) + { + nextdata = curdata->next; + free (curdata); + curdata = nextdata; + } + free (cursec); + cursec = nextsec; + } + + free (file); + + /* If there was an error, mention it. */ + if (write_err) + error ("I/O error writing COFF output file"); +} +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-lang.c =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-lang.c (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-lang.c (revision 288) @@ -0,0 +1,1190 @@ +/* Language-dependent hooks for LTO. + Copyright 2009 Free Software Foundation, Inc. + Contributed by CodeSourcery, Inc. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#include "config.h" +#include "system.h" +#include "coretypes.h" +#include "flags.h" +#include "tm.h" +#include "tree.h" +#include "expr.h" +#include "target.h" +#include "langhooks.h" +#include "langhooks-def.h" +#include "debug.h" +#include "lto-tree.h" +#include "lto.h" +#include "tree-inline.h" +#include "gimple.h" +#include "toplev.h" + +static tree handle_noreturn_attribute (tree *, tree, tree, int, bool *); +static tree handle_const_attribute (tree *, tree, tree, int, bool *); +static tree handle_malloc_attribute (tree *, tree, tree, int, bool *); +static tree handle_pure_attribute (tree *, tree, tree, int, bool *); +static tree handle_novops_attribute (tree *, tree, tree, int, bool *); +static tree handle_nonnull_attribute (tree *, tree, tree, int, bool *); +static tree handle_nothrow_attribute (tree *, tree, tree, int, bool *); +static tree handle_sentinel_attribute (tree *, tree, tree, int, bool *); +static tree handle_type_generic_attribute (tree *, tree, tree, int, bool *); +static tree handle_format_attribute (tree *, tree, tree, int, bool *); +static tree handle_format_arg_attribute (tree *, tree, tree, int, bool *); + +/* Table of machine-independent attributes supported in GIMPLE. */ +const struct attribute_spec lto_attribute_table[] = +{ + /* { name, min_len, max_len, decl_req, type_req, fn_type_req, handler } */ + { "noreturn", 0, 0, true, false, false, + handle_noreturn_attribute }, + /* The same comments as for noreturn attributes apply to const ones. */ + { "const", 0, 0, true, false, false, + handle_const_attribute }, + { "malloc", 0, 0, true, false, false, + handle_malloc_attribute }, + { "pure", 0, 0, true, false, false, + handle_pure_attribute }, + { "no vops", 0, 0, true, false, false, + handle_novops_attribute }, + { "nonnull", 0, -1, false, true, true, + handle_nonnull_attribute }, + { "nothrow", 0, 0, true, false, false, + handle_nothrow_attribute }, + { "sentinel", 0, 1, false, true, true, + handle_sentinel_attribute }, + { "type generic", 0, 0, false, true, true, + handle_type_generic_attribute }, + { NULL, 0, 0, false, false, false, NULL } +}; + +/* Give the specifications for the format attributes, used by C and all + descendants. */ + +const struct attribute_spec lto_format_attribute_table[] = +{ + /* { name, min_len, max_len, decl_req, type_req, fn_type_req, handler } */ + { "format", 3, 3, false, true, true, + handle_format_attribute }, + { "format_arg", 1, 1, false, true, true, + handle_format_arg_attribute }, + { NULL, 0, 0, false, false, false, NULL } +}; + +enum built_in_attribute +{ +#define DEF_ATTR_NULL_TREE(ENUM) ENUM, +#define DEF_ATTR_INT(ENUM, VALUE) ENUM, +#define DEF_ATTR_IDENT(ENUM, STRING) ENUM, +#define DEF_ATTR_TREE_LIST(ENUM, PURPOSE, VALUE, CHAIN) ENUM, +#include "builtin-attrs.def" +#undef DEF_ATTR_NULL_TREE +#undef DEF_ATTR_INT +#undef DEF_ATTR_IDENT +#undef DEF_ATTR_TREE_LIST + ATTR_LAST +}; + +static GTY(()) tree built_in_attributes[(int) ATTR_LAST]; + +/* Builtin types. */ + +enum lto_builtin_type +{ +#define DEF_PRIMITIVE_TYPE(NAME, VALUE) NAME, +#define DEF_FUNCTION_TYPE_0(NAME, RETURN) NAME, +#define DEF_FUNCTION_TYPE_1(NAME, RETURN, ARG1) NAME, +#define DEF_FUNCTION_TYPE_2(NAME, RETURN, ARG1, ARG2) NAME, +#define DEF_FUNCTION_TYPE_3(NAME, RETURN, ARG1, ARG2, ARG3) NAME, +#define DEF_FUNCTION_TYPE_4(NAME, RETURN, ARG1, ARG2, ARG3, ARG4) NAME, +#define DEF_FUNCTION_TYPE_5(NAME, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5) NAME, +#define DEF_FUNCTION_TYPE_6(NAME, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5, ARG6) NAME, +#define DEF_FUNCTION_TYPE_7(NAME, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7) NAME, +#define DEF_FUNCTION_TYPE_VAR_0(NAME, RETURN) NAME, +#define DEF_FUNCTION_TYPE_VAR_1(NAME, RETURN, ARG1) NAME, +#define DEF_FUNCTION_TYPE_VAR_2(NAME, RETURN, ARG1, ARG2) NAME, +#define DEF_FUNCTION_TYPE_VAR_3(NAME, RETURN, ARG1, ARG2, ARG3) NAME, +#define DEF_FUNCTION_TYPE_VAR_4(NAME, RETURN, ARG1, ARG2, ARG3, ARG4) NAME, +#define DEF_FUNCTION_TYPE_VAR_5(NAME, RETURN, ARG1, ARG2, ARG3, ARG4, ARG6) \ + NAME, +#define DEF_POINTER_TYPE(NAME, TYPE) NAME, +#include "builtin-types.def" +#undef DEF_PRIMITIVE_TYPE +#undef DEF_FUNCTION_TYPE_0 +#undef DEF_FUNCTION_TYPE_1 +#undef DEF_FUNCTION_TYPE_2 +#undef DEF_FUNCTION_TYPE_3 +#undef DEF_FUNCTION_TYPE_4 +#undef DEF_FUNCTION_TYPE_5 +#undef DEF_FUNCTION_TYPE_6 +#undef DEF_FUNCTION_TYPE_7 +#undef DEF_FUNCTION_TYPE_VAR_0 +#undef DEF_FUNCTION_TYPE_VAR_1 +#undef DEF_FUNCTION_TYPE_VAR_2 +#undef DEF_FUNCTION_TYPE_VAR_3 +#undef DEF_FUNCTION_TYPE_VAR_4 +#undef DEF_FUNCTION_TYPE_VAR_5 +#undef DEF_POINTER_TYPE + BT_LAST +}; + +typedef enum lto_builtin_type builtin_type; + +static GTY(()) tree builtin_types[(int) BT_LAST + 1]; + +static GTY(()) tree string_type_node; +static GTY(()) tree const_string_type_node; +static GTY(()) tree wint_type_node; +static GTY(()) tree intmax_type_node; +static GTY(()) tree uintmax_type_node; +static GTY(()) tree signed_size_type_node; + +/* Flags needed to process builtins.def. */ +int flag_no_builtin; +int flag_no_nonansi_builtin; +int flag_isoc94; +int flag_isoc99; + +/* Attribute handlers. */ + +/* Handle a "noreturn" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_noreturn_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + tree type = TREE_TYPE (*node); + + if (TREE_CODE (*node) == FUNCTION_DECL) + TREE_THIS_VOLATILE (*node) = 1; + else if (TREE_CODE (type) == POINTER_TYPE + && TREE_CODE (TREE_TYPE (type)) == FUNCTION_TYPE) + TREE_TYPE (*node) + = build_pointer_type + (build_type_variant (TREE_TYPE (type), + TYPE_READONLY (TREE_TYPE (type)), 1)); + else + gcc_unreachable (); + + return NULL_TREE; +} + + +/* Handle a "const" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_const_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + tree type = TREE_TYPE (*node); + + /* See FIXME comment on noreturn in c_common_attribute_table. */ + if (TREE_CODE (*node) == FUNCTION_DECL) + TREE_READONLY (*node) = 1; + else if (TREE_CODE (type) == POINTER_TYPE + && TREE_CODE (TREE_TYPE (type)) == FUNCTION_TYPE) + TREE_TYPE (*node) + = build_pointer_type + (build_type_variant (TREE_TYPE (type), 1, + TREE_THIS_VOLATILE (TREE_TYPE (type)))); + else + gcc_unreachable (); + + return NULL_TREE; +} + + +/* Handle a "malloc" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_malloc_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + if (TREE_CODE (*node) == FUNCTION_DECL + && POINTER_TYPE_P (TREE_TYPE (TREE_TYPE (*node)))) + DECL_IS_MALLOC (*node) = 1; + else + gcc_unreachable (); + + return NULL_TREE; +} + + +/* Handle a "pure" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_pure_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + if (TREE_CODE (*node) == FUNCTION_DECL) + DECL_PURE_P (*node) = 1; + else + gcc_unreachable (); + + return NULL_TREE; +} + + +/* Handle a "no vops" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_novops_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool *ARG_UNUSED (no_add_attrs)) +{ + gcc_assert (TREE_CODE (*node) == FUNCTION_DECL); + DECL_IS_NOVOPS (*node) = 1; + return NULL_TREE; +} + + +/* Helper for nonnull attribute handling; fetch the operand number + from the attribute argument list. */ + +static bool +get_nonnull_operand (tree arg_num_expr, unsigned HOST_WIDE_INT *valp) +{ + /* Verify the arg number is a constant. */ + if (TREE_CODE (arg_num_expr) != INTEGER_CST + || TREE_INT_CST_HIGH (arg_num_expr) != 0) + return false; + + *valp = TREE_INT_CST_LOW (arg_num_expr); + return true; +} + +/* Handle the "nonnull" attribute. */ + +static tree +handle_nonnull_attribute (tree *node, tree ARG_UNUSED (name), + tree args, int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + tree type = *node; + + /* If no arguments are specified, all pointer arguments should be + non-null. Verify a full prototype is given so that the arguments + will have the correct types when we actually check them later. */ + if (!args) + { + gcc_assert (TYPE_ARG_TYPES (type)); + return NULL_TREE; + } + + /* Argument list specified. Verify that each argument number references + a pointer argument. */ + for (; args; args = TREE_CHAIN (args)) + { + tree argument; + unsigned HOST_WIDE_INT arg_num = 0, ck_num; + + if (!get_nonnull_operand (TREE_VALUE (args), &arg_num)) + gcc_unreachable (); + + argument = TYPE_ARG_TYPES (type); + if (argument) + { + for (ck_num = 1; ; ck_num++) + { + if (!argument || ck_num == arg_num) + break; + argument = TREE_CHAIN (argument); + } + + gcc_assert (argument + && TREE_CODE (TREE_VALUE (argument)) == POINTER_TYPE); + } + } + + return NULL_TREE; +} + + +/* Handle a "nothrow" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_nothrow_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + if (TREE_CODE (*node) == FUNCTION_DECL) + TREE_NOTHROW (*node) = 1; + else + gcc_unreachable (); + + return NULL_TREE; +} + + +/* Handle a "sentinel" attribute. */ + +static tree +handle_sentinel_attribute (tree *node, tree ARG_UNUSED (name), tree args, + int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + tree params = TYPE_ARG_TYPES (*node); + gcc_assert (params); + + while (TREE_CHAIN (params)) + params = TREE_CHAIN (params); + + gcc_assert (!VOID_TYPE_P (TREE_VALUE (params))); + + if (args) + { + tree position = TREE_VALUE (args); + gcc_assert (TREE_CODE (position) == INTEGER_CST); + if (tree_int_cst_lt (position, integer_zero_node)) + gcc_unreachable (); + } + + return NULL_TREE; +} + +/* Handle a "type_generic" attribute. */ + +static tree +handle_type_generic_attribute (tree *node, tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool * ARG_UNUSED (no_add_attrs)) +{ + tree params; + + /* Ensure we have a function type. */ + gcc_assert (TREE_CODE (*node) == FUNCTION_TYPE); + + params = TYPE_ARG_TYPES (*node); + while (params && ! VOID_TYPE_P (TREE_VALUE (params))) + params = TREE_CHAIN (params); + + /* Ensure we have a variadic function. */ + gcc_assert (!params); + + return NULL_TREE; +} + +/* Handle a "format" attribute; arguments as in + struct attribute_spec.handler. */ + +static tree +handle_format_attribute (tree * ARG_UNUSED (node), tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool *no_add_attrs) +{ + *no_add_attrs = true; + return NULL_TREE; +} + + +/* Handle a "format_arg" attribute; arguments as in + struct attribute_spec.handler. */ + +tree +handle_format_arg_attribute (tree * ARG_UNUSED (node), tree ARG_UNUSED (name), + tree ARG_UNUSED (args), int ARG_UNUSED (flags), + bool *no_add_attrs) +{ + *no_add_attrs = true; + return NULL_TREE; +} + + +/* Cribbed from c-common.c. */ + +static void +def_fn_type (builtin_type def, builtin_type ret, bool var, int n, ...) +{ + tree args = NULL, t; + va_list list; + int i; + + va_start (list, n); + for (i = 0; i < n; ++i) + { + builtin_type a = (builtin_type) va_arg (list, int); + t = builtin_types[a]; + if (t == error_mark_node) + goto egress; + args = tree_cons (NULL_TREE, t, args); + } + va_end (list); + + args = nreverse (args); + if (!var) + args = chainon (args, void_list_node); + + t = builtin_types[ret]; + if (t == error_mark_node) + goto egress; + t = build_function_type (t, args); + + egress: + builtin_types[def] = t; +} + +/* Used to help initialize the builtin-types.def table. When a type of + the correct size doesn't exist, use error_mark_node instead of NULL. + The later results in segfaults even when a decl using the type doesn't + get invoked. */ + +static tree +builtin_type_for_size (int size, bool unsignedp) +{ + tree type = lang_hooks.types.type_for_size (size, unsignedp); + return type ? type : error_mark_node; +} + +/* Support for DEF_BUILTIN. */ + +static void +def_builtin_1 (enum built_in_function fncode, const char *name, + enum built_in_class fnclass, tree fntype, tree libtype, + bool both_p, bool fallback_p, bool nonansi_p, + tree fnattrs, bool implicit_p) +{ + tree decl; + const char *libname; + + if (fntype == error_mark_node) + return; + + libname = name + strlen ("__builtin_"); + decl = add_builtin_function (name, fntype, fncode, fnclass, + (fallback_p ? libname : NULL), + fnattrs); + + if (both_p + && !flag_no_builtin + && !(nonansi_p && flag_no_nonansi_builtin)) + add_builtin_function (libname, libtype, fncode, fnclass, + NULL, fnattrs); + + built_in_decls[(int) fncode] = decl; + if (implicit_p) + implicit_built_in_decls[(int) fncode] = decl; +} + + +/* Initialize the attribute table for all the supported builtins. */ + +static void +lto_init_attributes (void) +{ + /* Fill in the built_in_attributes array. */ +#define DEF_ATTR_NULL_TREE(ENUM) \ + built_in_attributes[(int) ENUM] = NULL_TREE; +#define DEF_ATTR_INT(ENUM, VALUE) \ + built_in_attributes[(int) ENUM] = build_int_cst (NULL_TREE, VALUE); +#define DEF_ATTR_IDENT(ENUM, STRING) \ + built_in_attributes[(int) ENUM] = get_identifier (STRING); +#define DEF_ATTR_TREE_LIST(ENUM, PURPOSE, VALUE, CHAIN) \ + built_in_attributes[(int) ENUM] \ + = tree_cons (built_in_attributes[(int) PURPOSE], \ + built_in_attributes[(int) VALUE], \ + built_in_attributes[(int) CHAIN]); +#include "builtin-attrs.def" +#undef DEF_ATTR_NULL_TREE +#undef DEF_ATTR_INT +#undef DEF_ATTR_IDENT +#undef DEF_ATTR_TREE_LIST +} + +/* Create builtin types and functions. VA_LIST_REF_TYPE_NODE and + VA_LIST_ARG_TYPE_NODE are used in builtin-types.def. */ + +static void +lto_define_builtins (tree va_list_ref_type_node ATTRIBUTE_UNUSED, + tree va_list_arg_type_node ATTRIBUTE_UNUSED) +{ +#define DEF_PRIMITIVE_TYPE(ENUM, VALUE) \ + builtin_types[ENUM] = VALUE; +#define DEF_FUNCTION_TYPE_0(ENUM, RETURN) \ + def_fn_type (ENUM, RETURN, 0, 0); +#define DEF_FUNCTION_TYPE_1(ENUM, RETURN, ARG1) \ + def_fn_type (ENUM, RETURN, 0, 1, ARG1); +#define DEF_FUNCTION_TYPE_2(ENUM, RETURN, ARG1, ARG2) \ + def_fn_type (ENUM, RETURN, 0, 2, ARG1, ARG2); +#define DEF_FUNCTION_TYPE_3(ENUM, RETURN, ARG1, ARG2, ARG3) \ + def_fn_type (ENUM, RETURN, 0, 3, ARG1, ARG2, ARG3); +#define DEF_FUNCTION_TYPE_4(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4) \ + def_fn_type (ENUM, RETURN, 0, 4, ARG1, ARG2, ARG3, ARG4); +#define DEF_FUNCTION_TYPE_5(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5) \ + def_fn_type (ENUM, RETURN, 0, 5, ARG1, ARG2, ARG3, ARG4, ARG5); +#define DEF_FUNCTION_TYPE_6(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5, \ + ARG6) \ + def_fn_type (ENUM, RETURN, 0, 6, ARG1, ARG2, ARG3, ARG4, ARG5, ARG6); +#define DEF_FUNCTION_TYPE_7(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5, \ + ARG6, ARG7) \ + def_fn_type (ENUM, RETURN, 0, 7, ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7); +#define DEF_FUNCTION_TYPE_VAR_0(ENUM, RETURN) \ + def_fn_type (ENUM, RETURN, 1, 0); +#define DEF_FUNCTION_TYPE_VAR_1(ENUM, RETURN, ARG1) \ + def_fn_type (ENUM, RETURN, 1, 1, ARG1); +#define DEF_FUNCTION_TYPE_VAR_2(ENUM, RETURN, ARG1, ARG2) \ + def_fn_type (ENUM, RETURN, 1, 2, ARG1, ARG2); +#define DEF_FUNCTION_TYPE_VAR_3(ENUM, RETURN, ARG1, ARG2, ARG3) \ + def_fn_type (ENUM, RETURN, 1, 3, ARG1, ARG2, ARG3); +#define DEF_FUNCTION_TYPE_VAR_4(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4) \ + def_fn_type (ENUM, RETURN, 1, 4, ARG1, ARG2, ARG3, ARG4); +#define DEF_FUNCTION_TYPE_VAR_5(ENUM, RETURN, ARG1, ARG2, ARG3, ARG4, ARG5) \ + def_fn_type (ENUM, RETURN, 1, 5, ARG1, ARG2, ARG3, ARG4, ARG5); +#define DEF_POINTER_TYPE(ENUM, TYPE) \ + builtin_types[(int) ENUM] = build_pointer_type (builtin_types[(int) TYPE]); + +#include "builtin-types.def" + +#undef DEF_PRIMITIVE_TYPE +#undef DEF_FUNCTION_TYPE_1 +#undef DEF_FUNCTION_TYPE_2 +#undef DEF_FUNCTION_TYPE_3 +#undef DEF_FUNCTION_TYPE_4 +#undef DEF_FUNCTION_TYPE_5 +#undef DEF_FUNCTION_TYPE_6 +#undef DEF_FUNCTION_TYPE_VAR_0 +#undef DEF_FUNCTION_TYPE_VAR_1 +#undef DEF_FUNCTION_TYPE_VAR_2 +#undef DEF_FUNCTION_TYPE_VAR_3 +#undef DEF_FUNCTION_TYPE_VAR_4 +#undef DEF_FUNCTION_TYPE_VAR_5 +#undef DEF_POINTER_TYPE + builtin_types[(int) BT_LAST] = NULL_TREE; + + lto_init_attributes (); + +#define DEF_BUILTIN(ENUM, NAME, CLASS, TYPE, LIBTYPE, BOTH_P, FALLBACK_P,\ + NONANSI_P, ATTRS, IMPLICIT, COND) \ + if (NAME && COND) \ + def_builtin_1 (ENUM, NAME, CLASS, builtin_types[(int) TYPE], \ + builtin_types[(int) LIBTYPE], BOTH_P, FALLBACK_P, \ + NONANSI_P, built_in_attributes[(int) ATTRS], IMPLICIT); +#include "builtins.def" +#undef DEF_BUILTIN +} + +static GTY(()) tree registered_builtin_types; + +/* A chain of builtin functions that we need to recognize. We will + assume that all other function names we see will be defined by the + user's program. */ +static GTY(()) tree registered_builtin_fndecls; + +/* Language hooks. */ + +static unsigned int +lto_init_options (unsigned int argc ATTRIBUTE_UNUSED, + const char **argv ATTRIBUTE_UNUSED) +{ + /* By default, C99-like requirements for complex multiply and divide. + ??? Until the complex method is encoded in the IL this is the only + safe choice. This will pessimize Fortran code with LTO unless + people specify a complex method manually or use -ffast-math. */ + flag_complex_method = 2; + + return CL_LTO; +} + +/* Handle command-line option SCODE. If the option takes an argument, it is + stored in ARG, which is otherwise NULL. VALUE holds either a numerical + argument or a binary value indicating whether the positive or negative form + of the option was supplied. */ + +const char *resolution_file_name; +static int +lto_handle_option (size_t scode, const char *arg, int value ATTRIBUTE_UNUSED) +{ + enum opt_code code = (enum opt_code) scode; + int result = 1; + + switch (code) + { + case OPT_fresolution: + resolution_file_name = arg; + result = 1; + break; + + case OPT_Wabi: + warn_psabi = value; + break; + + case OPT_fsigned_char: + flag_signed_char = value; + break; + + case OPT_funsigned_char: + flag_signed_char = !value; + break; + + default: + break; + } + + return result; +} + +/* Perform post-option processing. Does additional initialization based on + command-line options. PFILENAME is the main input filename. Returns false + to enable subsequent back-end initialization. */ + +static bool +lto_post_options (const char **pfilename ATTRIBUTE_UNUSED) +{ + /* -fltrans and -fwpa are mutually exclusive. Check for that here. */ + if (flag_wpa && flag_ltrans) + error ("-fwpa and -fltrans are mutually exclusive"); + + if (flag_ltrans) + { + flag_generate_lto = 0; + + /* During LTRANS, we are not looking at the whole program, only + a subset of the whole callgraph. */ + flag_whole_program = 0; + } + + if (flag_wpa) + flag_generate_lto = 1; + + /* Excess precision other than "fast" requires front-end + support. */ + flag_excess_precision_cmdline = EXCESS_PRECISION_FAST; + + lto_read_all_file_options (); + + /* Initialize the compiler back end. */ + return false; +} + +/* Return an integer type with PRECISION bits of precision, + that is unsigned if UNSIGNEDP is nonzero, otherwise signed. */ + +static tree +lto_type_for_size (unsigned precision, int unsignedp) +{ + if (precision == TYPE_PRECISION (integer_type_node)) + return unsignedp ? unsigned_type_node : integer_type_node; + + if (precision == TYPE_PRECISION (signed_char_type_node)) + return unsignedp ? unsigned_char_type_node : signed_char_type_node; + + if (precision == TYPE_PRECISION (short_integer_type_node)) + return unsignedp ? short_unsigned_type_node : short_integer_type_node; + + if (precision == TYPE_PRECISION (long_integer_type_node)) + return unsignedp ? long_unsigned_type_node : long_integer_type_node; + + if (precision == TYPE_PRECISION (long_long_integer_type_node)) + return unsignedp + ? long_long_unsigned_type_node + : long_long_integer_type_node; + + if (precision <= TYPE_PRECISION (intQI_type_node)) + return unsignedp ? unsigned_intQI_type_node : intQI_type_node; + + if (precision <= TYPE_PRECISION (intHI_type_node)) + return unsignedp ? unsigned_intHI_type_node : intHI_type_node; + + if (precision <= TYPE_PRECISION (intSI_type_node)) + return unsignedp ? unsigned_intSI_type_node : intSI_type_node; + + if (precision <= TYPE_PRECISION (intDI_type_node)) + return unsignedp ? unsigned_intDI_type_node : intDI_type_node; + + if (precision <= TYPE_PRECISION (intTI_type_node)) + return unsignedp ? unsigned_intTI_type_node : intTI_type_node; + + return NULL_TREE; +} + + +/* Return a data type that has machine mode MODE. + If the mode is an integer, + then UNSIGNEDP selects between signed and unsigned types. + If the mode is a fixed-point mode, + then UNSIGNEDP selects between saturating and nonsaturating types. */ + +static tree +lto_type_for_mode (enum machine_mode mode, int unsigned_p) +{ + tree t; + + if (mode == TYPE_MODE (integer_type_node)) + return unsigned_p ? unsigned_type_node : integer_type_node; + + if (mode == TYPE_MODE (signed_char_type_node)) + return unsigned_p ? unsigned_char_type_node : signed_char_type_node; + + if (mode == TYPE_MODE (short_integer_type_node)) + return unsigned_p ? short_unsigned_type_node : short_integer_type_node; + + if (mode == TYPE_MODE (long_integer_type_node)) + return unsigned_p ? long_unsigned_type_node : long_integer_type_node; + + if (mode == TYPE_MODE (long_long_integer_type_node)) + return unsigned_p ? long_long_unsigned_type_node : long_long_integer_type_node; + + if (mode == QImode) + return unsigned_p ? unsigned_intQI_type_node : intQI_type_node; + + if (mode == HImode) + return unsigned_p ? unsigned_intHI_type_node : intHI_type_node; + + if (mode == SImode) + return unsigned_p ? unsigned_intSI_type_node : intSI_type_node; + + if (mode == DImode) + return unsigned_p ? unsigned_intDI_type_node : intDI_type_node; + +#if HOST_BITS_PER_WIDE_INT >= 64 + if (mode == TYPE_MODE (intTI_type_node)) + return unsigned_p ? unsigned_intTI_type_node : intTI_type_node; +#endif + + if (mode == TYPE_MODE (float_type_node)) + return float_type_node; + + if (mode == TYPE_MODE (double_type_node)) + return double_type_node; + + if (mode == TYPE_MODE (long_double_type_node)) + return long_double_type_node; + + if (mode == TYPE_MODE (void_type_node)) + return void_type_node; + + if (mode == TYPE_MODE (build_pointer_type (char_type_node))) + return (unsigned_p + ? make_unsigned_type (GET_MODE_PRECISION (mode)) + : make_signed_type (GET_MODE_PRECISION (mode))); + + if (mode == TYPE_MODE (build_pointer_type (integer_type_node))) + return (unsigned_p + ? make_unsigned_type (GET_MODE_PRECISION (mode)) + : make_signed_type (GET_MODE_PRECISION (mode))); + + if (COMPLEX_MODE_P (mode)) + { + enum machine_mode inner_mode; + tree inner_type; + + if (mode == TYPE_MODE (complex_float_type_node)) + return complex_float_type_node; + if (mode == TYPE_MODE (complex_double_type_node)) + return complex_double_type_node; + if (mode == TYPE_MODE (complex_long_double_type_node)) + return complex_long_double_type_node; + + if (mode == TYPE_MODE (complex_integer_type_node) && !unsigned_p) + return complex_integer_type_node; + + inner_mode = GET_MODE_INNER (mode); + inner_type = lto_type_for_mode (inner_mode, unsigned_p); + if (inner_type != NULL_TREE) + return build_complex_type (inner_type); + } + else if (VECTOR_MODE_P (mode)) + { + enum machine_mode inner_mode = GET_MODE_INNER (mode); + tree inner_type = lto_type_for_mode (inner_mode, unsigned_p); + if (inner_type != NULL_TREE) + return build_vector_type_for_mode (inner_type, mode); + } + + if (mode == TYPE_MODE (dfloat32_type_node)) + return dfloat32_type_node; + if (mode == TYPE_MODE (dfloat64_type_node)) + return dfloat64_type_node; + if (mode == TYPE_MODE (dfloat128_type_node)) + return dfloat128_type_node; + + if (ALL_SCALAR_FIXED_POINT_MODE_P (mode)) + { + if (mode == TYPE_MODE (short_fract_type_node)) + return unsigned_p ? sat_short_fract_type_node : short_fract_type_node; + if (mode == TYPE_MODE (fract_type_node)) + return unsigned_p ? sat_fract_type_node : fract_type_node; + if (mode == TYPE_MODE (long_fract_type_node)) + return unsigned_p ? sat_long_fract_type_node : long_fract_type_node; + if (mode == TYPE_MODE (long_long_fract_type_node)) + return unsigned_p ? sat_long_long_fract_type_node + : long_long_fract_type_node; + + if (mode == TYPE_MODE (unsigned_short_fract_type_node)) + return unsigned_p ? sat_unsigned_short_fract_type_node + : unsigned_short_fract_type_node; + if (mode == TYPE_MODE (unsigned_fract_type_node)) + return unsigned_p ? sat_unsigned_fract_type_node + : unsigned_fract_type_node; + if (mode == TYPE_MODE (unsigned_long_fract_type_node)) + return unsigned_p ? sat_unsigned_long_fract_type_node + : unsigned_long_fract_type_node; + if (mode == TYPE_MODE (unsigned_long_long_fract_type_node)) + return unsigned_p ? sat_unsigned_long_long_fract_type_node + : unsigned_long_long_fract_type_node; + + if (mode == TYPE_MODE (short_accum_type_node)) + return unsigned_p ? sat_short_accum_type_node : short_accum_type_node; + if (mode == TYPE_MODE (accum_type_node)) + return unsigned_p ? sat_accum_type_node : accum_type_node; + if (mode == TYPE_MODE (long_accum_type_node)) + return unsigned_p ? sat_long_accum_type_node : long_accum_type_node; + if (mode == TYPE_MODE (long_long_accum_type_node)) + return unsigned_p ? sat_long_long_accum_type_node + : long_long_accum_type_node; + + if (mode == TYPE_MODE (unsigned_short_accum_type_node)) + return unsigned_p ? sat_unsigned_short_accum_type_node + : unsigned_short_accum_type_node; + if (mode == TYPE_MODE (unsigned_accum_type_node)) + return unsigned_p ? sat_unsigned_accum_type_node + : unsigned_accum_type_node; + if (mode == TYPE_MODE (unsigned_long_accum_type_node)) + return unsigned_p ? sat_unsigned_long_accum_type_node + : unsigned_long_accum_type_node; + if (mode == TYPE_MODE (unsigned_long_long_accum_type_node)) + return unsigned_p ? sat_unsigned_long_long_accum_type_node + : unsigned_long_long_accum_type_node; + + if (mode == QQmode) + return unsigned_p ? sat_qq_type_node : qq_type_node; + if (mode == HQmode) + return unsigned_p ? sat_hq_type_node : hq_type_node; + if (mode == SQmode) + return unsigned_p ? sat_sq_type_node : sq_type_node; + if (mode == DQmode) + return unsigned_p ? sat_dq_type_node : dq_type_node; + if (mode == TQmode) + return unsigned_p ? sat_tq_type_node : tq_type_node; + + if (mode == UQQmode) + return unsigned_p ? sat_uqq_type_node : uqq_type_node; + if (mode == UHQmode) + return unsigned_p ? sat_uhq_type_node : uhq_type_node; + if (mode == USQmode) + return unsigned_p ? sat_usq_type_node : usq_type_node; + if (mode == UDQmode) + return unsigned_p ? sat_udq_type_node : udq_type_node; + if (mode == UTQmode) + return unsigned_p ? sat_utq_type_node : utq_type_node; + + if (mode == HAmode) + return unsigned_p ? sat_ha_type_node : ha_type_node; + if (mode == SAmode) + return unsigned_p ? sat_sa_type_node : sa_type_node; + if (mode == DAmode) + return unsigned_p ? sat_da_type_node : da_type_node; + if (mode == TAmode) + return unsigned_p ? sat_ta_type_node : ta_type_node; + + if (mode == UHAmode) + return unsigned_p ? sat_uha_type_node : uha_type_node; + if (mode == USAmode) + return unsigned_p ? sat_usa_type_node : usa_type_node; + if (mode == UDAmode) + return unsigned_p ? sat_uda_type_node : uda_type_node; + if (mode == UTAmode) + return unsigned_p ? sat_uta_type_node : uta_type_node; + } + + for (t = registered_builtin_types; t; t = TREE_CHAIN (t)) + if (TYPE_MODE (TREE_VALUE (t)) == mode) + return TREE_VALUE (t); + + return NULL_TREE; +} + +static int +lto_global_bindings_p (void) +{ + return cfun == NULL; +} + +static void +lto_set_decl_assembler_name (tree decl) +{ + /* This is almost the same as lhd_set_decl_assembler_name, except that + we need to uniquify file-scope names, even if they are not + TREE_PUBLIC, to avoid conflicts between individual files. */ + tree id; + + if (TREE_PUBLIC (decl)) + id = targetm.mangle_decl_assembler_name (decl, DECL_NAME (decl)); + else + { + const char *name = IDENTIFIER_POINTER (DECL_NAME (decl)); + char *label; + + ASM_FORMAT_PRIVATE_NAME (label, name, DECL_UID (decl)); + id = get_identifier (label); + } + + SET_DECL_ASSEMBLER_NAME (decl, id); +} + +static tree +lto_pushdecl (tree t ATTRIBUTE_UNUSED) +{ + /* Do nothing, since we get all information from DWARF and LTO + sections. */ + return NULL_TREE; +} + +static tree +lto_getdecls (void) +{ + return registered_builtin_fndecls; +} + +static void +lto_write_globals (void) +{ + tree *vec = VEC_address (tree, lto_global_var_decls); + int len = VEC_length (tree, lto_global_var_decls); + wrapup_global_declarations (vec, len); + emit_debug_global_declarations (vec, len); + VEC_free (tree, gc, lto_global_var_decls); +} + +static tree +lto_builtin_function (tree decl) +{ + /* Record it. */ + TREE_CHAIN (decl) = registered_builtin_fndecls; + registered_builtin_fndecls = decl; + + return decl; +} + +static void +lto_register_builtin_type (tree type, const char *name) +{ + tree decl; + + decl = build_decl (UNKNOWN_LOCATION, TYPE_DECL, get_identifier (name), type); + DECL_ARTIFICIAL (decl) = 1; + if (!TYPE_NAME (type)) + TYPE_NAME (type) = decl; + + registered_builtin_types = tree_cons (0, type, registered_builtin_types); +} + +/* Build nodes that would have be created by the C front-end; necessary + for including builtin-types.def and ultimately builtins.def. */ + +static void +lto_build_c_type_nodes (void) +{ + gcc_assert (void_type_node); + + void_list_node = build_tree_list (NULL_TREE, void_type_node); + string_type_node = build_pointer_type (char_type_node); + const_string_type_node + = build_pointer_type (build_qualified_type (char_type_node, TYPE_QUAL_CONST)); + + if (strcmp (SIZE_TYPE, "unsigned int") == 0) + { + intmax_type_node = integer_type_node; + uintmax_type_node = unsigned_type_node; + signed_size_type_node = integer_type_node; + } + else if (strcmp (SIZE_TYPE, "long unsigned int") == 0) + { + intmax_type_node = long_integer_type_node; + uintmax_type_node = long_unsigned_type_node; + signed_size_type_node = long_integer_type_node; + } + else + gcc_unreachable (); + + wint_type_node = unsigned_type_node; + pid_type_node = integer_type_node; +} + + +/* Perform LTO-specific initialization. */ + +static bool +lto_init (void) +{ + /* We need to generate LTO if running in WPA mode. */ + flag_generate_lto = flag_wpa; + + /* Initialize libcpp line maps for gcc_assert to work. */ + linemap_add (line_table, LC_RENAME, 0, NULL, 0); + linemap_add (line_table, LC_RENAME, 0, NULL, 0); + + /* Create the basic integer types. */ + build_common_tree_nodes (flag_signed_char, /*signed_sizetype=*/false); + + /* Share char_type_node with whatever would be the default for the target. + char_type_node will be used for internal types such as + va_list_type_node but will not be present in the lto stream. */ + /* ??? This breaks the more common case of consistent but non-standard + setting of flag_signed_char, so share according to flag_signed_char. + See PR42528. */ + char_type_node + = flag_signed_char ? signed_char_type_node : unsigned_char_type_node; + + /* Tell the middle end what type to use for the size of objects. */ + if (strcmp (SIZE_TYPE, "unsigned int") == 0) + { + set_sizetype (unsigned_type_node); + size_type_node = unsigned_type_node; + } + else if (strcmp (SIZE_TYPE, "long unsigned int") == 0) + { + set_sizetype (long_unsigned_type_node); + size_type_node = long_unsigned_type_node; + } + else + gcc_unreachable (); + + /* The global tree for the main identifier is filled in by + language-specific front-end initialization that is not run in the + LTO back-end. It appears that all languages that perform such + initialization currently do so in the same way, so we do it here. */ + if (main_identifier_node == NULL_TREE) + main_identifier_node = get_identifier ("main"); + + /* In the C++ front-end, fileptr_type_node is defined as a variant + copy of of ptr_type_node, rather than ptr_node itself. The + distinction should only be relevant to the front-end, so we + always use the C definition here in lto1. */ + gcc_assert (fileptr_type_node == ptr_type_node); + + ptrdiff_type_node = integer_type_node; + + /* Create other basic types. */ + build_common_tree_nodes_2 (/*short_double=*/false); + lto_build_c_type_nodes (); + gcc_assert (va_list_type_node); + + if (TREE_CODE (va_list_type_node) == ARRAY_TYPE) + { + tree x = build_pointer_type (TREE_TYPE (va_list_type_node)); + lto_define_builtins (x, x); + } + else + { + lto_define_builtins (va_list_type_node, + build_reference_type (va_list_type_node)); + } + + targetm.init_builtins (); + build_common_builtin_nodes (); + + /* Initialize LTO-specific data structures. */ + lto_global_var_decls = VEC_alloc (tree, gc, 256); + in_lto_p = true; + + return true; +} + +/* Initialize tree structures required by the LTO front end. */ + +static void lto_init_ts (void) +{ + tree_contains_struct[NAMESPACE_DECL][TS_DECL_MINIMAL] = 1; +} + +#undef LANG_HOOKS_NAME +#define LANG_HOOKS_NAME "GNU GIMPLE" +#undef LANG_HOOKS_INIT_OPTIONS +#define LANG_HOOKS_INIT_OPTIONS lto_init_options +#undef LANG_HOOKS_HANDLE_OPTION +#define LANG_HOOKS_HANDLE_OPTION lto_handle_option +#undef LANG_HOOKS_POST_OPTIONS +#define LANG_HOOKS_POST_OPTIONS lto_post_options +#undef LANG_HOOKS_GET_ALIAS_SET +#define LANG_HOOKS_GET_ALIAS_SET gimple_get_alias_set +#undef LANG_HOOKS_TYPE_FOR_MODE +#define LANG_HOOKS_TYPE_FOR_MODE lto_type_for_mode +#undef LANG_HOOKS_TYPE_FOR_SIZE +#define LANG_HOOKS_TYPE_FOR_SIZE lto_type_for_size +#undef LANG_HOOKS_SET_DECL_ASSEMBLER_NAME +#define LANG_HOOKS_SET_DECL_ASSEMBLER_NAME lto_set_decl_assembler_name +#undef LANG_HOOKS_GLOBAL_BINDINGS_P +#define LANG_HOOKS_GLOBAL_BINDINGS_P lto_global_bindings_p +#undef LANG_HOOKS_PUSHDECL +#define LANG_HOOKS_PUSHDECL lto_pushdecl +#undef LANG_HOOKS_GETDECLS +#define LANG_HOOKS_GETDECLS lto_getdecls +#undef LANG_HOOKS_WRITE_GLOBALS +#define LANG_HOOKS_WRITE_GLOBALS lto_write_globals +#undef LANG_HOOKS_REGISTER_BUILTIN_TYPE +#define LANG_HOOKS_REGISTER_BUILTIN_TYPE lto_register_builtin_type +#undef LANG_HOOKS_BUILTIN_FUNCTION +#define LANG_HOOKS_BUILTIN_FUNCTION lto_builtin_function +#undef LANG_HOOKS_INIT +#define LANG_HOOKS_INIT lto_init +#undef LANG_HOOKS_PARSE_FILE +#define LANG_HOOKS_PARSE_FILE lto_main +#undef LANG_HOOKS_CALLGRAPH_EXPAND_FUNCTION +#define LANG_HOOKS_CALLGRAPH_EXPAND_FUNCTION tree_rest_of_compilation +#undef LANG_HOOKS_REDUCE_BIT_FIELD_OPERATIONS +#define LANG_HOOKS_REDUCE_BIT_FIELD_OPERATIONS true +#undef LANG_HOOKS_TYPES_COMPATIBLE_P +#define LANG_HOOKS_TYPES_COMPATIBLE_P NULL + +/* Attribute hooks. */ +#undef LANG_HOOKS_COMMON_ATTRIBUTE_TABLE +#define LANG_HOOKS_COMMON_ATTRIBUTE_TABLE lto_attribute_table +#undef LANG_HOOKS_FORMAT_ATTRIBUTE_TABLE +#define LANG_HOOKS_FORMAT_ATTRIBUTE_TABLE lto_format_attribute_table + +#undef LANG_HOOKS_BEGIN_SECTION +#define LANG_HOOKS_BEGIN_SECTION lto_obj_begin_section +#undef LANG_HOOKS_APPEND_DATA +#define LANG_HOOKS_APPEND_DATA lto_obj_append_data +#undef LANG_HOOKS_END_SECTION +#define LANG_HOOKS_END_SECTION lto_obj_end_section + +#undef LANG_HOOKS_INIT_TS +#define LANG_HOOKS_INIT_TS lto_init_ts + +struct lang_hooks lang_hooks = LANG_HOOKS_INITIALIZER; + +/* Language hooks that are not part of lang_hooks. */ + +tree +convert (tree type ATTRIBUTE_UNUSED, tree expr ATTRIBUTE_UNUSED) +{ + gcc_unreachable (); +} + +/* Tree walking support. */ + +static enum lto_tree_node_structure_enum +lto_tree_node_structure (union lang_tree_node *t ATTRIBUTE_UNUSED) +{ + return TS_LTO_GENERIC; +} + +#include "ggc.h" +#include "gtype-lto.h" +#include "gt-lto-lto-lang.h"
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-lang.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.h (revision 288) @@ -0,0 +1,406 @@ +/* LTO routines for COFF object files. + Copyright 2009, 2010 Free Software Foundation, Inc. + Contributed by Dave Korn. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#ifndef LTO_COFF_H +#define LTO_COFF_H + +/* Rather than implementing a libcoff to match libelf, or attempting to + integrate libbfd into GCC, this file is a self-contained (and very + minimal) COFF format object file reader/writer. The generated files + will contain a COFF header, a number of COFF section headers, the + section data itself, and a trailing string table for section names. */ + +/* Alignment of sections in a COFF object file. + + The LTO writer uses zlib compression on the data that it streams into + LTO sections in the output object file. Because these streams don't + have any embedded size information, the section in the object file must + be exactly sized to the data emitted; any trailing padding bytes will + be interpreted as partial and/or corrupt compressed data. + + This is easy enough to do on COFF targets (with binutils 2.20.1 or + above) because we can specify 1-byte alignment for the LTO sections. + They are then emitted precisely-sized and byte-packed into the object + and the reader is happy when it parses them later. This is currently + implemented in the x86/windows backed in i386_pe_asm_named_section() + in config/i386/winnt.c by detecting the LTO section name prefix, + + That would be sufficient, but for one thing. At the start of the LTO + data is a header struct with (currently) a couple of version numbers and + some type info; see struct lto_header in lto-streamer.h. If the sections + are byte-packed, this header will not necessarily be correctly-aligned + when it is read back into memory. + + On x86 targets, which are currently the only LTO-COFF targets, misaligned + memory accesses aren't problematic (okay, inefficient, but not worth + worrying about two half-word memory reads per section in the context of + everything else the compiler has to do at the time!), but RISC targets may + fail on trying to access the header struct. In this case, it will be + necessary to enable (preferably in a target-dependent fashion, but a few + bytes of padding are hardly an important issue if it comes down to it) the + COFF_ALIGNMENT macros below. + + As currently implemented, this will emit padding to the necessary number + of bytes after each LTO section. These bytes will constitute 'gaps' in + the object file structure, as they won't be covered by any section header. + This hasn't yet been tested, because no such RISC LTO-COFF target yet + exists. If it causes problems further down the toolchain, it will be + necessary to adapt the code to emit additional section headers for these + padding bytes, but the odds are that it will "just work". + + */ + +#if 0 +#define COFF_ALIGNMENT (4) +#define COFF_ALIGNMENTM1 (COFF_ALIGNMENT - 1) +#define COFF_ALIGN(x) (((x) + COFF_ALIGNMENTM1) & ~COFF_ALIGNMENTM1) +#else +#define COFF_ALIGNMENT (1) +#define COFF_ALIGN(x) (x) +#endif + +/* COFF header machine codes. */ + +#define IMAGE_FILE_MACHINE_I386 (0x014c) + +/* Known header magics for validation, as an array initialiser. */ + +#define COFF_KNOWN_MACHINES \ + { IMAGE_FILE_MACHINE_I386/*, ... add more here when working. */ } + +/* COFF object file header, section and symbol flags and types. These are + currently specific to PE-COFF, which is the only LTO-COFF format at the + time of writing. Maintainers adding support for new COFF formats will + need to make these into target macros of some kind. */ + +/* COFF header characteristics. */ + +#define IMAGE_FILE_EXECUTABLE_IMAGE (1 << 1) +#define IMAGE_FILE_32BIT_MACHINE (1 << 8) +#define IMAGE_FILE_SYSTEM (1 << 12) +#define IMAGE_FILE_DLL (1 << 13) + +/* Desired characteristics (for validation). */ + +#define COFF_CHARACTERISTICS \ + (IMAGE_FILE_32BIT_MACHINE) + +/* Unwanted characteristics (for validation). */ + +#define COFF_NOT_CHARACTERISTICS \ + (IMAGE_FILE_EXECUTABLE_IMAGE | IMAGE_FILE_SYSTEM | IMAGE_FILE_DLL) + +/* Section flags. LTO emits byte-aligned read-only loadable data sections. */ + +#define IMAGE_SCN_CNT_INITIALIZED_DATA (1 << 6) +#define IMAGE_SCN_CNT_UNINITIALIZED_DATA (1 << 7) +#define IMAGE_SCN_ALIGN_1BYTES (0x1 << 20) +#define IMAGE_SCN_MEM_DISCARDABLE (1 << 25) +#define IMAGE_SCN_MEM_SHARED (1 << 28) +#define IMAGE_SCN_MEM_READ (1 << 30) + +#define COFF_SECTION_CHARACTERISTICS \ + (IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_ALIGN_1BYTES | \ + IMAGE_SCN_MEM_DISCARDABLE | IMAGE_SCN_MEM_SHARED | IMAGE_SCN_MEM_READ) + +/* Symbol-related constants. */ + +#define IMAGE_SYM_DEBUG (-2) +#define IMAGE_SYM_TYPE_NULL (0) +#define IMAGE_SYM_DTYPE_NULL (0) +#define IMAGE_SYM_CLASS_STATIC (3) +#define IMAGE_SYM_CLASS_FILE (103) + +#define IMAGE_SYM_TYPE \ + ((IMAGE_SYM_DTYPE_NULL << 4) | IMAGE_SYM_TYPE_NULL) + +/* Size of a COFF symbol in bytes. */ + +#define COFF_SYMBOL_SIZE (18) + +/* On-disk file structures. */ + +struct Coff_header +{ + unsigned char Machine[2]; + unsigned char NumberOfSections[2]; + unsigned char TimeDateStamp[4]; + unsigned char PointerToSymbolTable[4]; + unsigned char NumberOfSymbols[4]; + unsigned char SizeOfOptionalHeader[2]; + unsigned char Characteristics[2]; +}; +typedef struct Coff_header Coff_header; + +struct Coff_section +{ + unsigned char Name[8]; + unsigned char VirtualSize[4]; + unsigned char VirtualAddress[4]; + unsigned char SizeOfRawData[4]; + unsigned char PointerToRawData[4]; + unsigned char PointerToRelocations[4]; + unsigned char PointerToLinenumbers[4]; + unsigned char NumberOfRelocations[2]; + unsigned char NumberOfLinenumbers[2]; + unsigned char Characteristics[4]; +}; +typedef struct Coff_section Coff_section; + +struct Coff_symbol +{ + unsigned char Name[8]; + unsigned char Value[4]; + unsigned char SectionNumber[2]; + unsigned char Type[2]; + unsigned char StorageClass[1]; + unsigned char NumberOfAuxSymbols[1]; +}; +typedef struct Coff_symbol Coff_symbol; + +struct Coff_aux_sym_file +{ + unsigned char FileName[18]; +}; +typedef struct Coff_aux_sym_file Coff_aux_sym_file; + +struct Coff_aux_sym_section +{ + unsigned char Length[4]; + unsigned char NumberOfRelocations[2]; + unsigned char NumberOfLineNumbers[2]; + unsigned char Checksum[4]; + unsigned char Number[2]; + unsigned char Selection[1]; + unsigned char Unused[3]; +}; +typedef struct Coff_aux_sym_section Coff_aux_sym_section; + +/* Accessor macros for the above structures. */ + +#define COFF_GET(struc,memb) \ + ((COFFENDIAN ? get_be : get_le) (&(struc)->memb[0], sizeof ((struc)->memb))) + +#define COFF_PUT(struc,memb,val) \ + ((COFFENDIAN ? put_be : put_le) (&(struc)->memb[0], sizeof ((struc)->memb), val)) + +#define COFF_PUT_NDXSZ(struc,memb,val,ndx,sz) \ + ((COFFENDIAN ? put_be : put_le) (&(struc)->memb[ndx], sz, val)) + +/* In-memory file structures. */ + +/* Forward declared structs. */ + +struct lto_coff_data; +struct lto_coff_section; +struct lto_coff_file; + +/* Section data in output files is made of these. */ + +struct lto_coff_data +{ + /* Pointer to data block. */ + void *d_buf; + + /* Size of data block. */ + ssize_t d_size; + + /* Next data block for this section. */ + struct lto_coff_data *next; +}; +typedef struct lto_coff_data lto_coff_data; + +/* This struct tracks the data for a section. */ + +struct lto_coff_section +{ + /* Singly-linked list of section's data blocks. */ + lto_coff_data *data_chain; + + /* Offset in string table of name. */ + size_t strtab_offs; + + /* Section type: 0 = real, 1 = dummy. */ + size_t type; + + /* Section name. */ + const char *name; + +#if COFF_ALIGNMENT > 1 + /* Number of trailing padding bytes needed. */ + ssize_t pad_needed; +#endif + + /* Raw section header data. */ + Coff_section coffsec; + + /* Next section for this file. */ + struct lto_coff_section *next; +}; +typedef struct lto_coff_section lto_coff_section; + +/* A COFF file. */ + +struct lto_coff_file +{ + /* The base information. */ + lto_file base; + + /* Common file members: */ + + /* The system file descriptor for the file. */ + int fd; + + /* The file's overall header. */ + Coff_header coffhdr; + + /* All sections in a singly-linked list. */ + lto_coff_section *section_chain; + + /* Readable file members: */ + + /* File total size. */ + off_t file_size; + + /* String table file offset, relative to base.offset. */ + off_t strtab_offs; + + /* Writable file members: */ + + /* The currently active section. */ + lto_coff_section *scn; + + /* The output stream for section header names. */ + struct lto_output_stream *shstrtab_stream; + + /* Linked list of data which must be freed *after* the file has been + closed. This is an annoying limitation of libelf. Which has been + faithfully reproduced here. */ + struct lto_char_ptr_base *data; +}; +typedef struct lto_coff_file lto_coff_file; + +/* Data hunk iterator. */ + +#define COFF_FOR_ALL_DATA(sec,var) \ + for (var = sec->data_chain; var; var = var->next) + +/* Section list iterator. */ + +#define COFF_FOR_ALL_SECTIONS(file,var) \ + for (var = file->section_chain; var; var = var->next) + +/* Very simple endian-ness layer. */ + +#ifndef COFFENDIAN +#define COFFENDIAN (BYTES_BIG_ENDIAN) +#endif + +static inline unsigned int +get_2_le (const unsigned char *ptr) +{ + return ptr[0] | (ptr[1] << 8); +} + +static inline unsigned int +get_4_le (const unsigned char *ptr) +{ + return ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24); +} + +static inline unsigned int +get_2_be (const unsigned char *ptr) +{ + return ptr[1] | (ptr[0] << 8); +} + +static inline unsigned int +get_4_be (const unsigned char *ptr) +{ + return ptr[3] | (ptr[2] << 8) | (ptr[1] << 16) | (ptr[0] << 24); +} + +static inline unsigned int +get_be (const unsigned char *ptr, size_t size) +{ + gcc_assert (size == 4 || size == 2); + return (size == 2) ? get_2_be (ptr) : get_4_be (ptr); +} + +static inline unsigned int +get_le (const unsigned char *ptr, size_t size) +{ + gcc_assert (size == 4 || size == 2); + return (size == 2) ? get_2_le (ptr) : get_4_le (ptr); +} + +static inline void +put_2_le (unsigned char *ptr, unsigned int data) +{ + ptr[0] = data & 0xff; + ptr[1] = (data >> 8) & 0xff; +} + +static inline void +put_4_le (unsigned char *ptr, unsigned int data) +{ + ptr[0] = data & 0xff; + ptr[1] = (data >> 8) & 0xff; + ptr[2] = (data >> 16) & 0xff; + ptr[3] = (data >> 24) & 0xff; +} + +static inline void +put_2_be (unsigned char *ptr, unsigned int data) +{ + ptr[1] = data & 0xff; + ptr[0] = (data >> 8) & 0xff; +} + +static inline void +put_4_be (unsigned char *ptr, unsigned int data) +{ + ptr[3] = data & 0xff; + ptr[2] = (data >> 8) & 0xff; + ptr[1] = (data >> 16) & 0xff; + ptr[0] = (data >> 24) & 0xff; +} + +static inline void +put_le (unsigned char *ptr, size_t size, unsigned int data) +{ + gcc_assert (size == 4 || size == 2); + (void) (size == 2 ? put_2_le : put_4_le) (ptr, data); +} + +static inline void +put_be (unsigned char *ptr, size_t size, unsigned int data) +{ + gcc_assert (size == 4 || size == 2); + (void) (size == 2 ? put_2_be : put_4_be) (ptr, data); +} + +/* We use this for putting the string table size. */ + +#define COFF_PUT4(ptr, data) \ + ((COFFENDIAN ? put_4_be : put_4_le) (ptr, data)) + + +#endif /* LTO_COFF_H */
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-coff.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/Make-lang.in =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/Make-lang.in (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/Make-lang.in (revision 288) @@ -0,0 +1,96 @@ +# Top level -*- makefile -*- fragment for LTO +# Copyright (C) 2009, 2010 +# Free Software Foundation, Inc. + +#This file is part of GCC. + +#GCC is free software; you can redistribute it and/or modify +#it under the terms of the GNU General Public License as published by +#the Free Software Foundation; either version 3, or (at your option) +#any later version. + +#GCC is distributed in the hope that it will be useful, +#but WITHOUT ANY WARRANTY; without even the implied warranty of +#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +#GNU General Public License for more details. + +# You should have received a copy of the GNU General Public License +# along with GCC; see the file COPYING3. If not see +# . + +# Variables + +# The name of the LTO compiler. +LTO_EXE = lto1$(exeext) +# The LTO-specific object files inclued in $(LTO_EXE). +LTO_OBJS = lto/lto-lang.o lto/lto.o lto/$(LTO_BINARY_READER).o attribs.o +LTO_H = lto/lto.h $(HASHTAB_H) +LINKER_PLUGIN_API_H = $(srcdir)/../include/plugin-api.h +LTO_TREE_H = lto/lto-tree.h $(LINKER_PLUGIN_API_H) + + +# Rules + +# These hooks are used by the main GCC Makefile. Consult that +# Makefile for documentation. +lto.all.cross: $(LTO_EXE) +lto.start.encap: $(LTO_EXE) +lto.rest.encap: +lto.tags: +lto.install-common: +lto.install-man: +lto.install-info: +lto.dvi: +lto.pdf: +lto.install-pdf: +lto.html: +lto.install-html: +lto.uninstall: +lto.info: +lto.man: +lto.srcextra: +lto.srcman: +lto.srcinfo: +lto.install-plugin: + +lto.mostlyclean: + rm -f $(LTO_OBJS) $(LTO_EXE) + +lto.clean: +lto.distclean: +lto.maintainer-clean: +lto.stage1: +lto.stage2: +lto.stage3: +lto.stage4: +lto.stageprofile: +lto.stagefeedback: + +# LTO rules. + +# Use strict warnings for this front end. +lto-warn = $(STRICT_WARN) + +$(LTO_EXE): $(LTO_OBJS) $(BACKEND) $(LIBDEPS) + $(LINKER) $(ALL_LINKERFLAGS) $(LDFLAGS) -o $@ \ + $(LTO_OBJS) $(BACKEND) $(BACKENDLIBS) $(LIBS) $(LTO_USE_LIBELF) + +# Dependencies +lto/lto-lang.o: lto/lto-lang.c $(CONFIG_H) coretypes.h debug.h \ + flags.h $(GGC_H) langhooks.h $(LANGHOOKS_DEF_H) $(SYSTEM_H) \ + $(TARGET_H) $(LTO_H) $(GIMPLE_H) gtype-lto.h gt-lto-lto-lang.h \ + $(EXPR_H) +lto/lto.o: lto/lto.c $(CONFIG_H) $(SYSTEM_H) coretypes.h opts.h \ + toplev.h $(TREE_H) $(DIAGNOSTIC_H) $(TM_H) $(LIBIBERTY_H) \ + $(CGRAPH_H) $(GGC_H) tree-ssa-operands.h $(TREE_PASS_H) \ + langhooks.h vec.h $(BITMAP_H) pointer-set.h $(IPA_PROP_H) \ + $(COMMON_H) $(TIMEVAR_H) $(GIMPLE_H) $(LTO_H) $(LTO_TREE_H) \ + $(LTO_TAGS_H) $(LTO_STREAMER_H) +lto/lto-elf.o: lto/lto-elf.c $(CONFIG_H) coretypes.h $(SYSTEM_H) \ + toplev.h $(LTO_H) $(TM_H) $(LIBIBERTY_H) $(GGC_H) $(LTO_STREAMER_H) +lto/lto-coff.o: lto/lto-coff.c $(CONFIG_H) coretypes.h $(SYSTEM_H) \ + toplev.h $(LTO_H) $(TM_H) $(LIBIBERTY_H) $(GGC_H) $(LTO_STREAMER_H) \ + lto/lto-coff.h +lto/lto-macho.o: lto/lto-macho.c $(CONFIG_H) coretypes.h $(SYSTEM_H) \ + toplev.h $(LTO_H) $(TM_H) $(LIBIBERTY_H) $(GGC_H) $(LTO_STREAMER_H) \ + lto/lto-macho.h lto/lto-endian.h Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.c =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.c (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.c (revision 288) @@ -0,0 +1,969 @@ +/* LTO routines for Mach-O object files. + Copyright 2010 Free Software Foundation, Inc. + Contributed by Steven Bosscher. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#include "config.h" +#include "system.h" +#include "coretypes.h" +#include "toplev.h" +#include "lto.h" +#include "tm.h" +#include "libiberty.h" +#include "lto-streamer.h" +#include "lto/lto-endian.h" +#include "lto/lto-macho.h" + +/* Rather than implementing a libmacho to match libelf, or attempting to + integrate libbfd into GCC, this file is a self-contained (and very + minimal) Mach-O format object file reader/writer. The generated files + will contain a Mach-O header, a number of Mach-O load commands an + section headers, the section data itself, and a trailing string table + for section names. */ + +/* This needs to be kept in sync with darwin.c. Better yet, lto-macho.c + and lto-macho.h should be moved to config/, and likewise for lto-coff.* + and lto-elf.*. */ + +/* Segment name for LTO sections. */ +#define LTO_SEGMENT_NAME "__GNU_LTO" + +/* Section name for LTO section names section. */ +#define LTO_NAMES_SECTION "__section_names" + +/* Handle opening elf files on hosts, such as Windows, that may use + text file handling that will break binary access. */ +#ifndef O_BINARY +# define O_BINARY 0 +#endif + +/* Cached object file header. We use a header_64 for this, since all + the fields we need are in there, in the same position as header_32. */ +mach_o_header_64 cached_mach_o_header; +uint32_t cached_mach_o_magic; + +/* The current output file. */ +static lto_file *current_out_file; + + +/* Is this a 32-bits or 64-bits Mach-O object file? */ +static int +mach_o_word_size (void) +{ + gcc_assert (cached_mach_o_magic != 0); + return (cached_mach_o_magic == MACH_O_MH_MAGIC_64 + || cached_mach_o_magic == MACH_O_MH_CIGAM_64) ? 64 : 32; +} + +/* Sets the current output file to FILE. Returns the old output file or + NULL. */ + +lto_file * +lto_set_current_out_file (lto_file *file) +{ + lto_file *old_file = current_out_file; + current_out_file = file; + return old_file; +} + + +/* Returns the current output file. */ + +lto_file * +lto_get_current_out_file (void) +{ + return current_out_file; +} + +/* Mach-O section structure constructor. */ + +static lto_mach_o_section +mach_o_new_section (lto_mach_o_file *mach_o_file, const char *name) +{ + lto_mach_o_section ptr; + + /* FIXME We could allocate these things on an obstack. */ + ptr = XCNEW (struct lto_mach_o_section_d); + if (name) + { + if (strncmp (name, LTO_SECTION_NAME_PREFIX, + strlen(LTO_SECTION_NAME_PREFIX)) != 0) + sorry ("not implemented: Mach-O writer for non-LTO sections"); + ptr->name = xstrdup (name); + } + + VEC_safe_push (lto_mach_o_section, heap, mach_o_file->section_vec, ptr); + + return ptr; +} + +/* Mach-O section data block structure constructor. */ + +static lto_mach_o_data +mach_o_new_data (lto_mach_o_section sec) +{ + lto_mach_o_data ptr, *chain_ptr_ptr; + + /* FIXME We could allocate these things on an obstack. */ + ptr = XCNEW (struct lto_mach_o_data_d); + + chain_ptr_ptr = &sec->data_chain; + while (*chain_ptr_ptr) + chain_ptr_ptr = &(*chain_ptr_ptr)->next; + *chain_ptr_ptr = ptr; + + return ptr; +} + +/* Initialize FILE, an LTO file object for FILENAME. Offset is the + offset into FILE where the object is located (e.g. in an archive). */ + +static void +lto_file_init (lto_file *file, const char *filename, off_t offset) +{ + file->filename = filename; + file->offset = offset; +} + +/* Returns a hash code for P. */ + +static hashval_t +hash_name (const void *p) +{ + const struct lto_section_slot *s = (const struct lto_section_slot *) p; + return (hashval_t) htab_hash_string (s->name); +} + +/* Returns nonzero if P1 and P2 are equal. */ + +static int +eq_name (const void *p1, const void *p2) +{ + const struct lto_section_slot *s1 = + (const struct lto_section_slot *) p1; + const struct lto_section_slot *s2 = + (const struct lto_section_slot *) p2; + + return strcmp (s1->name, s2->name) == 0; +} + +/* Build a hash table whose key is the section names and whose data is + the start and size of each section in the .o file. */ + +htab_t +lto_obj_build_section_table (lto_file *lto_file) +{ + lto_mach_o_file *mach_o_file = (lto_mach_o_file *)lto_file; + lto_mach_o_section sec; + htab_t section_hash_table; + off_t strtab_offs; + ssize_t strtab_size; + char *strtab = NULL; + int i; + + section_hash_table = htab_create (37, hash_name, eq_name, free); + + /* Seek the string table. */ + /* FIXME The segment name should be in darwin.h, but can we include it + here in this file? */ + for (i = 0; + VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + { + if (strncmp (sec->u.section.segname, "__GNU_LTO", 16) != 0) + continue; + if (strncmp (sec->u.section.sectname, "__section_names", 16) == 0) + break; + } + if (! sec) + { + error ("invalid Mach-O LTO object file: no __section_names section found"); + goto done; + } + mach_o_file->section_names_section = sec; + + if (mach_o_word_size () == 64) + { + strtab_offs = (off_t) get_uint32 (&sec->u.section_64.offset[0]); + strtab_size = (size_t) get_uint64 (&sec->u.section_64.size[0]); + } + else + { + strtab_offs = (off_t) get_uint32 (&sec->u.section_32.offset[0]); + strtab_size = (size_t) get_uint32 (&sec->u.section_32.size[0]); + } + + /* Seek to start of string table. */ + if (strtab_offs != lseek (mach_o_file->fd, + mach_o_file->base.offset + strtab_offs, + SEEK_SET)) + { + error ("altered or invalid Mach-O object file"); + goto done; + } + + strtab = XNEWVEC (char, strtab_size); + if (read (mach_o_file->fd, strtab, strtab_size) != strtab_size) + { + error ("invalid Mach-O LTO object file __section_names section"); + goto done; + } + + /* Scan sections looking at names. */ + for (i = 0; + VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + { + struct lto_section_slot s_slot; + void **slot; + char *new_name; + unsigned long stringoffset; + char name[17]; + + /* Ignore non-LTO sections. Also ignore the __section_names section + which does not need renaming. */ + if (strncmp (sec->u.section.segname, "__GNU_LTO", 16) != 0) + continue; + if (sec == mach_o_file->section_names_section) + continue; + + /* Try to extract the offset of the real name for this section from + __section_names. */ + memcpy (&name[0], sec->u.section.sectname, 16); + name[16] = '\0'; + if (name[0] != '_' || name[1] != '_' + || sscanf (&name[2], "%08lX", &stringoffset) != 1 + || strtab_size < (ssize_t) stringoffset) + { + error ("invalid Mach-O LTO section name string: %s", name); + continue; + } + + new_name = XNEWVEC (char, strlen (strtab + stringoffset) + 1); + strcpy (new_name, strtab + stringoffset); + s_slot.name = new_name; + slot = htab_find_slot (section_hash_table, &s_slot, INSERT); + if (*slot == NULL) + { + struct lto_section_slot *new_slot = XNEW (struct lto_section_slot); + + new_slot->name = new_name; + if (mach_o_word_size() == 64) + { + new_slot->start = + (intptr_t) get_uint32 (&sec->u.section_64.offset[0]); + new_slot->len = + (size_t) get_uint64 (&sec->u.section_64.size[0]); + } + else + { + new_slot->start = + (intptr_t) get_uint32 (&sec->u.section_32.offset[0]); + new_slot->len = + (size_t) get_uint32 (&sec->u.section_32.size[0]); + } + + *slot = new_slot; + } + else + { + error ("two or more sections for %s:", new_name); + goto done; + } + } + + done: + if (strtab) + free (strtab); + return section_hash_table; +} + + +/* Begin a new Mach-O section named NAME in the current output file. */ + +void +lto_obj_begin_section (const char *name) +{ + lto_mach_o_file *file; + + if (strncmp (name, LTO_SECTION_NAME_PREFIX, + strlen(LTO_SECTION_NAME_PREFIX)) != 0) + sorry ("not implemented: Mach-O writer for non-LTO sections"); + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_mach_o_file *) lto_get_current_out_file (), + gcc_assert (file && file->writable && !file->scn); + + /* Create a new section. */ + file->scn = mach_o_new_section (file, name); + if (!file->scn) + fatal_error ("could not create a new Mach-O section: %m"); +} + + +/* Append DATA of length LEN to the current output section. BASE is a pointer + to the output page containing DATA. It is freed once the output file has + been written. */ + +void +lto_obj_append_data (const void *data, size_t len, void *block) +{ + lto_mach_o_file *file; + lto_mach_o_data mach_o_data; + struct lto_char_ptr_base *base = (struct lto_char_ptr_base *) block; + + /* Grab the current output file and do some basic assertion checking. */ + file = (lto_mach_o_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + mach_o_data = mach_o_new_data (file->scn); + if (!mach_o_data) + fatal_error ("could not append data to Mach-O section: %m"); + + mach_o_data->d_buf = CONST_CAST (void *, data); + mach_o_data->d_size = len; + + /* Chain all data blocks (from all sections) on one singly-linked + list for freeing en masse after the file is closed. */ + base->ptr = (char *)file->data; + file->data = base; +} + + +/* End the current output section. This just does some assertion checking + and sets the current output file's scn member to NULL. */ + +void +lto_obj_end_section (void) +{ + lto_mach_o_file *file; + + /* Grab the current output file and validate some basic assertions. */ + file = (lto_mach_o_file *) lto_get_current_out_file (); + gcc_assert (file); + gcc_assert (file->scn); + + file->scn = NULL; +} + + +/* Read a Mach-O header from MACH_O_FILE and validate it. + The file descriptor in MACH_O_FILE points at the start of the file. + If cached_mach_o_header is uninitialized, caches the results. + On succes, returns true and moves file pointer to the start of the + load commands. On failure, returns false. */ + +static bool +validate_mach_o_header (lto_mach_o_file *mach_o_file) +{ + ssize_t i, n; + unsigned char magic[4]; + uint32_t cputype; + off_t startpos; + + /* Known header magics for validation, as an array. */ + static const unsigned int mach_o_known_formats[] = { + MACH_O_MH_MAGIC, + MACH_O_MH_CIGAM, + MACH_O_MH_MAGIC_64, + MACH_O_MH_CIGAM_64, + }; +#define MACH_O_NUM_KNOWN_FORMATS \ + ((ssize_t) ARRAY_SIZE (mach_o_known_formats)) + + startpos = lseek (mach_o_file->fd, 0, SEEK_CUR); + if (read (mach_o_file->fd, &magic, sizeof (magic)) != 4 + || lseek (mach_o_file->fd, -4, SEEK_CUR) != startpos) + { + error ("cannot read file %s", mach_o_file->base.filename); + return false; + } + + for (i = 0; i < MACH_O_NUM_KNOWN_FORMATS; ++i) + if (get_uint32 (&magic[0]) == mach_o_known_formats[i]) + break; + if (i == MACH_O_NUM_KNOWN_FORMATS) + goto not_for_target; + + /* Check the endian-ness. */ + if (BYTES_BIG_ENDIAN && magic[0] != 0xfe) + goto not_for_target; + + /* Set or check cached magic number. */ + if (cached_mach_o_magic == 0) + cached_mach_o_magic = get_uint32 (&magic[0]); + else if (cached_mach_o_magic != get_uint32 (&magic[0])) + goto not_for_target; + + n = mach_o_word_size () == 64 + ? sizeof (mach_o_header_64) : sizeof (mach_o_header_32); + if (read (mach_o_file->fd, &mach_o_file->u.header, n) != n) + goto not_for_target; + + /* Is this a supported CPU? */ + /* ??? Would be nice to validate the exact target architecture. */ + cputype = get_uint32 (&mach_o_file->u.header.cputype[0]); + if (cputype == MACH_O_CPU_TYPE_I386 + || cputype == MACH_O_CPU_TYPE_POWERPC) + { + if (mach_o_word_size () != 32) + goto not_for_target; + } + else if (cputype == MACH_O_CPU_TYPE_X86_64 + || cputype == MACH_O_CPU_TYPE_POWERPC_64) + { + if (mach_o_word_size () != 64) + goto not_for_target; + } + + /* Is this an MH_OBJECT file? */ + if (get_uint32 (&mach_o_file->u.header.filetype[0]) != MACH_O_MH_OBJECT) + error ("Mach-O file %s is not an MH_OBJECT file", + mach_o_file->base.filename); + + /* Save the header for future use. */ + memcpy (&cached_mach_o_header, &mach_o_file->u.header, + sizeof (cached_mach_o_header)); + + return true; + + not_for_target: + error ("file %s is not a Mach-O object file for target", + mach_o_file->base.filename); + return false; +} + + +/* Read a Mach-O LC_SEGMENT command (32 bits) from MACH_O_FILE and + validate it. + The file descriptor in MACH_O_FILE points at the start of the load + command. On sucess, returns true and advances the file pointer + past the end of the load command. On failure, returns false. */ + +static bool +validate_mach_o_segment_command_32 (lto_mach_o_file *mach_o_file) +{ + mach_o_segment_command_32 seg_cmd_32; + unsigned int i; + ssize_t n; + off_t startpos; + + /* Fields we're interested in. */ + uint32_t cmd; + uint32_t cmdsize; + uint32_t nsects; + + startpos = lseek (mach_o_file->fd, 0, SEEK_CUR); + + n = sizeof (mach_o_segment_command_32); + if (read (mach_o_file->fd, (void *) &seg_cmd_32, n) != n) + goto fail; + + cmd = get_uint32 (&seg_cmd_32.cmd[0]); + cmdsize = get_uint32 (&seg_cmd_32.cmdsize[0]); + nsects = get_uint32 (&seg_cmd_32.nsects[0]); + gcc_assert (cmd == MACH_O_LC_SEGMENT); + + /* Validate section table entries. */ + for (i = 0; i < nsects; i++) + { + mach_o_section_32 sec_32; + lto_mach_o_section ltosec; + + n = sizeof (mach_o_section_32); + if (read (mach_o_file->fd, &sec_32, n) != n) + goto fail; + + /* ??? Perform some checks. */ + + /* Looks ok, so record its details. We don't read the + string table or set up names yet; we'll do that when + we build the hash table. */ + ltosec = mach_o_new_section (mach_o_file, NULL); + memcpy (<osec->u.section_32, &sec_32, sizeof (sec_32)); + } + + if (lseek (mach_o_file->fd, 0, SEEK_CUR) != startpos + cmdsize) + goto fail; + + return true; + + fail: + error ("could not read LC_SEGMENT command in Mach-O file %s", + mach_o_file->base.filename); + return false; +} + + +/* Read a Mach-O LC_SEGMENT_64 command from MACH_O_FILE and validate it. + The file descriptor in MACH_O_FILE points at the start of the load + command. On sucess, returns true and advances the file pointer + past the end of the load command. On failure, returns false. */ + +static bool +validate_mach_o_segment_command_64 (lto_mach_o_file *mach_o_file) +{ + mach_o_segment_command_64 seg_cmd_64; + unsigned int i; + ssize_t n; + off_t startpos; + + /* Fields we're interested in. */ + uint32_t cmd; + uint32_t cmdsize; + uint32_t nsects; + + startpos = lseek (mach_o_file->fd, 0, SEEK_CUR); + + n = sizeof (mach_o_segment_command_64); + if (read (mach_o_file->fd, (void *) &seg_cmd_64, n) != n) + goto fail; + + cmd = get_uint32 (&seg_cmd_64.cmd[0]); + cmdsize = get_uint32 (&seg_cmd_64.cmdsize[0]); + nsects = get_uint32 (&seg_cmd_64.nsects[0]); + gcc_assert (cmd == MACH_O_LC_SEGMENT_64); + + /* Validate section table entries. */ + for (i = 0; i < nsects; i++) + { + mach_o_section_64 sec_64; + lto_mach_o_section ltosec; + + n = sizeof (mach_o_section_64); + if (read (mach_o_file->fd, &sec_64, n) != n) + goto fail; + + /* ??? Perform some checks. */ + + /* Looks ok, so record its details. We don't read the + string table or set up names yet; we'll do that when + we build the hash table. */ + ltosec = mach_o_new_section (mach_o_file, NULL); + memcpy (<osec->u.section_64, &sec_64, sizeof (sec_64)); + } + + if (lseek (mach_o_file->fd, 0, SEEK_CUR) != startpos + cmdsize) + goto fail; + + return true; + + fail: + error ("could not read LC_SEGMENT_64 command in Mach-O file %s", + mach_o_file->base.filename); + return false; +} + +/* Read a Mach-O load commands from MACH_O_FILE and validate it. + The file descriptor in MACH_O_FILE points at the start of the load + command. On sucess, returns true and advances the file pointer + past the end of the load command. On failure, returns false. */ + +static bool +validate_mach_o_load_command (lto_mach_o_file *mach_o_file) +{ + mach_o_load_command load_command; + uint32_t cmd; + uint32_t cmdsize; + ssize_t n; + + n = sizeof (load_command); + if (read (mach_o_file->fd, &load_command, n) != n) + { + error ("could not read load commands in Mach-O file %s", + mach_o_file->base.filename); + return false; + } + lseek (mach_o_file->fd, -1 * (off_t) sizeof (load_command), SEEK_CUR); + + cmd = get_uint32 (&load_command.cmd[0]); + cmdsize = get_uint32 (&load_command.cmdsize[0]); + switch (cmd) + { + case MACH_O_LC_SEGMENT: + return validate_mach_o_segment_command_32 (mach_o_file); + case MACH_O_LC_SEGMENT_64: + return validate_mach_o_segment_command_64 (mach_o_file); + + default: + /* Just skip over it. */ + lseek (mach_o_file->fd, cmdsize, SEEK_CUR); + return true; + } +} + +/* Validate's MACH_O_FILE's executable header and, if cached_mach_o_header is + uninitialized, caches the results. Also records the section header string + table's section index. Returns true on success, false on failure. */ + +static bool +validate_file (lto_mach_o_file *mach_o_file) +{ + uint32_t i, ncmds; + + /* Read and sanity check the raw header. */ + if (! validate_mach_o_header (mach_o_file)) + return false; + + ncmds = get_uint32 (&mach_o_file->u.header.ncmds[0]); + for (i = 0; i < ncmds; ++i) + if (! validate_mach_o_load_command (mach_o_file)) + return false; + + return true; +} + +/* Initialize MACH_O_FILE's executable header using cached data from previously + read files. */ + +static void +init_mach_o_header (lto_mach_o_file *mach_o_file) +{ + gcc_assert (cached_mach_o_magic != 0); + memcpy (&mach_o_file->u.header, + &cached_mach_o_header, + sizeof (mach_o_file->u.header)); + put_uint32 (&mach_o_file->u.header.ncmds[0], 0); + put_uint32 (&mach_o_file->u.header.sizeofcmds[0], 0); +} + +/* Open Mach-O file FILENAME. If WRITABLE is true, the file is opened for write + and, if necessary, created. Otherwise, the file is opened for reading. + Returns the opened file. */ + +lto_file * +lto_obj_file_open (const char *filename, bool writable) +{ + lto_mach_o_file *mach_o_file; + lto_file *result = NULL; + off_t offset; + const char *offset_p; + char *fname; + struct stat statbuf; + + offset_p = strchr (filename, '@'); + if (!offset_p) + { + fname = xstrdup (filename); + offset = 0; + } + else + { + /* The file started with '@' is a file containing command line + options. Stop if it doesn't exist. */ + if (offset_p == filename) + fatal_error ("command line option file '%s' does not exist", + filename); + + fname = (char *) xmalloc (offset_p - filename + 1); + memcpy (fname, filename, offset_p - filename); + fname[offset_p - filename] = '\0'; + offset_p += 3; /* skip the @0x */ + offset = lto_parse_hex (offset_p); + } + + /* Set up. */ + mach_o_file = XCNEW (lto_mach_o_file); + result = (lto_file *) mach_o_file; + lto_file_init (result, fname, offset); + mach_o_file->fd = -1; + mach_o_file->writable = writable; + + /* Open the file. */ + mach_o_file->fd = open (fname, + O_BINARY | (writable ? O_WRONLY | O_CREAT | O_TRUNC : O_RDONLY), 0666); + + if (mach_o_file->fd == -1) + { + error ("could not open file %s", fname); + goto fail; + } + + if (stat (fname, &statbuf) < 0) + { + error ("could not stat file %s", fname); + goto fail; + } + + mach_o_file->file_size = statbuf.st_size; + + /* If the object is in an archive, get it out. */ + if (offset != 0) + { + char ar_tail[12]; + int size; + + /* Surely not? */ + gcc_assert (!writable); + + /* Seek to offset, or error. */ + if (lseek (mach_o_file->fd, offset, SEEK_SET) != (ssize_t) offset) + { + error ("could not find archive member @0x%lx", (long) offset); + goto fail; + } + + /* Now seek back 12 chars and read the tail of the AR header to + find the length of the member file. */ + if (lseek (mach_o_file->fd, -12, SEEK_CUR) < 0 + || read (mach_o_file->fd, ar_tail, 12) != 12 + || lseek (mach_o_file->fd, 0, SEEK_CUR) != (ssize_t) offset + || ar_tail[10] != '`' || ar_tail[11] != '\n') + { + error ("could not find archive header @0x%lx", (long) offset); + goto fail; + } + + ar_tail[11] = 0; + if (sscanf (ar_tail, "%d", &size) != 1) + { + error ("invalid archive header @0x%lx", (long) offset); + goto fail; + } + mach_o_file->file_size = size; + } + + if (writable) + { + init_mach_o_header (mach_o_file); + } + else + if (! validate_file (mach_o_file)) + goto fail; + + return result; + + fail: + if (result) + lto_obj_file_close (result); + return NULL; +} + + +/* Write the data in MACH_O_FILE to a real Mach-O binary object. + We write a header, a segment load command, and section data. */ + +static bool +mach_o_write_object_file (lto_mach_o_file *mach_o_file) +{ + lto_mach_o_section sec, snsec; + lto_mach_o_data snsec_data; + ssize_t hdrsize, cmdsize, secsize; + size_t num_sections, snsec_size, total_sec_size; + unsigned int sec_offs, strtab_offs; + int i; + bool write_err = false; + + /* The number of sections we will write is the number of sections added by + the streamer, plus 1 for the section names section. */ + num_sections = VEC_length (lto_mach_o_section, mach_o_file->section_vec) + 1; + + /* Calculate the size of the basic data structures on disk. */ + if (mach_o_word_size () == 64) + { + hdrsize = sizeof (mach_o_header_64); + secsize = sizeof (mach_o_section_64); + cmdsize = sizeof (mach_o_segment_command_64) + num_sections * secsize; + } + else + { + hdrsize = sizeof (mach_o_header_32); + secsize = sizeof (mach_o_section_32); + cmdsize = sizeof (mach_o_segment_command_32) + num_sections * secsize; + } + + /* Allocate the section names section. */ + snsec_size = 0; + for (i = 0; + VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + snsec_size += strlen (sec->name) + 1; + snsec = mach_o_new_section (mach_o_file, NULL); + snsec->name = LTO_NAMES_SECTION; + snsec_data = mach_o_new_data (snsec); + snsec_data->d_buf = XCNEWVEC (char, snsec_size); + snsec_data->d_size = snsec_size; + + /* Position all the sections, and fill out their headers. */ + sec_offs = hdrsize + cmdsize; + strtab_offs = 0; + total_sec_size = 0; + for (i = 0; + VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + { + lto_mach_o_data data; + size_t data_size; + /* Put the section and segment names. Add the section name to the + section names section (unless, of course, this *is* the section + names section). */ + if (sec == snsec) + snprintf (sec->u.section.sectname, 16, "%s", LTO_NAMES_SECTION); + else + { + sprintf (sec->u.section.sectname, "__%08X", strtab_offs); + memcpy ((char *) snsec_data->d_buf + strtab_offs, sec->name, strlen (sec->name)); + } + memcpy (&sec->u.section.segname[0], + LTO_SEGMENT_NAME, strlen (LTO_SEGMENT_NAME)); + + /* Add layout and attributes. */ + for (data = sec->data_chain, data_size = 0; data; data = data->next) + data_size += data->d_size; + if (mach_o_word_size () == 64) + { + put_uint64 (&sec->u.section_64.addr[0], total_sec_size); + put_uint64 (&sec->u.section_64.size[0], data_size); + put_uint32 (&sec->u.section_64.offset[0], sec_offs); + put_uint32 (&sec->u.section_64.flags[0], MACH_O_S_ATTR_DEBUG); + } + else + { + put_uint32 (&sec->u.section_64.addr[0], total_sec_size); + put_uint32 (&sec->u.section_32.size[0], data_size); + put_uint32 (&sec->u.section_32.offset[0], sec_offs); + put_uint32 (&sec->u.section_32.flags[0], MACH_O_S_ATTR_DEBUG); + } + + sec_offs += data_size; + total_sec_size += data_size; + strtab_offs += strlen (sec->name) + 1; + } + + /* We can write the data now. As there's no way to indicate an error return + from this hook, error handling is limited to not wasting our time doing + any more writes in the event that any one fails. */ + + /* Write the header. */ + put_uint32 (&mach_o_file->u.header.ncmds[0], 1); + put_uint32 (&mach_o_file->u.header.sizeofcmds[0], cmdsize); + write_err = (write (mach_o_file->fd, + &mach_o_file->u.header, hdrsize) != hdrsize); + /* Write the segment load command. */ + if (mach_o_word_size () == 64) + { + mach_o_segment_command_64 lc; + ssize_t lc_size = sizeof (lc); + memset (&lc, 0, lc_size); + put_uint32 (&lc.cmd[0], MACH_O_LC_SEGMENT_64); + put_uint32 (&lc.cmdsize[0], cmdsize); + put_uint64 (&lc.fileoff[0], hdrsize + cmdsize); + put_uint64 (&lc.filesize[0], total_sec_size); + put_uint32 (&lc.nsects[0], num_sections); + write_err = (write (mach_o_file->fd, &lc, lc_size) != lc_size); + } + else + { + mach_o_segment_command_32 lc; + ssize_t lc_size = sizeof (lc); + memset (&lc, 0, lc_size); + put_uint32 (&lc.cmd[0], MACH_O_LC_SEGMENT); + put_uint32 (&lc.cmdsize[0], cmdsize); + put_uint32 (&lc.fileoff[0], hdrsize + cmdsize); + put_uint32 (&lc.filesize[0], total_sec_size); + put_uint32 (&lc.nsects[0], num_sections); + write_err = (write (mach_o_file->fd, &lc, lc_size) != lc_size); + } + for (i = 0; + !write_err + && VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + write_err = (write (mach_o_file->fd, + &sec->u.section, secsize) != secsize); + + gcc_assert (lseek (mach_o_file->fd, 0, SEEK_CUR) == hdrsize + cmdsize); + + /* Write the section data. */ + for (i = 0; + !write_err + && VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + { + lto_mach_o_data data; + + for (data = sec->data_chain; data; data = data->next) + { + if (!write_err) + write_err = (write (mach_o_file->fd, data->d_buf, data->d_size) + != data->d_size); + else + break; + } + } + + return !write_err; +} + +/* Close Mach-O file FILE and clean up any associated data structures. If FILE + was opened for writing, the file's Mach-O data is written at this time. Any + cached data buffers are freed. */ + +void +lto_obj_file_close (lto_file *file) +{ + lto_mach_o_file *mach_o_file = (lto_mach_o_file *) file; + struct lto_char_ptr_base *cur, *tmp; + lto_mach_o_section sec; + bool write_err = false; + int i; + + /* If this file is open for writing, write a Mach-O object file. */ + if (mach_o_file->writable) + { + if (! mach_o_write_object_file (mach_o_file)) + fatal_error ("cannot write Mach-O object file"); + } + + /* Close the file, we're done. */ + if (mach_o_file->fd != -1) + close (mach_o_file->fd); + + /* Free any data buffers. */ + cur = mach_o_file->data; + while (cur) + { + tmp = cur; + cur = (struct lto_char_ptr_base *) cur->ptr; + free (tmp); + } + + /* Free any sections and their data chains. */ + for (i = 0; + VEC_iterate (lto_mach_o_section, mach_o_file->section_vec, i, sec); + i++) + { + lto_mach_o_data curdata, nextdata; + curdata = sec->data_chain; + while (curdata) + { + nextdata = curdata->next; + free (curdata); + curdata = nextdata; + } + free (sec); + } + VEC_free (lto_mach_o_section, heap, mach_o_file->section_vec); + + free (file); + + /* If there was an error, mention it. */ + if (write_err) + error ("I/O error writing Mach-O output file"); +} +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.c =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.c (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.c (revision 288) @@ -0,0 +1,46 @@ +/* Common code for the plugin and lto1. + Copyright (C) 2009 Free Software Foundation, Inc. + Contributed by Rafael Avila de Espindola (espindola@google.com). + +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 3 of the License, or +(at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; if not, write to the Free Software +Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA +02110-1301, USA. */ + +#include "common.h" + +const char *lto_kind_str[5] __attribute__ ((visibility ("hidden"))) = +{ + "DEF", "WEAKDEF", "UNDEF", + "WEAKUNDEF", "COMMON" +}; + +const char *lto_visibility_str[4] __attribute__ ((visibility ("hidden"))) = +{ + "DEFAULT", "PROTECTED", + "INTERNAL", "HIDDEN" +}; + +const char *lto_resolution_str[9] __attribute__ ((visibility ("hidden"))) = +{ + "UNKNOWN", + "UNDEF", + "PREVAILING_DEF", + "PREVAILING_DEF_IRONLY", + "PREEMPTED_REG", + "PREEMPTED_IR", + "RESOLVED_IR", + "RESOLVED_EXEC", + "RESOLVED_DYN" +}; +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.c Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.h (revision 288) @@ -0,0 +1,251 @@ +/* LTO routines for Mach-O object files. + Copyright 2010 Free Software Foundation, Inc. + Contributed by Steven Bosscher. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + +#ifndef LTO_MACH_O_H +#define LTO_MACH_O_H + +/* On-disk file structures. */ + +/* Mach-O header (32 bits version). */ +struct mach_o_header_32 +{ + unsigned char magic[4]; /* Magic number. */ + unsigned char cputype[4]; /* CPU that this object is for. */ + unsigned char cpusubtype[4]; /* CPU subtype. */ + unsigned char filetype[4]; /* Type of file. */ + unsigned char ncmds[4]; /* Number of load commands. */ + unsigned char sizeofcmds[4]; /* Total size of load commands. */ + unsigned char flags[4]; /* Flags for special featues. */ +}; +typedef struct mach_o_header_32 mach_o_header_32; + +/* Mach-O header (64 bits version). */ +struct mach_o_header_64 +{ + unsigned char magic[4]; /* Magic number. */ + unsigned char cputype[4]; /* CPU that this object is for. */ + unsigned char cpusubtype[4]; /* CPU subtype. */ + unsigned char filetype[4]; /* Type of file. */ + unsigned char ncmds[4]; /* Number of load commands. */ + unsigned char sizeofcmds[4]; /* Total size of load commands. */ + unsigned char flags[4]; /* Flags for special featues. */ + unsigned char reserved[4]; /* Reserved. Duh. */ +}; +typedef struct mach_o_header_64 mach_o_header_64; + +/* Magic number. */ +#define MACH_O_MH_MAGIC 0xfeedface +#define MACH_O_MH_CIGAM 0xcefaedfe +#define MACH_O_MH_MAGIC_64 0xfeedfacf +#define MACH_O_MH_CIGAM_64 0xcffaedfe + +/* Supported CPU types. */ +#define MACH_O_CPU_TYPE_I386 7 +#define MACH_O_CPU_TYPE_X86_64 7 + 0x1000000 +#define MACH_O_CPU_TYPE_POWERPC 18 +#define MACH_O_CPU_TYPE_POWERPC_64 18 + 0x1000000 + +/* Supported file types. */ +#define MACH_O_MH_OBJECT 0x01 + +/* Mach-O load command data structure. */ +struct mach_o_load_command +{ + unsigned char cmd[4]; /* The type of load command. */ + unsigned char cmdsize[4]; /* Size in bytes of load command data structure. */ +}; +typedef struct mach_o_load_command mach_o_load_command; + +/* Supported load commands. We support only the segment load commands. */ +#define MACH_O_LC_SEGMENT 0x01 +#define MACH_O_LC_SEGMENT_64 0x19 + +/* LC_SEGMENT load command. */ +struct mach_o_segment_command_32 +{ + unsigned char cmd[4]; /* The type of load command (LC_SEGMENT). */ + unsigned char cmdsize[4]; /* Size in bytes of load command data structure. */ + unsigned char segname[16]; /* Name of this segment. */ + unsigned char vmaddr[4]; /* Virtual memory address of this segment. */ + unsigned char vmsize[4]; /* Size there, in bytes. */ + unsigned char fileoff[4]; /* Offset in bytes of the data to be mapped. */ + unsigned char filesize[4]; /* Size in bytes on disk. */ + unsigned char maxprot[4]; /* Maximum permitted vmem protection. */ + unsigned char initprot[4]; /* Initial vmem protection. */ + unsigned char nsects[4]; /* Number of sections in this segment. */ + unsigned char flags[4]; /* Flags that affect the loading. */ +}; +typedef struct mach_o_segment_command_32 mach_o_segment_command_32; + +/* LC_SEGMENT_64 load command. Only nsects matters for us, really. */ +struct mach_o_segment_command_64 +{ + unsigned char cmd[4]; /* The type of load command (LC_SEGMENT_64). */ + unsigned char cmdsize[4]; /* Size in bytes of load command data structure. */ + unsigned char segname[16]; /* Name of this segment. */ + unsigned char vmaddr[8]; /* Virtual memory address of this segment. */ + unsigned char vmsize[8]; /* Size there, in bytes. */ + unsigned char fileoff[8]; /* Offset in bytes of the data to be mapped. */ + unsigned char filesize[8]; /* Size in bytes on disk. */ + unsigned char maxprot[4]; /* Maximum permitted vmem protection. */ + unsigned char initprot[4]; /* Initial vmem protection. */ + unsigned char nsects[4]; /* Number of sections in this segment. */ + unsigned char flags[4]; /* Flags that affect the loading. */ +}; +typedef struct mach_o_segment_command_64 mach_o_segment_command_64; + +/* A Mach-O 32-bits section. */ +struct mach_o_section_32 +{ + unsigned char sectname[16]; /* Section name. */ + unsigned char segname[16]; /* Segment that the section belongs to. */ + unsigned char addr[4]; /* Address of this section in memory. */ + unsigned char size[4]; /* Size in bytes of this section. */ + unsigned char offset[4]; /* File offset of this section. */ + unsigned char align[4]; /* log2 of this section's alignment. */ + unsigned char reloff[4]; /* File offset of this section's relocs. */ + unsigned char nreloc[4]; /* Number of relocs for this section. */ + unsigned char flags[4]; /* Section flags/attributes. */ + unsigned char reserved1[4]; + unsigned char reserved2[4]; +}; +typedef struct mach_o_section_32 mach_o_section_32; + +/* A Mach-O 64-bits section. */ +struct mach_o_section_64 +{ + unsigned char sectname[16]; /* Section name. */ + unsigned char segname[16]; /* Segment that the section belongs to. */ + unsigned char addr[8]; /* Address of this section in memory. */ + unsigned char size[8]; /* Size in bytes of this section. */ + unsigned char offset[4]; /* File offset of this section. */ + unsigned char align[4]; /* log2 of this section's alignment. */ + unsigned char reloff[4]; /* File offset of this section's relocs. */ + unsigned char nreloc[4]; /* Number of relocs for this section. */ + unsigned char flags[4]; /* Section flags/attributes. */ + unsigned char reserved1[4]; + unsigned char reserved2[4]; + unsigned char reserved3[4]; +}; +typedef struct mach_o_section_64 mach_o_section_64; + +/* Flags for Mach-O sections. LTO sections are marked with S_ATTR_DEBUG + to instruct the linker to ignore the sections. */ +#define MACH_O_S_ATTR_DEBUG 0x02000000 + +/* In-memory file structures. */ + +/* Section data in output files is made of these. */ +struct lto_mach_o_data_d +{ + /* Pointer to data block. */ + void *d_buf; + + /* Size of data block. */ + ssize_t d_size; + + /* Next data block for this section. */ + struct lto_mach_o_data_d *next; +}; +typedef struct lto_mach_o_data_d *lto_mach_o_data; + +/* This struct tracks the data for a section. */ +struct lto_mach_o_section_d +{ + /* Singly-linked list of section's data blocks. */ + lto_mach_o_data data_chain; + + /* Offset in string table of the section name. */ + size_t strtab_offs; + + /* Section name. */ + const char *name; + + /* Number of trailing padding bytes needed. */ + ssize_t pad_needed; + + /* Raw section header data. */ + size_t section_size; + union { + struct { + char sectname[16]; + char segname[16]; + } section; + mach_o_section_32 section_32; + mach_o_section_64 section_64; + } u; + + /* Next section for this file. */ + struct lto_mach_o_section_d *next; +}; +typedef struct lto_mach_o_section_d *lto_mach_o_section; +DEF_VEC_P (lto_mach_o_section); +DEF_VEC_ALLOC_P (lto_mach_o_section, heap); + +/* A Mach-O file. */ +struct lto_mach_o_file_d +{ + /* The base information. */ + lto_file base; + + /* Common file members: */ + + /* The system file descriptor for the file. */ + int fd; + + /* The file's overall header. */ + union { + /* We make use here of the fact that section_32 and section_64 + have the same layout (except for section_64.reserved3). We + read the struct of proper size, but only address the first + member of this union. */ + mach_o_header_64 header; + mach_o_header_32 header_32; + mach_o_header_64 header_64; + } u; + + /* All sections in a varray. */ + VEC(lto_mach_o_section, heap) *section_vec; + + /* Readable file members: */ + + /* File total size. */ + off_t file_size; + + /* True if this file is open for writing. */ + bool writable; + + /* Section containing the __section_names section. */ + lto_mach_o_section section_names_section; + + /* Writable file members: */ + + /* The currently active section. */ + lto_mach_o_section scn; + + /* Linked list of data which must be freed *after* the file has been + closed. This is an annoying limitation of libelf. Which has been + faithfully reproduced here. */ + struct lto_char_ptr_base *data; +}; +typedef struct lto_mach_o_file_d lto_mach_o_file; + +#endif /* LTO_MACH_O_H */ +
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/lto-macho.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property Index: openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.h =================================================================== --- openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.h (nonexistent) +++ openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.h (revision 288) @@ -0,0 +1,34 @@ +/* Common code for the plugin and lto1. + Copyright (C) 2008 Free Software Foundation, Inc. + Contributed by Rafael Avila de Espindola (espindola@google.com). + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with GCC; see the file COPYING3. If not see +. */ + + + +static const char *lto_resolution_str[9] = +{ + "UNKNOWN", + "UNDEF", + "PREVAILING_DEF", + "PREVAILING_DEF_IRONLY", + "PREEMPTED_REG", + "PREEMPTED_IR", + "RESOLVED_IR", + "RESOLVED_EXEC", + "RESOLVED_DYN" +};
openrisc/trunk/gnu-src/gcc-4.5.1/gcc/lto/common.h Property changes : Added: svn:eol-style ## -0,0 +1 ## +native \ No newline at end of property Added: svn:keywords ## -0,0 +1 ## +Id \ No newline at end of property

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.