be_gas_flavour_t be_gas_flavour = GAS_FLAVOUR_ELF;
static be_gas_section_t current_section = (be_gas_section_t) -1;
-static int force_section = 0;
/**
* Return the pseudo-instruction to be issued for a section switch
".section\t.rodata",
".section\t.bss",
".section\t.tbss,\"awT\",@nobits",
- ".section\t.ctors,\"aw\",@progbits"
+ ".section\t.ctors,\"aw\",@progbits",
+ NULL, /* no cstring section */
+ NULL,
+ NULL
},
{ /* GAS_FLAVOUR_MINGW */
".section\t.text",
".section .rdata,\"dr\"",
".section\t.bss",
".section\t.tbss,\"awT\",@nobits",
- ".section\t.ctors,\"aw\",@progbits"
+ ".section\t.ctors,\"aw\",@progbits",
+ NULL,
+ NULL,
+ NULL
},
{ /* GAS_FLAVOUR_YASM */
".section\t.text",
".section\t.rodata",
".section\t.bss",
".section\t.tbss,\"awT\",@nobits",
- ".section\t.ctors,\"aw\",@progbits"
+ ".section\t.ctors,\"aw\",@progbits",
+ NULL,
+ NULL,
+ NULL
},
{ /* GAS_FLAVOUR_MACH_O */
".text",
".data",
".const",
".data",
- NULL, /* TLS is not supported on Mach-O */
- NULL /* constructors aren't marked with sections in Mach-O */
+ NULL, /* TLS is not supported on Mach-O */
+ ".mod_init_func",
+ ".cstring",
+ ".section\t__IMPORT,__jump_table,symbol_stubs,self_modifying_code+pure_instructions,5",
+ ".section\t__IMPORT,__pointers,non_lazy_symbol_pointers"
}
};
}
void be_gas_emit_switch_section(be_gas_section_t section) {
- if(current_section == section || force_section)
+ if(current_section == section)
return;
be_emit_char('\t');
void be_gas_emit_function_prolog(ir_entity *entity, unsigned alignment)
{
const char *name = get_entity_ld_name(entity);
+ const char *fill_byte = "";
+ unsigned maximum_skip;
be_gas_emit_switch_section(GAS_SECTION_TEXT);
be_emit_char('\n');
be_emit_write_line();
- const char *fill_byte = "";
/* gcc fills space between function with 0x90, no idea if this is needed */
- if(be_gas_flavour == GAS_FLAVOUR_MACH_O) {
+ if (be_gas_flavour == GAS_FLAVOUR_MACH_O) {
fill_byte = "0x90";
}
- unsigned maximum_skip = (1 << alignment) - 1;
- be_emit_cstring("\t.p2align ");
- be_emit_irprintf("%u,%s,%u\n", alignment, fill_byte, maximum_skip);
- be_emit_write_line();
-
+ if (alignment > 0) {
+ maximum_skip = (1 << alignment) - 1;
+ be_emit_cstring("\t.p2align ");
+ be_emit_irprintf("%u,%s,%u\n", alignment, fill_byte, maximum_skip);
+ be_emit_write_line();
+ }
if (get_entity_visibility(entity) == visibility_external_visible) {
be_emit_cstring(".globl ");
be_emit_string(name);
*/
typedef struct _be_gas_decl_env {
const be_main_env_t *main_env; /**< The main backend environment, used for it's debug handle. */
- waitq *worklist; /**< A worklist we use to place not yet handled entities on. */
+ be_gas_section_t section;
+ waitq *worklist; /**< A worklist we use to place not yet handled entities on. */
} be_gas_decl_env_t;
/************************************************************************/
switch (bytes) {
case 1:
be_emit_irprintf("0x%02x", get_tarval_sub_bits(tv, 0));
- break;
+ return;
case 2:
be_emit_irprintf("0x%02x%02x", get_tarval_sub_bits(tv, 1), get_tarval_sub_bits(tv, 0));
- break;
+ return;
case 4:
be_emit_irprintf("0x%02x%02x%02x%02x",
get_tarval_sub_bits(tv, 3), get_tarval_sub_bits(tv, 2), get_tarval_sub_bits(tv, 1), get_tarval_sub_bits(tv, 0));
- break;
+ return;
case 8:
be_emit_irprintf("0x%02x%02x%02x%02x%02x%02x%02x%02x",
get_tarval_sub_bits(tv, 7), get_tarval_sub_bits(tv, 6), get_tarval_sub_bits(tv, 5), get_tarval_sub_bits(tv, 4),
get_tarval_sub_bits(tv, 3), get_tarval_sub_bits(tv, 2), get_tarval_sub_bits(tv, 1), get_tarval_sub_bits(tv, 0));
- break;
+ return;
- case 10:
case 12:
- break;
+ be_emit_irprintf("0x%02x%02x%02x%02x%02x%02x%02x%02x"
+ "%02x%02x%02x%02x", get_tarval_sub_bits(tv, 11),
+ get_tarval_sub_bits(tv, 10), get_tarval_sub_bits(tv, 9),
+ get_tarval_sub_bits(tv, 8), get_tarval_sub_bits(tv, 7),
+ get_tarval_sub_bits(tv, 6), get_tarval_sub_bits(tv, 5),
+ get_tarval_sub_bits(tv, 4), get_tarval_sub_bits(tv, 3),
+ get_tarval_sub_bits(tv, 2), get_tarval_sub_bits(tv, 1),
+ get_tarval_sub_bits(tv, 0));
+ return;
case 16:
be_emit_irprintf("0x%02x%02x%02x%02x%02x%02x%02x%02x"
get_tarval_sub_bits(tv, 5), get_tarval_sub_bits(tv, 4),
get_tarval_sub_bits(tv, 3), get_tarval_sub_bits(tv, 2),
get_tarval_sub_bits(tv, 1), get_tarval_sub_bits(tv, 0));
- break;
-
-
- default:
- fprintf(stderr, "Try to dump an tarval with %d bytes\n", bytes);
- assert(0);
+ return;
}
+
+ panic("Can't dump a tarval with %d bytes\n", bytes);
}
/**
* Return the label prefix for labeled blocks.
*/
-const char *be_gas_label_prefix(void) {
+const char *be_gas_block_label_prefix(void) {
return ".LG";
}
+/**
+ * Return the label prefix for labeled instructions.
+ */
+const char *be_gas_insn_label_prefix(void) {
+ return ".LE";
+}
+
/**
* Dump a label.
*/
static void dump_label(ir_label_t label) {
- be_emit_irprintf("%s%ld", be_gas_label_prefix(), label);
+ be_emit_irprintf("%s%u", be_gas_block_label_prefix(), label);
}
/**
return;
case iro_Add:
+ if (!mode_is_int(mode) && !mode_is_reference(mode)) {
+ panic("Constant must be int or pointer for '+' to work");
+ }
do_dump_atomic_init(env, get_Add_left(init));
be_emit_cstring(" + ");
do_dump_atomic_init(env, get_Add_right(init));
return;
case iro_Sub:
+ if (!mode_is_int(mode) && !mode_is_reference(mode)) {
+ panic("Constant must be int or pointer for '-' to work");
+ }
do_dump_atomic_init(env, get_Sub_left(init));
be_emit_cstring(" - ");
do_dump_atomic_init(env, get_Sub_right(init));
return;
case iro_Mul:
+ if (!mode_is_int(mode) && !mode_is_reference(mode)) {
+ panic("Constant must be int or pointer for '*' to work");
+ }
do_dump_atomic_init(env, get_Mul_left(init));
be_emit_cstring(" * ");
do_dump_atomic_init(env, get_Mul_right(init));
static int initializer_is_string_const(const ir_initializer_t *initializer)
{
size_t i, len;
+ int found_printable = 0;
if(initializer->kind != IR_INITIALIZER_COMPOUND)
return 0;
return 0;
c = get_tarval_long(tv);
- if (i < len - 1) {
- if (!isgraph(c) && !isspace(c))
- return 0;
- } else {
- if (c != 0)
- return 0;
- }
+ if (isgraph(c) || isspace(c))
+ found_printable = 1;
+ else if(c != 0)
+ return 0;
+
+ if (i == len - 1 && c != '\0')
+ return 0;
}
- return 1;
+ return found_printable;
}
/**
ir_type *type, *element_type;
ir_mode *mode;
int i, c, n;
+ int found_printable = 0;
type = get_entity_type(ent);
c = (int) get_tarval_long(get_Const_tarval(irn));
- if((i < n - 1 && !(isgraph(c) || isspace(c)))
- || (i == n - 1 && c != '\0'))
+ if (isgraph(c) || isspace(c))
+ found_printable = 1;
+ else if(c != 0)
+ return 0;
+
+ if (i == n - 1 && c != '\0')
return 0;
}
}
/* then we can emit it as a string constant */
- return 1;
+ return found_printable;
}
/**
int type_size;
int remaining_space;
- be_gas_section_t last_section = current_section;
len = get_compound_ent_n_values(ent);
if (be_gas_flavour == GAS_FLAVOUR_MACH_O) {
- if (get_entity_variability(ent) == variability_constant) {
- be_gas_emit_switch_section(GAS_SECTION_CSTRING);
- }
be_emit_cstring("\t.ascii \"");
} else {
be_emit_cstring("\t.string \"");
if(remaining_space > 0) {
be_emit_irprintf("\t.skip\t%d\n", remaining_space);
}
- if(be_gas_flavour == GAS_FLAVOUR_MACH_O) {
- be_gas_emit_switch_section(last_section);
- }
}
static void dump_string_initializer(const ir_initializer_t *initializer)
len = initializer->compound.n_initializers;
if(be_gas_flavour == GAS_FLAVOUR_MACH_O) {
+ be_emit_cstring("\t.ascii \"");
+ } else {
be_emit_cstring("\t.string \"");
len -= 1;
- } else {
- be_emit_cstring("\t.ascii \"");
}
for(i = 0; i < len; ++i) {
/* now write values sorted */
for (k = 0; k < size; ) {
- int space = 0, skip = 0;
+ int space = 0;
+ int elem_size = 1;
if (vals[k].kind == NORMAL) {
if(vals[k].v.value != NULL) {
dump_atomic_init(env, vals[k].v.value);
- skip = get_mode_size_bytes(get_irn_mode(vals[k].v.value)) - 1;
+ elem_size = get_mode_size_bytes(get_irn_mode(vals[k].v.value));
} else {
- space = 1;
+ elem_size = 0;
}
} else if(vals[k].kind == TARVAL) {
tarval *tv = vals[k].v.tarval;
assert(tv != NULL);
- skip = size - 1;
+ elem_size = size;
dump_size_type(size);
dump_arith_tarval(tv, size);
be_emit_char('\n');
be_emit_write_line();
}
- ++k;
+ k += elem_size;
while (k < size && vals[k].kind == NORMAL && vals[k].v.value == NULL) {
++space;
++k;
}
- space -= skip;
- assert(space >= 0);
/* a gap */
if (space > 0) {
*
* @param env the gas output environment
* @param ent the entity to be dumped
- * @param emit_commons if non-zero, emit commons (non-local uninitialized entities)
*/
-static void dump_global(be_gas_decl_env_t *env, ir_entity *ent, int emit_commons)
+static void dump_global(be_gas_decl_env_t *env, ir_entity *ent)
{
ir_type *type = get_entity_type(ent);
ident *ld_ident = get_entity_ld_ident(ent);
unsigned align = get_type_alignment_bytes(type);
int emit_as_common = 0;
- be_gas_section_t section;
- ir_variability variability;
- ir_visibility visibility;
-
- if (is_Method_type(type)) {
- if (be_gas_flavour != GAS_FLAVOUR_MACH_O
- && get_method_img_section(ent) == section_constructors) {
- be_gas_emit_switch_section(GAS_SECTION_CTOR);
- emit_align(align);
- dump_size_type(align);
- be_emit_ident(ld_ident);
- be_emit_char('\n');
- be_emit_write_line();
- }
+ be_gas_section_t section = env->section;
+ ir_variability variability = get_entity_variability(ent);
+ ir_visibility visibility = get_entity_visibility(ent);
+ if (is_Method_type(type) && section != GAS_SECTION_PIC_TRAMPOLINES) {
return;
}
- variability = get_entity_variability(ent);
- visibility = get_entity_visibility(ent);
- section = GAS_SECTION_DATA;
- if (variability == variability_constant) {
+ if (section != (be_gas_section_t) -1) {
+ emit_as_common = 0;
+ } else if (variability == variability_constant) {
/* a constant entity, put it on the rdata */
section = GAS_SECTION_RODATA;
+ if (be_gas_flavour == GAS_FLAVOUR_MACH_O
+ && ent_is_string_const(ent)) {
+ section = GAS_SECTION_CSTRING;
+ }
} else if (variability == variability_uninitialized) {
/* uninitialized entity put it in bss segment */
section = GAS_SECTION_COMMON;
- if (emit_commons && visibility != visibility_local)
+ if (visibility != visibility_local)
emit_as_common = 1;
+ } else {
+ section = GAS_SECTION_DATA;
}
if(!emit_as_common) {
return;
}
/* alignment */
- if (align > 1 && !emit_as_common) {
+ if (align > 1 && !emit_as_common && section != GAS_SECTION_PIC_TRAMPOLINES
+ && section != GAS_SECTION_PIC_SYMBOLS) {
emit_align(align);
}
be_emit_write_line();
break;
}
+ } else if (section == GAS_SECTION_PIC_TRAMPOLINES) {
+ if (be_gas_flavour == GAS_FLAVOUR_MACH_O) {
+ be_emit_cstring("\t.indirect_symbol ");
+ be_emit_ident(get_entity_ident(ent));
+ be_emit_char('\n');
+ be_emit_write_line();
+ be_emit_cstring("\thlt ; hlt ; hlt ; hlt ; hlt\n");
+ be_emit_write_line();
+ } else {
+ panic("PIC trampolines not yet supported in this gas mode");
+ }
} else {
- be_emit_irprintf("\t.zero %u\n", get_type_size_bytes(type));
+ be_emit_irprintf("\t.space %u\n", get_type_size_bytes(type));
be_emit_write_line();
}
} else {
*
* @param gt a global like type, either the global or the TLS one
* @param env an environment
- * @param emit_commons if non-zero, emit commons (non-local uninitialized entities)
* @param only_emit_marked if non-zero, external allocated entities that do not have
* its visited flag set are ignored
*/
static void be_gas_dump_globals(ir_type *gt, be_gas_decl_env_t *env,
- int emit_commons, int only_emit_marked)
+ int only_emit_marked)
{
int i, n = get_compound_n_members(gt);
waitq *worklist = new_waitq();
while (!waitq_empty(worklist)) {
ir_entity *ent = waitq_get(worklist);
- dump_global(env, ent, emit_commons);
+ dump_global(env, ent);
}
del_waitq(worklist);
int only_emit_marked_entities)
{
be_gas_decl_env_t env;
+ memset(&env, 0, sizeof(env));
env.main_env = main_env;
/* dump global type */
- be_gas_dump_globals(get_glob_type(), &env, 1, only_emit_marked_entities);
-
- /* dump the Thread Local Storage */
- if (be_gas_flavour != GAS_FLAVOUR_MACH_O) {
- be_gas_emit_switch_section(GAS_SECTION_TLS);
- force_section = 1;
- be_gas_dump_globals(get_tls_type(), &env, 0, only_emit_marked_entities);
- force_section = 0;
+ env.section = (be_gas_section_t) -1;
+ be_gas_dump_globals(get_glob_type(), &env, only_emit_marked_entities);
+ env.section = GAS_SECTION_TLS;
+ be_gas_dump_globals(get_tls_type(), &env, only_emit_marked_entities);
+ env.section = GAS_SECTION_CTOR;
+ be_gas_dump_globals(get_constructors_type(), &env,
+ only_emit_marked_entities);
+ env.section = GAS_SECTION_PIC_SYMBOLS;
+ be_gas_dump_globals(main_env->pic_symbols_type, &env,
+ only_emit_marked_entities);
+
+ if (get_compound_n_members(main_env->pic_trampolines_type) > 0) {
+ env.section = GAS_SECTION_PIC_TRAMPOLINES;
+ be_gas_dump_globals(main_env->pic_trampolines_type, &env,
+ only_emit_marked_entities);
+ if (be_gas_flavour == GAS_FLAVOUR_MACH_O) {
+ be_emit_cstring("\t.subsections_via_symbols\n");
+ be_emit_write_line();
+ }
}
}