binutils-gdb/gas/config/tc-mmix.h

233 lines
8.0 KiB
C
Raw Normal View History

2001-10-30 23:20:14 +08:00
/* tc-mmix.h -- Header file for tc-mmix.c.
Copyright (C) 2001-2022 Free Software Foundation, Inc.
2001-10-30 23:20:14 +08:00
Written by Hans-Peter Nilsson (hp@bitrange.com).
This file is part of GAS, the GNU Assembler.
GAS is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
2007-07-03 19:01:12 +08:00
the Free Software Foundation; either version 3, or (at your option)
2001-10-30 23:20:14 +08:00
any later version.
GAS 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 GAS; see the file COPYING. If not, write to the Free
Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
02110-1301, USA. */
2001-10-30 23:20:14 +08:00
#define TC_MMIX
/* See gas/doc/internals.texi for explanation of these macros. */
#define TARGET_FORMAT "elf64-mmix"
#define TARGET_ARCH bfd_arch_mmix
#define TARGET_BYTES_BIG_ENDIAN 1
extern const char mmix_comment_chars[];
#define tc_comment_chars mmix_comment_chars
extern const char mmix_symbol_chars[];
#define tc_symbol_chars mmix_symbol_chars
/* "@" is a synonym for ".". */
#define LEX_AT (LEX_BEGIN_NAME)
extern int mmix_label_without_colon_this_line (void);
2001-10-30 23:20:14 +08:00
#define LABELS_WITHOUT_COLONS mmix_label_without_colon_this_line ()
extern int mmix_next_semicolon_is_eoln;
#define TC_EOL_IN_INSN(p) (*(p) == ';' && ! mmix_next_semicolon_is_eoln)
/* This is one direction we can get mmixal compatibility. */
extern void mmix_handle_mmixal (void);
2001-10-30 23:20:14 +08:00
#define md_start_line_hook mmix_handle_mmixal
extern void mmix_md_begin (void);
2001-10-30 23:20:14 +08:00
#define md_begin mmix_md_begin
gas: rename md_end to md_finish Currently md_end is typically used for some final actions rather than freeing memory like other *_end functions. Rename it to md_finish, and rename target implementation. The renaming of target functions makes it possible to find them all with "grep md_finish", eg. md_mips_end is renamed to mips_md_finish, not md_mips_finish. This patch leaves a number of md_end functions unchanged, those that either do nothing or deallocate memory, and calls them late. The idea here is that target maintainers implement md_end functions to tidy memory, if anyone cares. Freeing persistent memory in gas is not at all important, except that it can hide more important memory leaks, those that happen once per some frequent gas operation, amongst these unimportant memory leaks. * as.c (main): Rename md_end to md_finish. * config/tc-alpha.c, * config/tc-alpha.h, * config/tc-arc.c, * config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h, * config/tc-csky.c, * config/tc-csky.h, * config/tc-ia64.c, * config/tc-ia64.h, * config/tc-mcore.c, * config/tc-mcore.h, * config/tc-mips.c, * config/tc-mips.h, * config/tc-mmix.c, * config/tc-mmix.h, * config/tc-msp430.c, * config/tc-msp430.h, * config/tc-nds32.c, * config/tc-nds32.h, * config/tc-ppc.c, * config/tc-ppc.h, * config/tc-pru.c, * config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h, * config/tc-s390.c, * config/tc-s390.h, * config/tc-sparc.c, * config/tc-sparc.h, * config/tc-tic4x.c, * config/tc-tic4x.h, * config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-v850.c, * config/tc-v850.h, * config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c, * config/tc-z80.h: Similarly. * output-file.c (output_file_close): Call md_end.
2022-07-05 11:56:38 +08:00
extern void mmix_md_finish (void);
#define md_finish mmix_md_finish
2001-10-30 23:20:14 +08:00
extern int mmix_current_location \
(void (*fn) (expressionS *), expressionS *);
extern int mmix_parse_predefined_name (char *, expressionS *);
2001-10-30 23:20:14 +08:00
extern char *mmix_current_prefix;
/* A bit ugly, since we "know" that there's a static function
current_location that does what we want. We also strip off a leading
':' in another ugly way.
The [DVWIOUZX]_Handler symbols are provided when-used. */
extern int mmix_gnu_syntax;
#define md_parse_name(name, exp, mode, cpos) \
2001-10-30 23:20:14 +08:00
(! mmix_gnu_syntax \
&& (name[0] == '@' \
? (! is_part_of_name (name[1]) \
&& mmix_current_location (current_location, exp)) \
: ((name[0] == ':' || ISUPPER (name[0])) \
2001-10-30 23:20:14 +08:00
&& mmix_parse_predefined_name (name, exp))))
extern char *mmix_prefix_name (char *);
2001-10-30 23:20:14 +08:00
/* We implement when *creating* a symbol, we also need to strip a ':' or
prepend a prefix. */
#define tc_canonicalize_symbol_name(x) \
(mmix_current_prefix == NULL && (x)[0] != ':' ? (x) : mmix_prefix_name (x))
#define md_undefined_symbol(x) NULL
extern void mmix_fb_label (expressionS *);
2001-10-30 23:20:14 +08:00
/* Since integer_constant is local to expr.c, we have to make this a
macro. FIXME: Do it cleaner. */
#define md_operand(exp) \
do \
2001-10-30 23:20:14 +08:00
{ \
if (input_line_pointer[0] == '#') \
{ \
input_line_pointer++; \
integer_constant (16, (exp)); \
} \
else if (input_line_pointer[0] == '&' \
&& input_line_pointer[1] != '&') \
as_bad (_("`&' serial number operator is not supported")); \
else \
mmix_fb_label (exp); \
2001-10-30 23:20:14 +08:00
} \
while (0)
2001-10-30 23:20:14 +08:00
/* Gas dislikes the 2ADD, 8ADD etc. insns, so we have to assemble them in
the error-recovery loop. Hopefully there are no significant
differences. Also, space on a line isn't gracefully handled. */
extern int mmix_assemble_return_nonzero (char *);
2001-10-30 23:20:14 +08:00
#define tc_unrecognized_line(c) \
((c) == ' ' \
|| (((c) == '1' || (c) == '2' || (c) == '4' || (c) == '8') \
&& mmix_assemble_return_nonzero (input_line_pointer - 1)))
#define md_number_to_chars number_to_chars_bigendian
#define WORKING_DOT_WORD
extern const struct relax_type mmix_relax_table[];
#define TC_GENERIC_RELAX_TABLE mmix_relax_table
/* We use the relax table for everything except the GREG frags and PUSHJ. */
extern long mmix_md_relax_frag (segT, fragS *, long);
2001-10-30 23:20:14 +08:00
#define md_relax_frag mmix_md_relax_frag
2002-09-05 08:01:18 +08:00
#define tc_fix_adjustable(FIX) \
(((FIX)->fx_addsy == NULL \
|| S_GET_SEGMENT ((FIX)->fx_addsy) != reg_section) \
&& (FIX)->fx_r_type != BFD_RELOC_VTABLE_INHERIT \
&& (FIX)->fx_r_type != BFD_RELOC_VTABLE_ENTRY \
&& (FIX)->fx_r_type != BFD_RELOC_MMIX_LOCAL)
2001-10-30 23:20:14 +08:00
/* Adjust symbols which are registers. */
#define tc_adjust_symtab() mmix_adjust_symtab ()
extern void mmix_adjust_symtab (void);
2001-10-30 23:20:14 +08:00
2002-09-05 08:01:18 +08:00
/* Here's where we make all symbols global, when so requested.
2001-10-30 23:20:14 +08:00
We must avoid doing that for expression symbols or section symbols,
though. */
extern int mmix_globalize_symbols;
#define tc_frob_symbol(sym, punt) \
do \
{ \
2002-09-05 08:01:18 +08:00
if (S_GET_SEGMENT (sym) == reg_section) \
{ \
if (S_GET_NAME (sym)[0] != '$' \
&& S_GET_VALUE (sym) < 256) \
{ \
if (mmix_globalize_symbols) \
S_SET_EXTERNAL (sym); \
else \
symbol_mark_used_in_reloc (sym); \
} \
} \
else if (mmix_globalize_symbols \
&& ! symbol_section_p (sym) \
&& sym != section_symbol (absolute_section) \
&& ! S_IS_LOCAL (sym)) \
S_SET_EXTERNAL (sym); \
} \
while (0)
2001-10-30 23:20:14 +08:00
2002-09-05 08:01:18 +08:00
/* No shared lib support, so we don't need to ensure externally
visible symbols can be overridden. */
#define EXTERN_FORCE_RELOC 0
2001-10-30 23:20:14 +08:00
/* When relaxing, we need to emit various relocs we otherwise wouldn't. */
#define TC_FORCE_RELOCATION(fix) mmix_force_relocation (fix)
extern int mmix_force_relocation (struct fix *);
2001-10-30 23:20:14 +08:00
/* Call md_pcrel_from_section(), not md_pcrel_from(). */
2002-09-05 08:01:18 +08:00
#define MD_PCREL_FROM_SECTION(FIX, SEC) md_pcrel_from_section (FIX, SEC)
2001-10-30 23:20:14 +08:00
#define md_section_align(seg, size) (size)
#define LISTING_HEADER "GAS for MMIX"
/* The default of 4 means Bcc expansion looks like it's missing a line. */
#define LISTING_LHS_CONT_LINES 5
extern fragS *mmix_opcode_frag;
#define TC_FRAG_TYPE fragS *
gas: Pass max_bytes to TC_FRAG_INIT ommit 3ae729d5a4f63740ed9a778960b17c2912b0bbdd Author: H.J. Lu <hjl.tools@gmail.com> Date: Wed Mar 7 04:18:45 2018 -0800 x86: Rewrite NOP generation for fill and alignment increased MAX_MEM_FOR_RS_ALIGN_CODE to 4095 which resulted in increase of assembler time and memory usage by 5 times for inputs with many .p2align directives, which is typical for LTO output. This patch passes max_bytes to TC_FRAG_INIT so that MAX_MEM_FOR_RS_ALIGN_CODE can be set as needed and tracked by backend it so that HANDLE_ALIGN can check the maximum alignment for each rs_align_code frag. Wall time to assemble the same cc1plus.s: before: 423.78user 0.89system 7:05.71elapsed 99%CPU after: 102.35user 0.27system 1:42.89elapsed 99%CPU PR gas/24165 * frags.c (frag_var_init): Pass max_chars to TC_FRAG_INIT as max_bytes. * config/tc-aarch64.h (TC_FRAG_INIT): Add and pass max_bytes to aarch64_init_frag. * /config/tc-arm.h (TC_FRAG_INIT): And and pass max_bytes to arm_init_frag. * config/tc-avr.h (TC_FRAG_INIT): And and ignore max_bytes. * config/tc-ia64.h (TC_FRAG_INIT): Likewise. * config/tc-mmix.h (TC_FRAG_INIT): Likewise. * config/tc-nds32.h (TC_FRAG_INIT): Likewise. * config/tc-ns32k.h (TC_FRAG_INIT): Likewise. * config/tc-rl78.h (TC_FRAG_INIT): Likewise. * config/tc-rx.h (TC_FRAG_INIT): Likewise. * config/tc-score.h (TC_FRAG_INIT): Likewise. * config/tc-tic54x.h (TC_FRAG_INIT): Likewise. * config/tc-tic6x.h (TC_FRAG_INIT): Likewise. * config/tc-xtensa.h (TC_FRAG_INIT): Likewise. * config/tc-i386.h (MAX_MEM_FOR_RS_ALIGN_CODE): Set to (alignment ? ((1 << alignment) - 1) : 1) (i386_tc_frag_data): Add max_bytes. (TC_FRAG_INIT): Add and track max_bytes. (HANDLE_ALIGN): Replace MAX_MEM_FOR_RS_ALIGN_CODE with fragP->tc_frag_data.max_bytes. * doc/internals.texi: Update TC_FRAG_TYPE with max_bytes.
2019-02-10 20:34:10 +08:00
#define TC_FRAG_INIT(frag, max_bytes) (frag)->tc_frag_data = mmix_opcode_frag
2001-10-30 23:20:14 +08:00
/* We need to associate each section symbol with a list of GREGs defined
for that section/segment and sorted on offset, between the point where
all symbols have been evaluated and all frags mapped, and when the
fixups are done and relocs are output. Similarly for each unknown
symbol. */
extern void mmix_frob_file (void);
#define tc_frob_file_before_fix() \
do \
{ \
int i = 0; \
\
/* It's likely mmix_frob_file changed (removed) sections, so make \
sure sections are correctly numbered as per renumber_sections, \
(static to write.c where this macro is called). */ \
mmix_frob_file (); \
bfd_map_over_sections (stdoutput, renumber_sections, &i); \
} \
while (0)
2001-10-30 23:20:14 +08:00
/* Used by mmix_frob_file. Hangs on section symbols and unknown symbols. */
struct mmix_symbol_gregs;
#define TC_SYMFIELD_TYPE struct mmix_symbol_gregs *
/* Used by relaxation, counting maximum needed PUSHJ stubs for a section. */
struct mmix_segment_info_type
{
/* We only need to keep track of the last stubbable frag because
there's no less hackish way to keep track of different relaxation
rounds. */
fragS *last_stubfrag;
bfd_size_type nstubs;
};
#define TC_SEGMENT_INFO_TYPE struct mmix_segment_info_type
extern void mmix_md_elf_section_change_hook (void);
2001-10-30 23:20:14 +08:00
#define md_elf_section_change_hook mmix_md_elf_section_change_hook
extern void mmix_md_do_align (int, char *, int, int);
2001-10-30 23:20:14 +08:00
#define md_do_align(n, fill, len, max, label) \
mmix_md_do_align (n, fill, len, max)
/* Each insn is a tetrabyte (4 bytes) long, but if there are BYTE
sequences sprinkled in, we can get unaligned DWARF2 offsets, so let's
explicitly say one byte. */
#define DWARF2_LINE_MIN_INSN_LENGTH 1
/* MMIX has global register symbols. */
#define TC_GLOBAL_REGISTER_SYMBOL_OK
#define md_single_noop_insn "swym 0"