binutils-gdb/ld/ldgram.y

732 lines
14 KiB
Plaintext
Raw Normal View History

1992-01-25 02:41:32 +08:00
/* A YACC grammer to parse a superset of the AT&T linker scripting languaue.
Copyright (C) 1991 Free Software Foundation, Inc.
Written by Steve Chamberlain of Cygnus Support (steve@cygnus.com).
1991-03-22 05:29:06 +08:00
1992-01-25 02:41:32 +08:00
This file is part of GNU ld.
1991-03-22 05:29:06 +08:00
1992-01-25 02:41:32 +08:00
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 2 of the License, or
(at your option) any later version.
1991-03-22 05:29:06 +08:00
1992-01-25 02:41:32 +08:00
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.
1991-03-22 05:29:06 +08:00
1992-01-25 02:41:32 +08:00
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., 675 Mass Ave, Cambridge, MA 02139, USA. */
%{
/*
1992-01-25 02:41:32 +08:00
*/
1991-03-22 05:29:06 +08:00
#define DONTDECLARE_MALLOC
1991-03-22 05:29:06 +08:00
#include "bfd.h"
#include "sysdep.h"
1991-03-22 05:29:06 +08:00
#include "ld.h"
#include "ldexp.h"
1991-05-18 12:08:59 +09:00
#include "ldver.h"
1991-03-22 05:29:06 +08:00
#include "ldlang.h"
#include "ldemul.h"
1991-03-22 05:29:06 +08:00
#include "ldfile.h"
#include "ldmisc.h"
1991-03-22 05:29:06 +08:00
#define YYDEBUG 1
boolean option_v;
extern unsigned int lineno;
extern boolean trace_files;
extern boolean write_map;
1992-01-25 02:41:32 +08:00
extern boolean option_longmap;
1991-03-22 05:29:06 +08:00
boolean hex_mode;
strip_symbols_type strip_symbols=STRIP_NONE;
discard_locals_type discard_locals=DISCARD_NONE;
1991-03-22 05:29:06 +08:00
lang_memory_region_type *region;
lang_memory_region_type *lang_memory_region_lookup();
lang_output_section_statement_type *lang_output_section_statement_lookup();
#ifdef __STDC__
void lang_add_data(int type, union etree_union *exp);
void lang_enter_output_section_statement(char *output_section_statement_name, etree_type *address_exp, int flags, bfd_vma block_value);
1991-03-22 05:29:06 +08:00
#else
void lang_add_data();
void lang_enter_output_section_statement();
#endif /* __STDC__ */
extern args_type command_line;
char *current_file;
boolean ldgram_want_filename = true;
boolean had_script = false;
boolean force_make_executable = false;
1991-04-19 09:59:53 +09:00
1991-04-14 12:22:42 +09:00
boolean ldgram_in_script = false;
1991-04-19 09:59:53 +09:00
boolean ldgram_had_equals = false;
1991-03-22 05:29:06 +08:00
/* LOCALS */
%}
%union {
bfd_vma integer;
int voidval;
char *name;
int token;
union etree_union *etree;
struct sec *section;
1991-03-22 05:29:06 +08:00
struct lang_output_section_statement_struct *output_section_statement;
union lang_statement_union **statement_ptr;
int lineno;
struct {
FILE *file;
char *name;
unsigned int lineno;
} state;
}
%type <etree> exp opt_exp mustbe_exp
%type <integer> fill_opt opt_block opt_type
1991-03-22 05:29:06 +08:00
%type <name> memspec_opt
%token <integer> INT
1991-03-22 05:29:06 +08:00
%token <name> NAME
%type <integer> length
1991-04-24 00:59:24 +09:00
%right <token> PLUSEQ MINUSEQ MULTEQ DIVEQ '=' LSHIFTEQ RSHIFTEQ ANDEQ OREQ
1991-03-22 05:29:06 +08:00
%right <token> '?' ':'
%left <token> OROR
%left <token> ANDAND
%left <token> '|'
%left <token> '^'
%left <token> '&'
%left <token> EQ NE
%left <token> '<' '>' LE GE
%left <token> LSHIFT RSHIFT
1991-03-22 05:29:06 +08:00
%left <token> '+' '-'
%left <token> '*' '/' '%'
/*%token <token> '+' '-' '*' '/' '%'*/
1991-03-22 05:29:06 +08:00
%right UNARY
%token END
1991-03-22 05:29:06 +08:00
%left <token> '('
%token <token> ALIGN_K BLOCK LONG SHORT BYTE
%token SECTIONS
%token '{' '}'
%token SIZEOF_HEADERS OUTPUT_FORMAT FORCE_COMMON_ALLOCATION OUTPUT_ARCH
%token SIZEOF_HEADERS
%token MEMORY
%token NOLOAD DSECT COPY INFO OVERLAY
1991-03-22 05:29:06 +08:00
%token NAME DEFINED TARGET_K SEARCH_DIR MAP ENTRY
1991-05-04 08:52:48 +09:00
%token OPTION_e OPTION_c OPTION_noinhibit_exec OPTION_s OPTION_S OPTION_sort_common
%token OPTION_format OPTION_F OPTION_u OPTION_Bstatic OPTION_N
%token <integer> SIZEOF NEXT ADDR
1991-04-24 00:59:24 +09:00
%token OPTION_d OPTION_dc OPTION_dp OPTION_x OPTION_X OPTION_defsym
%token OPTION_v OPTION_V OPTION_M OPTION_t STARTUP HLL SYSLIB FLOAT NOFLOAT
%token OPTION_Map
1992-01-25 02:41:32 +08:00
%token OPTION_n OPTION_r OPTION_o OPTION_b OPTION_R OPTION_relax
1991-03-22 05:29:06 +08:00
%token <name> OPTION_l OPTION_L OPTION_T OPTION_Aarch OPTION_Tfile OPTION_Texp
%token OPTION_Ur
%token ORIGIN FILL OPTION_g
%token LENGTH CREATE_OBJECT_SYMBOLS INPUT OUTPUT CONSTRUCTORS
%type <token> assign_op
1991-03-22 05:29:06 +08:00
%type <name> filename
%{
ld_config_type config;
%}
%%
file: command_line { lang_final(); };
filename: NAME;
1991-03-22 05:29:06 +08:00
1991-03-22 05:29:06 +08:00
command_line:
command_line command_line_option
|
;
command_line_option:
OPTION_Bstatic { }
1991-04-14 12:22:42 +09:00
| OPTION_v
1991-03-22 05:29:06 +08:00
{
ldversion(0);
1991-03-22 05:29:06 +08:00
option_v = true;
}
1992-01-25 02:41:32 +08:00
| OPTION_V
{
ldversion(1);
1992-01-25 02:41:32 +08:00
option_v = true;
}
1991-03-22 05:29:06 +08:00
| OPTION_t {
trace_files = true;
}
| OPTION_Map NAME
{
write_map = true;
config.map_filename = $2;
}
1991-03-22 05:29:06 +08:00
| OPTION_M {
config.map_filename = "-";
1991-03-22 05:29:06 +08:00
}
| OPTION_n {
config.magic_demand_paged = false;
}
| OPTION_N {
1992-01-25 02:41:32 +08:00
config.text_read_only = false;
config.magic_demand_paged = false;
}
1991-03-22 05:29:06 +08:00
| OPTION_s {
strip_symbols = STRIP_ALL;
}
| OPTION_S {
strip_symbols = STRIP_DEBUGGER;
}
1991-04-14 12:22:42 +09:00
| OPTION_u NAME {
ldlang_add_undef($2);
}
1991-03-22 05:29:06 +08:00
| OPTION_r {
config.relocateable_output = true;
config.build_constructors = false;
config.magic_demand_paged = false;
1992-01-25 02:41:32 +08:00
config.text_read_only = false;
1991-03-22 05:29:06 +08:00
}
| OPTION_Ur {
config.relocateable_output = true;
config.build_constructors = true;
config.magic_demand_paged = false;
1992-01-25 02:41:32 +08:00
config.text_read_only = false;
1991-03-22 05:29:06 +08:00
}
| OPTION_o filename
{
lang_add_output($2);
}
| OPTION_e NAME
{ lang_add_entry($2);
}
| OPTION_X {
discard_locals = DISCARD_L;
}
| OPTION_x {
discard_locals = DISCARD_ALL;
}
| OPTION_noinhibit_exec
{
force_make_executable = true;
}
1991-05-04 08:52:48 +09:00
| OPTION_sort_common {
config.sort_common = true;
}
1991-04-14 12:22:42 +09:00
| OPTION_d {
1991-03-22 05:29:06 +08:00
command_line.force_common_definition = true;
}
1992-01-25 02:41:32 +08:00
| OPTION_relax {
command_line.relax = true;
}
1991-04-14 12:22:42 +09:00
| OPTION_dc
1991-03-22 05:29:06 +08:00
{
command_line.force_common_definition = true;
}
| OPTION_g
{
/* Ignored */
}
1991-04-14 12:22:42 +09:00
| OPTION_dp
1991-03-22 05:29:06 +08:00
{
command_line.force_common_definition = true;
}
1991-04-14 12:22:42 +09:00
| OPTION_format NAME
{
lang_add_target($2);
}
1991-04-09 07:26:05 +08:00
| OPTION_Texp
1991-04-14 12:22:42 +09:00
{
hex_mode = 16;
1991-04-14 12:22:42 +09:00
}
INT
{
lang_section_start($1,exp_intop($3));
hex_mode = 0;
1991-04-14 12:22:42 +09:00
}
1991-03-22 05:29:06 +08:00
| OPTION_Aarch
1991-04-14 12:22:42 +09:00
{
ldfile_add_arch($1);
}
| OPTION_b NAME
1991-03-22 05:29:06 +08:00
{
lang_add_target($2);
}
| OPTION_L
{
ldfile_add_library_path($1);
1991-04-14 12:22:42 +09:00
}
| OPTION_F
{
/* Ignore */
}
1991-04-14 12:22:42 +09:00
| NAME
{ lang_add_input_file($1,lang_input_file_is_file_enum,
(char *)NULL); }
| OPTION_c filename
{ ldfile_open_command_file($2); } script_file END { ldlex_command()};
1991-04-14 12:22:42 +09:00
| OPTION_Tfile
{ ldfile_open_command_file($1); } script_file
END { ldlex_command();}
1991-03-22 05:29:06 +08:00
1991-04-14 12:22:42 +09:00
| OPTION_T filename
{ ldfile_open_command_file($2); } script_file
END { ldlex_command();}
1991-03-22 05:29:06 +08:00
| OPTION_l
{
lang_add_input_file($1,
lang_input_file_is_l_enum,
(char *)NULL);
}
1991-04-14 12:22:42 +09:00
| OPTION_R filename
1991-03-22 05:29:06 +08:00
{
lang_add_input_file($2,
lang_input_file_is_symbols_only_enum,
(char *)NULL);
}
| OPTION_defsym { ldlex_expression();
}
NAME '=' mustbe_exp { ldlex_popstate();
1991-04-19 09:59:53 +09:00
lang_add_assignment(exp_assop($4,$3,$5));
}
1991-04-14 12:22:42 +09:00
| '-' NAME
{ info("%P%F Unrecognised option -%s\n", $2); }
| '{' script_file '}'
1991-03-22 05:29:06 +08:00
;
1991-04-14 12:22:42 +09:00
1991-03-22 05:29:06 +08:00
1991-04-14 12:22:42 +09:00
script_file:
{
ldlex_both();
}
ifile_list
{
ldlex_popstate();
}
1991-04-14 12:22:42 +09:00
;
ifile_list:
ifile_list ifile_p1
|
1991-03-22 05:29:06 +08:00
;
ifile_p1:
memory
| sections
| startup
| high_level_library
| low_level_library
| floating_point_support
1991-04-18 09:06:26 +09:00
| statement_anywhere
| ';'
1991-03-22 05:29:06 +08:00
| TARGET_K '(' NAME ')'
{ lang_add_target($3); }
| SEARCH_DIR '(' filename ')'
{ ldfile_add_library_path($3); }
| OUTPUT '(' filename ')'
{ lang_add_output($3); }
| OUTPUT_FORMAT '(' NAME ')'
{ lang_add_output_format($3); }
1991-04-24 00:59:24 +09:00
| OUTPUT_ARCH '(' NAME ')'
{ ldfile_set_output_arch($3); }
| FORCE_COMMON_ALLOCATION
{ command_line.force_common_definition = true ; }
1991-03-22 05:29:06 +08:00
| INPUT '(' input_list ')'
| MAP '(' filename ')'
{ lang_add_map($3); }
;
input_list:
NAME
{ lang_add_input_file($1,lang_input_file_is_file_enum,
(char *)NULL); }
| input_list ',' NAME
{ lang_add_input_file($3,lang_input_file_is_file_enum,
(char *)NULL); }
| input_list NAME
{ lang_add_input_file($2, lang_input_file_is_file_enum,
(char *)NULL); }
;
sections:
SECTIONS '{' sec_or_group_p1 '}'
1991-03-22 05:29:06 +08:00
;
sec_or_group_p1:
sec_or_group_p1 section
| sec_or_group_p1 statement_anywhere
|
;
statement_anywhere:
ENTRY '(' NAME ')'
{ lang_add_entry($3); }
| assignment end
;
1991-04-14 12:22:42 +09:00
file_NAME_list:
NAME
{ lang_add_wild($1, current_file); }
| file_NAME_list opt_comma NAME
{ lang_add_wild($3, current_file); }
;
input_section_spec:
NAME
{
lang_add_wild((char *)NULL, $1);
}
| '['
{
current_file = (char *)NULL;
}
file_NAME_list
']'
| NAME
{
current_file =$1;
}
'(' file_NAME_list ')'
| '*'
{
current_file = (char *)NULL;
}
'(' file_NAME_list ')'
;
1991-03-22 05:29:06 +08:00
statement:
statement assignment end
| statement CREATE_OBJECT_SYMBOLS
{
lang_add_attribute(lang_object_symbols_statement_enum); }
| statement ';'
| statement CONSTRUCTORS
{
lang_add_attribute(lang_constructors_statement_enum); }
1991-03-22 05:29:06 +08:00
| statement input_section_spec
| statement length '(' exp ')'
1991-03-22 05:29:06 +08:00
{
lang_add_data($2,$4);
}
| statement FILL '(' exp ')'
1991-03-22 05:29:06 +08:00
{
lang_add_fill
(exp_get_value_int($4,
0,
"fill value",
lang_first_phase_enum));
}
|
;
length:
LONG
{ $$ = $1; }
| SHORT
{ $$ = $1; }
| BYTE
{ $$ = $1; }
;
fill_opt:
'=' mustbe_exp
1991-03-22 05:29:06 +08:00
{
$$ = exp_get_value_int($2,
0,
"fill value",
lang_first_phase_enum);
}
| { $$ = 0; }
;
assign_op:
PLUSEQ
{ $$ = '+'; }
| MINUSEQ
{ $$ = '-'; }
| MULTEQ
{ $$ = '*'; }
| DIVEQ
{ $$ = '/'; }
| LSHIFTEQ
{ $$ = LSHIFT; }
| RSHIFTEQ
{ $$ = RSHIFT; }
| ANDEQ
{ $$ = '&'; }
| OREQ
{ $$ = '|'; }
;
end: ';' | ','
1991-03-22 05:29:06 +08:00
;
assignment:
NAME '=' mustbe_exp
1991-03-22 05:29:06 +08:00
{
lang_add_assignment(exp_assop($2,$1,$3));
}
| NAME assign_op mustbe_exp
1991-03-22 05:29:06 +08:00
{
lang_add_assignment(exp_assop('=',$1,exp_binop($2,exp_nameop(NAME,$1),$3)));
}
;
opt_comma:
',' | ;
memory:
MEMORY '{' memory_spec memory_spec_list '}'
1991-03-22 05:29:06 +08:00
;
memory_spec_list:
memory_spec_list memory_spec
| memory_spec_list ',' memory_spec
|
;
memory_spec:
NAME
{ region = lang_memory_region_lookup($1); }
attributes_opt ':'
origin_spec opt_comma length_spec
1991-03-22 05:29:06 +08:00
;
origin_spec:
ORIGIN '=' mustbe_exp
1991-03-22 05:29:06 +08:00
{ region->current =
region->origin =
exp_get_vma($3, 0L,"origin", lang_first_phase_enum); }
;
length_spec:
LENGTH '=' mustbe_exp
1991-03-22 05:29:06 +08:00
{ region->length = exp_get_vma($3,
~((bfd_vma)0),
"length",
lang_first_phase_enum);
}
attributes_opt:
'(' NAME ')'
{
lang_set_flags(&region->flags, $2);
}
|
;
startup:
STARTUP '(' filename ')'
{ lang_startup($3); }
;
high_level_library:
HLL '(' high_level_library_NAME_list ')'
| HLL '(' ')'
{ ldemul_hll((char *)NULL); }
;
high_level_library_NAME_list:
high_level_library_NAME_list opt_comma filename
{ ldemul_hll($3); }
| filename
{ ldemul_hll($1); }
;
low_level_library:
SYSLIB '(' low_level_library_NAME_list ')'
;
low_level_library_NAME_list:
low_level_library_NAME_list opt_comma filename
{ ldemul_syslib($3); }
|
;
floating_point_support:
FLOAT
{ lang_float(true); }
| NOFLOAT
{ lang_float(false); }
;
mustbe_exp: { ldlex_expression(); }
exp
{ ldlex_popstate(); $$=$2;}
;
1991-03-22 05:29:06 +08:00
exp :
'-' exp %prec UNARY
{ $$ = exp_unop('-', $2); }
| '(' exp ')'
1991-03-22 05:29:06 +08:00
{ $$ = $2; }
| NEXT '(' exp ')' %prec UNARY
{ $$ = exp_unop($1,$3); }
| '!' exp %prec UNARY
{ $$ = exp_unop('!', $2); }
| '+' exp %prec UNARY
{ $$ = $2; }
| '~' exp %prec UNARY
{ $$ = exp_unop('~', $2);}
| exp '*' exp
{ $$ = exp_binop('*', $1, $3); }
| exp '/' exp
{ $$ = exp_binop('/', $1, $3); }
| exp '%' exp
{ $$ = exp_binop('%', $1, $3); }
| exp '+' exp
{ $$ = exp_binop('+', $1, $3); }
| exp '-' exp
{ $$ = exp_binop('-' , $1, $3); }
| exp LSHIFT exp
{ $$ = exp_binop(LSHIFT , $1, $3); }
| exp RSHIFT exp
{ $$ = exp_binop(RSHIFT , $1, $3); }
| exp EQ exp
{ $$ = exp_binop(EQ , $1, $3); }
| exp NE exp
{ $$ = exp_binop(NE , $1, $3); }
| exp LE exp
{ $$ = exp_binop(LE , $1, $3); }
| exp GE exp
{ $$ = exp_binop(GE , $1, $3); }
| exp '<' exp
{ $$ = exp_binop('<' , $1, $3); }
| exp '>' exp
{ $$ = exp_binop('>' , $1, $3); }
| exp '&' exp
{ $$ = exp_binop('&' , $1, $3); }
| exp '^' exp
{ $$ = exp_binop('^' , $1, $3); }
| exp '|' exp
{ $$ = exp_binop('|' , $1, $3); }
| exp '?' exp ':' exp
{ $$ = exp_trinop('?' , $1, $3, $5); }
| exp ANDAND exp
{ $$ = exp_binop(ANDAND , $1, $3); }
| exp OROR exp
{ $$ = exp_binop(OROR , $1, $3); }
| DEFINED '(' NAME ')'
{ $$ = exp_nameop(DEFINED, $3); }
| INT
{ $$ = exp_intop($1); }
1991-04-25 05:53:36 +09:00
| SIZEOF_HEADERS
{ $$ = exp_nameop(SIZEOF_HEADERS,0); }
1991-03-22 05:29:06 +08:00
| SIZEOF '(' NAME ')'
{ $$ = exp_nameop(SIZEOF,$3); }
1991-03-22 05:29:06 +08:00
| ADDR '(' NAME ')'
{ $$ = exp_nameop(ADDR,$3); }
1991-03-22 05:29:06 +08:00
| ALIGN_K '(' exp ')'
{ $$ = exp_unop(ALIGN_K,$3); }
1991-03-22 05:29:06 +08:00
| NAME
{ $$ = exp_nameop(NAME,$1); }
;
section: NAME { ldlex_expression(); }
opt_exp { ldlex_popstate(); }
opt_type opt_block ':' opt_things'{'
1991-03-22 05:29:06 +08:00
{
lang_enter_output_section_statement($1,$3,$5,$6);
1991-03-22 05:29:06 +08:00
}
statement '}' fill_opt memspec_opt
{
lang_leave_output_section_statement($13, $14);
1991-03-22 05:29:06 +08:00
}
;
opt_type:
'(' NOLOAD ')' { $$ = SEC_NO_FLAGS; }
| '(' DSECT ')' { $$ = 0; }
| '(' COPY ')' { $$ = 0; }
| '(' INFO ')' { $$ = 0; }
| '(' OVERLAY ')' { $$ = 0; }
| { $$ = SEC_ALLOC | SEC_LOAD | SEC_HAS_CONTENTS; }
;
opt_things:
{
};
1991-03-22 05:29:06 +08:00
opt_exp:
exp
1991-03-22 05:29:06 +08:00
{ $$ = $1; }
| { $$= (etree_type *)NULL; }
;
opt_block:
BLOCK '(' exp ')'
1991-03-22 05:29:06 +08:00
{ $$ = exp_get_value_int($3,
1L,
"block",
lang_first_phase_enum);
}
| { $$ = 1; }
;
memspec_opt:
'>' NAME
1991-03-22 05:29:06 +08:00
{ $$ = $2; }
| { $$ = "*default*"; }
;