123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200 |
- /* write.c - emit .o file - Copyright(C)1986 Free Software Foundation, Inc.
- Copyright (C) 1986,1987 Free Software Foundation, Inc.
- 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
- the Free Software Foundation; either version 1, or (at your option)
- 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, 675 Mass Ave, Cambridge, MA 02139, USA. */
- /*
- Umm, with real good luck, this thing should be set up to do byteordering
- correctly, but I may have managed to miss a place or two. Treat a.out
- very carefully until you're SURE that it works. . .
- In order to cross-assemble the target machine must have an a.out header
- similar to the one in a.out.h on THIS machine. Byteorder doesn't matter;
- we take special care of it, but the numbers must be the same SIZE (# of
- bytes) and in the same PLACE. If this is not true, you will have some
- trouble.
- */
- #include "as.h"
- #include "md.h"
- #include "subsegs.h"
- #include "obstack.h"
- #include "struc-symbol.h"
- #include "write.h"
- #include "symbols.h"
- #ifdef SPARC
- #include "sparc.h"
- #endif
- void append();
- #ifdef hpux
- #define EXEC_MACHINE_TYPE HP9000S200_ID
- #endif
- /*
- * In: length of relocation (or of address) in chars: 1, 2 or 4.
- * Out: GNU LD relocation length code: 0, 1, or 2.
- */
- static unsigned char
- nbytes_r_length [] = {
- 42, 0, 1, 42, 2
- };
- static struct frag * text_frag_root;
- static struct frag * data_frag_root;
- static struct frag * text_last_frag; /* Last frag in segment. */
- static struct frag * data_last_frag; /* Last frag in segment. */
- static struct exec the_exec;
- static long int string_byte_count;
- static char * the_object_file;
- #ifndef SPARC
- static
- #endif
- char * next_object_file_charP; /* Tracks object file bytes. */
- static long int size_of_the_object_file; /* # bytes in object file. */
- /* static long int length; JF unused */ /* String length, including trailing '\0'. */
- static void relax_segment();
- void emit_segment();
- static relax_addressT relax_align();
- static long int fixup_segment();
- #ifndef SPARC
- static void emit_relocations();
- #endif
- /*
- * fix_new()
- *
- * Create a fixS in obstack 'notes'.
- */
- void
- #ifdef SPARC
- fix_new (frag, where, size, add_symbol, sub_symbol, offset, pcrel, r_type)
- #else
- fix_new (frag, where, size, add_symbol, sub_symbol, offset, pcrel)
- #endif
- fragS * frag; /* Which frag? */
- int where; /* Where in that frag? */
- short int size; /* 1, 2 or 4 usually. */
- symbolS * add_symbol; /* X_add_symbol. */
- symbolS * sub_symbol; /* X_subtract_symbol. */
- long int offset; /* X_add_number. */
- int pcrel; /* TRUE if PC-relative relocation. */
- #ifdef SPARC
- int r_type;
- #endif
- {
- register fixS * fixP;
- fixP = (fixS *)obstack_alloc(¬es,sizeof(fixS));
- fixP -> fx_frag = frag;
- fixP -> fx_where = where;
- fixP -> fx_size = size;
- fixP -> fx_addsy = add_symbol;
- fixP -> fx_subsy = sub_symbol;
- fixP -> fx_offset = offset;
- fixP -> fx_pcrel = pcrel;
- fixP -> fx_next = * seg_fix_rootP;
- /* JF these 'cuz of the NS32K stuff */
- fixP -> fx_im_disp = 0;
- fixP -> fx_pcrel_adjust = 0;
- fixP -> fx_bsr = 0;
- fixP ->fx_bit_fixP = 0;
- #ifdef SPARC
- fixP->fx_r_type = r_type;
- #endif
- * seg_fix_rootP = fixP;
- }
- void
- write_object_file()
- {
- register struct frchain * frchainP; /* Track along all frchains. */
- register fragS * fragP; /* Track along all frags. */
- register struct frchain * next_frchainP;
- register fragS * * prev_fragPP;
- register char * name;
- register symbolS * symbolP;
- register symbolS ** symbolPP;
- /* register fixS * fixP; JF unused */
- unsigned
- text_siz,
- data_siz,
- syms_siz,
- tr_siz,
- dr_siz;
- void output_file_create();
- void output_file_append();
- void output_file_close();
- void gdb_emit();
- void gdb_end();
- extern long omagic; /* JF magic # to write out. Is different for
- Suns and Vaxen and other boxes */
- #ifdef VMS
- /*
- * Under VMS we try to be compatible with VAX-11 "C". Thus, we
- * call a routine to check for the definition of the procedure
- * "_main", and if so -- fix it up so that it can be program
- * entry point.
- */
- VMS_Check_For_Main();
- #endif /* VMS */
- /*
- * After every sub-segment, we fake an ".align ...". This conforms to BSD4.2
- * brane-damage. We then fake ".fill 0" because that is the kind of frag
- * that requires least thought. ".align" frags like to have a following
- * frag since that makes calculating their intended length trivial.
- */
- #define SUB_SEGMENT_ALIGN (2)
- for ( frchainP=frchain_root; frchainP; frchainP=frchainP->frch_next )
- {
- #ifdef VMS
- /*
- * Under VAX/VMS, the linker (and PSECT specifications)
- * take care of correctly aligning the segments.
- * Doing the alignment here (on initialized data) can
- * mess up the calculation of global data PSECT sizes.
- */
- #undef SUB_SEGMENT_ALIGN
- #define SUB_SEGMENT_ALIGN ((frchainP->frch_seg != SEG_DATA) ? 2 : 0)
- #endif /* VMS */
- subseg_new (frchainP -> frch_seg, frchainP -> frch_subseg);
- frag_align (SUB_SEGMENT_ALIGN, 0);
- /* frag_align will have left a new frag. */
- /* Use this last frag for an empty ".fill". */
- /*
- * For this segment ...
- * Create a last frag. Do not leave a "being filled in frag".
- */
- frag_wane (frag_now);
- frag_now -> fr_fix = 0;
- know( frag_now -> fr_next == NULL );
- /* know( frags . obstack_c_base == frags . obstack_c_next_free ); */
- /* Above shows we haven't left a half-completed object on obstack. */
- }
- /*
- * From now on, we don't care about sub-segments.
- * Build one frag chain for each segment. Linked thru fr_next.
- * We know that there is at least 1 text frchain & at least 1 data frchain.
- */
- prev_fragPP = &text_frag_root;
- for ( frchainP=frchain_root; frchainP; frchainP=next_frchainP )
- {
- know( frchainP -> frch_root );
- * prev_fragPP = frchainP -> frch_root;
- prev_fragPP = & frchainP -> frch_last -> fr_next;
- if ( ((next_frchainP = frchainP->frch_next) == NULL)
- || next_frchainP == data0_frchainP)
- {
- prev_fragPP = & data_frag_root;
- if ( next_frchainP )
- {
- text_last_frag = frchainP -> frch_last;
- }
- else
- {
- data_last_frag = frchainP -> frch_last;
- }
- }
- } /* for(each struct frchain) */
- /*
- * We have two segments. If user gave -R flag, then we must put the
- * data frags into the text segment. Do this before relaxing so
- * we know to take advantage of -R and make shorter addresses.
- */
- if ( flagseen [ 'R' ] )
- {
- fixS *tmp;
- text_last_frag -> fr_next = data_frag_root;
- text_last_frag = data_last_frag;
- data_last_frag = NULL;
- data_frag_root = NULL;
- if(text_fix_root) {
- for(tmp=text_fix_root;tmp->fx_next;tmp=tmp->fx_next)
- ;
- tmp->fx_next=data_fix_root;
- } else
- text_fix_root=data_fix_root;
- data_fix_root=NULL;
- }
- relax_segment (text_frag_root, SEG_TEXT);
- relax_segment (data_frag_root, SEG_DATA);
- /*
- * Now the addresses of frags are correct within the segment.
- */
- know( text_last_frag -> fr_type == rs_fill && text_last_frag -> fr_offset == 0 );
- text_siz=text_last_frag->fr_address;
- #ifdef SPARC
- text_siz= (text_siz+7)&(~7);
- text_last_frag->fr_address=text_siz;
- #endif
- md_number_to_chars((char *)&the_exec.a_text,text_siz, sizeof(the_exec.a_text));
- /* the_exec . a_text = text_last_frag -> fr_address; */
- /*
- * Join the 2 segments into 1 huge segment.
- * To do this, re-compute every rn_address in the SEG_DATA frags.
- * Then join the data frags after the text frags.
- *
- * Determine a_data [length of data segment].
- */
- if (data_frag_root)
- {
- register relax_addressT slide;
- know( text_last_frag -> fr_type == rs_fill && text_last_frag -> fr_offset == 0 );
- data_siz=data_last_frag->fr_address;
- #ifdef SPARC
- data_siz += (8 - (data_siz % 8)) % 8;
- data_last_frag->fr_address = data_siz;
- #endif
- md_number_to_chars((char *)&the_exec.a_data,data_siz,sizeof(the_exec.a_data));
- /* the_exec . a_data = data_last_frag -> fr_address; */
- slide = text_siz; /* Address in file of the data segment. */
- for (fragP = data_frag_root;
- fragP;
- fragP = fragP -> fr_next)
- {
- fragP -> fr_address += slide;
- }
- know( text_last_frag );
- text_last_frag -> fr_next = data_frag_root;
- }
- else {
- md_number_to_chars((char *)&the_exec.a_data,0,sizeof(the_exec.a_data));
- data_siz = 0;
- }
- bss_address_frag . fr_address = text_siz + data_siz;
- #ifdef SPARC
- local_bss_counter=(local_bss_counter+7)&(~7);
- #endif
- md_number_to_chars((char *)&the_exec.a_bss,local_bss_counter,sizeof(the_exec.a_bss));
-
- /*
- *
- * Crawl the symbol chain.
- *
- * For each symbol whose value depends on a frag, take the address of
- * that frag and subsume it into the value of the symbol.
- * After this, there is just one way to lookup a symbol value.
- * Values are left in their final state for object file emission.
- * We adjust the values of 'L' local symbols, even if we do
- * not intend to emit them to the object file, because their values
- * are needed for fix-ups.
- *
- * Unless we saw a -L flag, remove all symbols that begin with 'L'
- * from the symbol chain.
- *
- * Count the (length of the nlists of the) (remaining) symbols.
- * Assign a symbol number to each symbol.
- * Count the number of string-table chars we will emit.
- *
- */
- know( zero_address_frag . fr_address == 0 );
- string_byte_count = sizeof( string_byte_count );
- /* JF deal with forward references first. . . */
- for(symbolP=symbol_rootP;symbolP;symbolP=symbolP->sy_next) {
- if(symbolP->sy_forward) {
- symbolP->sy_value+=symbolP->sy_forward->sy_value+symbolP->sy_forward->sy_frag->fr_address;
- symbolP->sy_forward=0;
- }
- }
- symbolPP = & symbol_rootP; /* -> last symbol chain link. */
- {
- register long int symbol_number;
- symbol_number = 0;
- while (symbolP = * symbolPP)
- {
- name = symbolP -> sy_name;
- if(flagseen['R'] && (symbolP->sy_nlist.n_type&N_DATA)) {
- symbolP->sy_nlist.n_type&= ~N_DATA;
- symbolP->sy_nlist.n_type|= N_TEXT;
- }
- /* if(symbolP->sy_forward) {
- symbolP->sy_value += symbolP->sy_forward->sy_value + symbolP->sy_forward->sy_frag->fr_address;
- } */
-
- symbolP -> sy_value += symbolP -> sy_frag -> fr_address;
- /* JF the 128 bit is a hack so stabs like
- "LET_STMT:23. . ." don't go away */
- /* CPH: 128 bit hack is moby loser. N_SO for file "Lower.c"
- fell through the cracks. I think that N_STAB should be
- used instead of 128. */
- /* JF the \001 bit is to make sure that local labels
- ( 1: - 9: don't make it into the symtable either */
- #ifndef VMS /* Under VMS we need to keep local symbols */
- if ( !name || (symbolP->sy_nlist.n_type&N_STAB)
- || (name[0]!='\001' && (flagseen ['L'] || name [0] != 'L' )))
- #endif /* not VMS */
- {
- symbolP -> sy_number = symbol_number ++;
- #ifndef VMS
- if (name)
- { /* Ordinary case. */
- symbolP -> sy_name_offset = string_byte_count;
- string_byte_count += strlen (symbolP -> sy_name) + 1;
- }
- else /* .Stabd case. */
- #endif /* not VMS */
- symbolP -> sy_name_offset = 0;
- symbolPP = & (symbolP -> sy_next);
- }
- #ifndef VMS
- else
- * symbolPP = symbolP -> sy_next;
- #endif /* not VMS */
- } /* for each symbol */
- syms_siz = sizeof( struct nlist) * symbol_number;
- md_number_to_chars((char *)&the_exec.a_syms,syms_siz,sizeof(the_exec.a_syms));
- /* the_exec . a_syms = sizeof( struct nlist) * symbol_number; */
- }
- /*
- * Addresses of frags now reflect addresses we use in the object file.
- * Symbol values are correct.
- * Scan the frags, converting any ".org"s and ".align"s to ".fill"s.
- * Also converting any machine-dependent frags using md_convert_frag();
- */
- subseg_change( SEG_TEXT, 0);
- for (fragP = text_frag_root; fragP; fragP = fragP -> fr_next)
- {
- switch (fragP -> fr_type)
- {
- case rs_align:
- case rs_org:
- fragP -> fr_type = rs_fill;
- know( fragP -> fr_var == 1 );
- know( fragP -> fr_next );
- fragP -> fr_offset
- = fragP -> fr_next -> fr_address
- - fragP -> fr_address
- - fragP -> fr_fix;
- break;
- case rs_fill:
- break;
- case rs_machine_dependent:
- md_convert_frag (fragP);
- /*
- * After md_convert_frag, we make the frag into a ".space 0".
- * Md_convert_frag() should set up any fixSs and constants
- * required.
- */
- frag_wane (fragP);
- break;
- #ifndef WORKING_DOT_WORD
- case rs_broken_word:
- {
- struct broken_word *lie;
- extern md_short_jump_size;
- extern md_long_jump_size;
- if(fragP->fr_subtype) {
- fragP->fr_fix+=md_short_jump_size;
- for(lie=(struct broken_word *)(fragP->fr_symbol);lie && lie->dispfrag==fragP;lie=lie->next_broken_word)
- if(lie->added==1)
- fragP->fr_fix+=md_long_jump_size;
- }
- frag_wane(fragP);
- }
- break;
- #endif
- default:
- BAD_CASE( fragP -> fr_type );
- break;
- } /* switch (fr_type) */
- } /* for each frag. */
- #ifndef WORKING_DOT_WORD
- {
- struct broken_word *lie;
- struct broken_word **prevP;
- prevP= &broken_words;
- for(lie=broken_words; lie; lie=lie->next_broken_word)
- if(!lie->added) {
- #ifdef SPARC
- fix_new( lie->frag, lie->word_goes_here - lie->frag->fr_literal,
- 2, lie->add,
- lie->sub, lie->addnum,
- 0, NO_RELOC);
- #endif
- #ifdef NS32K
- fix_new_ns32k(lie->frag,
- lie->word_goes_here - lie->frag->fr_literal,
- 2,
- lie->add,
- lie->sub,
- lie->addnum,
- 0, 2, 0, 0);
- #endif
- #if !defined(SPARC) && !defined(NS32K)
- fix_new( lie->frag, lie->word_goes_here - lie->frag->fr_literal,
- 2, lie->add,
- lie->sub, lie->addnum,
- 0);
- #endif
- /* md_number_to_chars(lie->word_goes_here,
- lie->add->sy_value
- + lie->addnum
- - (lie->sub->sy_value),
- 2); */
- *prevP=lie->next_broken_word;
- } else
- prevP= &(lie->next_broken_word);
- for(lie=broken_words;lie;) {
- struct broken_word *untruth;
- char *table_ptr;
- long table_addr;
- long from_addr,
- to_addr;
- int n,
- m;
- extern md_short_jump_size;
- extern md_long_jump_size;
- void md_create_short_jump();
- void md_create_long_jump();
- fragP=lie->dispfrag;
- /* Find out how many broken_words go here */
- n=0;
- for(untruth=lie;untruth && untruth->dispfrag==fragP;untruth=untruth->next_broken_word)
- if(untruth->added==1)
- n++;
- table_ptr=lie->dispfrag->fr_opcode;
- table_addr=lie->dispfrag->fr_address+(table_ptr - lie->dispfrag->fr_literal);
- /* Create the jump around the long jumps */
- /* This is a short jump from table_ptr+0 to table_ptr+n*long_jump_size */
- from_addr=table_addr;
- to_addr = table_addr + md_short_jump_size + n * md_long_jump_size;
- md_create_short_jump(table_ptr,from_addr,to_addr,lie->dispfrag,lie->add);
- table_ptr+=md_short_jump_size;
- table_addr+=md_short_jump_size;
- for(m=0;lie && lie->dispfrag==fragP;m++,lie=lie->next_broken_word) {
- if(lie->added==2)
- continue;
- /* Patch the jump table */
- /* This is the offset from ??? to table_ptr+0 */
- to_addr = table_addr
- - (lie->sub->sy_value);
- md_number_to_chars(lie->word_goes_here,to_addr,2);
- for(untruth=lie->next_broken_word;untruth && untruth->dispfrag==fragP;untruth=untruth->next_broken_word) {
- if(untruth->use_jump==lie)
- md_number_to_chars(untruth->word_goes_here,to_addr,2);
- }
- /* Install the long jump */
- /* this is a long jump from table_ptr+0 to the final target */
- from_addr=table_addr;
- to_addr=lie->add->sy_value+lie->addnum;
- md_create_long_jump(table_ptr,from_addr,to_addr,lie->dispfrag,lie->add);
- table_ptr+=md_long_jump_size;
- table_addr+=md_long_jump_size;
- }
- }
- }
- #endif
- #ifndef VMS
- /*
- * Scan every FixS performing fixups. We had to wait until now to do
- * this because md_convert_frag() may have made some fixSs.
- */
- /* the_exec . a_trsize
- = sizeof(struct relocation_info) * fixup_segment (text_fix_root, N_TEXT);
- the_exec . a_drsize
- = sizeof(struct relocation_info) * fixup_segment (data_fix_root, N_DATA); */
- tr_siz=sizeof(struct relocation_info) * fixup_segment (text_fix_root, N_TEXT);
- md_number_to_chars((char *)&the_exec.a_trsize, tr_siz ,sizeof(the_exec.a_trsize));
- dr_siz=sizeof(struct relocation_info) * fixup_segment (data_fix_root, N_DATA);
- md_number_to_chars((char *)&the_exec.a_drsize, dr_siz, sizeof(the_exec.a_drsize));
- #ifdef EXEC_MACHINE_TYPE
- md_number_to_chars((char *)&the_exec.a_machtype, EXEC_MACHINE_TYPE, sizeof(the_exec.a_machtype));
- #endif
- md_number_to_chars((char *)&the_exec.a_magic,omagic,sizeof(the_exec.a_magic));
- md_number_to_chars((char *)&the_exec.a_entry,0,sizeof(the_exec.a_entry));
- /* the_exec . a_entry = 0; */
- size_of_the_object_file =
- sizeof( the_exec ) +
- text_siz +
- data_siz +
- syms_siz +
- tr_siz +
- dr_siz +
- string_byte_count;
-
- next_object_file_charP
- = the_object_file
- = xmalloc ( size_of_the_object_file );
- output_file_create (out_file_name);
- append (& next_object_file_charP, (char *)(&the_exec), (unsigned long)sizeof(the_exec));
- /*
- * Emit code.
- */
- for (fragP = text_frag_root; fragP; fragP = fragP -> fr_next)
- {
- register long int count;
- register char * fill_literal;
- register long int fill_size;
- know( fragP -> fr_type == rs_fill );
- append (& next_object_file_charP, fragP -> fr_literal, (unsigned long)fragP -> fr_fix);
- fill_literal= fragP -> fr_literal + fragP -> fr_fix;
- fill_size = fragP -> fr_var;
- know( fragP -> fr_offset >= 0 );
- for (count = fragP -> fr_offset; count; count --)
- append (& next_object_file_charP, fill_literal, (unsigned long)fill_size);
- } /* for each code frag. */
- /*
- * Emit relocations.
- */
- emit_relocations (text_fix_root, (relax_addressT)0);
- emit_relocations (data_fix_root, text_last_frag -> fr_address);
- /*
- * Emit all symbols left in the symbol chain.
- * Any symbol still undefined is made N_EXT.
- */
- for ( symbolP = symbol_rootP; symbolP; symbolP = symbolP -> sy_next )
- {
- register char * temp;
- temp = symbolP -> sy_nlist . n_un . n_name;
- /* JF fix the numbers up. Call by value RULES! */
- md_number_to_chars((char *)&(symbolP -> sy_nlist . n_un . n_strx ),symbolP -> sy_name_offset,sizeof(symbolP -> sy_nlist . n_un . n_strx ));
- md_number_to_chars((char *)&(symbolP->sy_nlist.n_desc),symbolP->sy_nlist.n_desc,sizeof(symbolP -> sy_nlist . n_desc));
- md_number_to_chars((char *)&(symbolP->sy_nlist.n_value),symbolP->sy_nlist.n_value,sizeof(symbolP->sy_nlist.n_value));
- /* symbolP -> sy_nlist . n_un . n_strx = symbolP -> sy_name_offset; JF replaced by md above */
- if (symbolP -> sy_type == N_UNDF)
- symbolP -> sy_type |= N_EXT; /* Any undefined symbols become N_EXT. */
- append (& next_object_file_charP, (char *)(& symbolP -> sy_nlist),
- (unsigned long)sizeof(struct nlist));
- symbolP -> sy_nlist . n_un . n_name = temp;
- } /* for each symbol */
- /*
- * next_object_file_charP -> slot for next object byte.
- * Emit strings.
- * Find strings by crawling along symbol table chain.
- */
- /* Gotta do md_ byte-ordering stuff for string_byte_count first - KWK */
- md_number_to_chars((char *)&string_byte_count, string_byte_count, sizeof(string_byte_count));
- append (& next_object_file_charP, (char *)&string_byte_count, (unsigned long)sizeof(string_byte_count));
- for ( symbolP = symbol_rootP; symbolP; symbolP = symbolP -> sy_next )
- {
- if (symbolP -> sy_name)
- { /* Ordinary case: not .stabd. */
- append (& next_object_file_charP, symbolP -> sy_name,
- (unsigned long)(strlen (symbolP -> sy_name) + 1));
- }
- } /* for each symbol */
- know( next_object_file_charP == the_object_file + size_of_the_object_file );
- output_file_append (the_object_file, size_of_the_object_file, out_file_name);
- if (flagseen['G']) /* GDB symbol file to be appended? */
- {
- gdb_emit (out_file_name);
- gdb_end ();
- }
- output_file_close (out_file_name);
- #else /* VMS */
- /*
- * Now do the VMS-dependent part of writing the object file
- */
- VMS_write_object_file(text_siz, data_siz, text_frag_root, data_frag_root);
- #endif /* VMS */
- } /* write_object_file() */
- /*
- * relax_segment()
- *
- * Now we have a segment, not a crowd of sub-segments, we can make fr_address
- * values.
- *
- * Relax the frags.
- *
- * After this, all frags in this segment have addresses that are correct
- * within the segment. Since segments live in different file addresses,
- * these frag addresses may not be the same as final object-file addresses.
- */
- #ifndef VMS
- static
- #endif /* not VMS */
- void
- relax_segment (segment_frag_root, segment_type)
- struct frag * segment_frag_root;
- segT segment_type; /* N_DATA or N_TEXT */
- {
- register struct frag * fragP;
- register relax_addressT address;
- /* register relax_addressT old_address; JF unused */
- /* register relax_addressT new_address; JF unused */
- know( segment_type == SEG_DATA || segment_type == SEG_TEXT );
- /* In case md_estimate_size_before_relax() wants to make fixSs. */
- subseg_change (segment_type, 0);
- /*
- * For each frag in segment: count and store (a 1st guess of) fr_address.
- */
- address = 0;
- for ( fragP = segment_frag_root; fragP; fragP = fragP -> fr_next )
- {
- fragP -> fr_address = address;
- address += fragP -> fr_fix;
- switch (fragP -> fr_type)
- {
- case rs_fill:
- address += fragP -> fr_offset * fragP -> fr_var;
- break;
- case rs_align:
- address += relax_align (address, fragP -> fr_offset);
- break;
- case rs_org:
- /*
- * Assume .org is nugatory. It will grow with 1st relax.
- */
- break;
- case rs_machine_dependent:
- address += md_estimate_size_before_relax
- (fragP, seg_N_TYPE [(int) segment_type]);
- break;
- #ifndef WORKING_DOT_WORD
- /* Broken words don't concern us yet */
- case rs_broken_word:
- break;
- #endif
- default:
- BAD_CASE( fragP -> fr_type );
- break;
- } /* switch(fr_type) */
- } /* for each frag in the segment */
- /*
- * Do relax().
- */
- {
- register long int stretch; /* May be any size, 0 or negative. */
- /* Cumulative number of addresses we have */
- /* relaxed this pass. */
- /* We may have relaxed more than one address. */
- register long int stretched; /* Have we stretched on this pass? */
- /* This is 'cuz stretch may be zero, when,
- in fact some piece of code grew, and
- another shrank. If a branch instruction
- doesn't fit anymore, we could be scrod */
- do
- {
- stretch = stretched = 0;
- for (fragP = segment_frag_root; fragP; fragP = fragP -> fr_next)
- {
- register long int growth;
- register long int was_address;
- /* register long int var; */
- register long int offset;
- register symbolS * symbolP;
- register long int target;
- register long int after;
- register long int aim;
- was_address = fragP -> fr_address;
- address = fragP -> fr_address += stretch;
- symbolP = fragP -> fr_symbol;
- offset = fragP -> fr_offset;
- /* var = fragP -> fr_var; */
- switch (fragP -> fr_type)
- {
- case rs_fill: /* .fill never relaxes. */
- growth = 0;
- break;
- #ifndef WORKING_DOT_WORD
- /* JF: This is RMS's idea. I do *NOT* want to be blamed
- for it I do not want to write it. I do not want to have
- anything to do with it. This is not the proper way to
- implement this misfeature. */
- case rs_broken_word:
- {
- struct broken_word *lie;
- struct broken_word *untruth;
- extern int md_short_jump_size;
- extern int md_long_jump_size;
- /* Yes this is ugly (storing the broken_word pointer
- in the symbol slot). Still, this whole chunk of
- code is ugly, and I don't feel like doing anything
- about it. Think of it as stubbornness in action */
- growth=0;
- for(lie=(struct broken_word *)(fragP->fr_symbol);
- lie && lie->dispfrag==fragP;
- lie=lie->next_broken_word) {
- if(lie->added)
- continue;
- offset= lie->add->sy_frag->fr_address+lie->add->sy_value + lie->addnum -
- (lie->sub->sy_frag->fr_address+lie->sub->sy_value);
- if(offset<=-32768 || offset>=32767) {
- if(flagseen['k'])
- as_warn(".word %s-%s+%ld didn't fit",lie->add->sy_name,lie->sub->sy_name,lie->addnum);
- lie->added=1;
- if(fragP->fr_subtype==0) {
- fragP->fr_subtype++;
- growth+=md_short_jump_size;
- }
- for(untruth=lie->next_broken_word;untruth && untruth->dispfrag==lie->dispfrag;untruth=untruth->next_broken_word)
- if(untruth->add->sy_frag==lie->add->sy_frag && untruth->add->sy_value==lie->add->sy_value) {
- untruth->added=2;
- untruth->use_jump=lie;
- }
- growth+=md_long_jump_size;
- }
- }
- }
- break;
- #endif
- case rs_align:
- growth = relax_align ((relax_addressT)(address + fragP -> fr_fix), offset)
- - relax_align ((relax_addressT)(was_address + fragP -> fr_fix), offset);
- break;
- case rs_org:
- target = offset;
- if (symbolP)
- {
- know( ((symbolP -> sy_type & N_TYPE) == N_ABS) || ((symbolP -> sy_type & N_TYPE) == N_DATA) || ((symbolP -> sy_type & N_TYPE) == N_TEXT));
- know( symbolP -> sy_frag );
- know( (symbolP->sy_type&N_TYPE)!=N_ABS || symbolP->sy_frag==&zero_address_frag );
- target +=
- symbolP -> sy_value
- + symbolP -> sy_frag -> fr_address;
- }
- know( fragP -> fr_next );
- after = fragP -> fr_next -> fr_address;
- growth - ((target - after ) > 0) ? (target - after) : 0;
- /* Growth may be -ve, but variable part */
- /* of frag cannot have < 0 chars. */
- /* That is, we can't .org backwards. */
- growth -= stretch; /* This is an absolute growth factor */
- break;
- case rs_machine_dependent:
- {
- register relax_typeS * this_type;
- register relax_typeS * start_type;
- register relax_substateT next_state;
- register relax_substateT this_state;
- start_type = this_type
- = md_relax_table + (this_state = fragP -> fr_subtype);
- target = offset;
- if (symbolP)
- {
- know( ((symbolP -> sy_type & N_TYPE) == N_ABS) || ((symbolP -> sy_type &
- N_TYPE) == N_DATA) || ((symbolP -> sy_type & N_TYPE) == N_TEXT));
- know( symbolP -> sy_frag );
- know( (symbolP->sy_type&N_TYPE)!=N_ABS || symbolP->sy_frag==&zero_address_frag );
- target +=
- symbolP -> sy_value
- + symbolP -> sy_frag -> fr_address;
- /* If frag has yet to be reached on this pass,
- assume it will move by STRETCH just as we did.
- If this is not so, it will be because some frag
- between grows, and that will force another pass. */
- /* JF was just address */
- /* JF also added isdnrange hack */
- /* There's gotta be a better/faster/etc way
- to do this. . . */
- if (symbolP->sy_frag->fr_address > was_address && isdnrange(fragP,symbolP->sy_frag))
- target += stretch;
- }
- aim = target - address - fragP -> fr_fix;
- if (aim < 0)
- {
- /* Look backwards. */
- for (next_state = this_type -> rlx_more; next_state; )
- {
- if (aim >= this_type -> rlx_backward)
- next_state = 0;
- else
- { /* Grow to next state. */
- this_type = md_relax_table + (this_state = next_state);
- next_state = this_type -> rlx_more;
- }
- }
- }
- else
- {
- #ifdef DONTDEF
- /* JF these next few lines of code are for the mc68020 which can't handle short
- offsets of zero in branch instructions. What a kludge! */
- if(aim==0 && this_state==(1<<2+0)) { /* FOO hard encoded from m.c */
- aim=this_type->rlx_forward+1; /* Force relaxation into word mode */
- }
- #endif
- /* JF end of 68020 code */
- /* Look forwards. */
- for (next_state = this_type -> rlx_more; next_state; )
- {
- if (aim <= this_type -> rlx_forward)
- next_state = 0;
- else
- { /* Grow to next state. */
- this_type = md_relax_table + (this_state = next_state);
- next_state = this_type -> rlx_more;
- }
- }
- }
- if (growth = this_type -> rlx_length - start_type -> rlx_length)
- fragP -> fr_subtype = this_state;
- }
- break;
- default:
- BAD_CASE( fragP -> fr_type );
- break;
- }
- if(growth) {
- stretch += growth;
- stretched++;
- }
- } /* For each frag in the segment. */
- } while (stretched); /* Until nothing further to relax. */
- }
- /*
- * We now have valid fr_address'es for each frag.
- */
- /*
- * All fr_address's are correct, relative to their own segment.
- * We have made all the fixS we will ever make.
- */
- } /* relax_segment() */
- /*
- * Relax_align. Advance location counter to next address that has 'alignment'
- * lowest order bits all 0s.
- */
- static relax_addressT /* How many addresses does the .align take? */
- relax_align (address, alignment)
- register relax_addressT address; /* Address now. */
- register long int alignment; /* Alignment (binary). */
- {
- relax_addressT mask;
- relax_addressT new_address;
- mask = ~ ( (~0) << alignment );
- new_address = (address + mask) & (~ mask);
- return (new_address - address);
- }
- /*
- * fixup_segment()
- */
- static long int
- fixup_segment (fixP, this_segment_type)
- register fixS * fixP;
- int this_segment_type; /* N_TYPE bits for segment. */
- {
- register long int seg_reloc_count;
- /* JF these all used to be local to the for loop, but GDB doesn't seem to be able to deal with them there, so I moved them here for a bit. */
- register symbolS * add_symbolP;
- register symbolS * sub_symbolP;
- register long int add_number;
- register int size;
- register char * place;
- register long int where;
- register char pcrel;
- register fragS * fragP;
- register int add_symbol_N_TYPE;
- seg_reloc_count = 0;
- for ( ; fixP; fixP = fixP -> fx_next)
- {
- fragP = fixP -> fx_frag;
- know( fragP );
- where = fixP -> fx_where;
- place = fragP -> fr_literal + where;
- size = fixP -> fx_size;
- add_symbolP = fixP -> fx_addsy;
- sub_symbolP = fixP -> fx_subsy;
- add_number = fixP -> fx_offset;
- pcrel = fixP -> fx_pcrel;
- if(add_symbolP)
- add_symbol_N_TYPE = add_symbolP -> sy_type & N_TYPE;
- if (sub_symbolP)
- {
- if(!add_symbolP) /* Its just -sym */
- {
- if(sub_symbolP->sy_type!=N_ABS)
- as_warn("Negative of non-absolute symbol %s", sub_symbolP->sy_name);
- add_number-=sub_symbolP->sy_value;
- }
- else if ( ((sub_symbolP -> sy_type ^ add_symbol_N_TYPE) & N_TYPE) == 0
- && ( add_symbol_N_TYPE == N_DATA
- || add_symbol_N_TYPE == N_TEXT
- || add_symbol_N_TYPE == N_BSS
- || add_symbol_N_TYPE == N_ABS))
- {
- /* Difference of 2 symbols from same segment. */
- /* Can't make difference of 2 undefineds: 'value' means */
- /* something different for N_UNDF. */
- add_number += add_symbolP -> sy_value - sub_symbolP -> sy_value;
- add_symbolP = NULL;
- fixP -> fx_addsy = NULL;
- }
- else
- {
- /* Different segments in subtraction. */
- know( sub_symbolP -> sy_type != (N_ABS | N_EXT))
- if (sub_symbolP -> sy_type == N_ABS)
- add_number -= sub_symbolP -> sy_value;
- else
- {
- as_warn("Can't emit reloc {- %s-seg symbol \"%s\"} @ file address %d.",
- seg_name[(int)N_TYPE_seg[sub_symbolP->sy_type&N_TYPE]],
- sub_symbolP -> sy_name, fragP -> fr_address + where);
- }
- }
- }
- if (add_symbolP)
- {
- if (add_symbol_N_TYPE == this_segment_type && pcrel)
- {
- /*
- * This fixup was made when the symbol's segment was
- * SEG_UNKNOWN, but it is now in the local segment.
- * So we know how to do the address without relocation.
- */
- add_number += add_symbolP -> sy_value;
- add_number -= size + where + fragP -> fr_address;
- pcrel = 0; /* Lie. Don't want further pcrel processing. */
- fixP -> fx_addsy = NULL; /* No relocations please. */
- /*
- * It would be nice to check that the address does not overflow.
- * I didn't do this check because:
- * + It is machine dependent in the general case (eg 32032)
- * + Compiler output will never need this checking, so why
- * slow down the usual case?
- */
- }
- else
- {
- switch (add_symbol_N_TYPE)
- {
- case N_ABS:
- add_number += add_symbolP -> sy_value;
- fixP -> fx_addsy = NULL;
- add_symbolP = NULL;
- break;
-
- case N_BSS:
- case N_DATA:
- case N_TEXT:
- seg_reloc_count ++;
- add_number += add_symbolP -> sy_value;
- break;
-
- case N_UNDF:
- seg_reloc_count ++;
- break;
-
- default:
- BAD_CASE( add_symbol_N_TYPE );
- break;
- } /* switch on symbol seg */
- } /* if not in local seg */
- } /* if there was a + symbol */
- if (pcrel)
- {
- add_number -= size + where + fragP -> fr_address;
- if (add_symbolP == 0)
- {
- fixP -> fx_addsy = & abs_symbol;
- seg_reloc_count ++;
- }
- }
- /* OVE added fx_im_disp for ns32k and others */
- if (!fixP->fx_bit_fixP) {
- /* JF I hope this works . . . */
- if((size==1 && (add_number& ~0xFF) && (add_number&~0xFF!=(-1&~0xFF))) ||
- (size==2 && (add_number& ~0xFFFF) && (add_number&~0xFFFF!=(-1&~0xFFFF))))
- as_warn("Fixup of %d too large for field width of %d",add_number, size);
- switch (fixP->fx_im_disp) {
- case 0:
- #ifdef SPARC
- fixP->fx_addnumber = add_number;
- md_number_to_imm(place, add_number, size, fixP, this_segment_type);
- #else
- md_number_to_imm (place, add_number, size);
- /* OVE: the immediates, like disps, have lsb at lowest address */
- #endif
- break;
- case 1:
- md_number_to_disp (place,
- fixP->fx_pcrel ? add_number+fixP->fx_pcrel_adjust:add_number,
- size);
- break;
- case 2: /* fix requested for .long .word etc */
- md_number_to_chars (place, add_number, size);
- break;
- default:
- as_fatal("Internal error in write.c in fixup_segment");
- } /* OVE: maybe one ought to put _imm _disp _chars in one md-func */
- } else {
- md_number_to_field (place, add_number, fixP->fx_bit_fixP);
- }
- } /* For each fixS in this segment. */
- return (seg_reloc_count);
- } /* fixup_segment() */
- /* The sparc needs its own emit_relocations() */
- #ifndef SPARC
- /*
- * emit_relocations()
- *
- * Crawl along a fixS chain. Emit the segment's relocations.
- */
- static void
- emit_relocations (fixP, segment_address_in_file)
- register fixS * fixP; /* Fixup chain for this segment. */
- relax_addressT segment_address_in_file;
- {
- struct relocation_info ri;
- register symbolS * symbolP;
- /* JF this is for paranoia */
- bzero((char *)&ri,sizeof(ri));
- for ( ; fixP; fixP = fixP -> fx_next)
- {
- if (symbolP = fixP -> fx_addsy)
- {
- #ifndef hpux
- /* These two 'cuz of NS32K */
- ri . r_bsr = fixP -> fx_bsr;
- ri . r_disp = fixP -> fx_im_disp;
- #endif
- ri . r_length = nbytes_r_length [fixP -> fx_size];
- ri . r_pcrel = fixP -> fx_pcrel;
- ri . r_address = fixP -> fx_frag -> fr_address
- + fixP -> fx_where
- - segment_address_in_file;
- if ((symbolP -> sy_type & N_TYPE) == N_UNDF)
- {
- ri . r_extern = 1;
- ri . r_symbolnum = symbolP -> sy_number;
- }
- else
- {
- ri . r_extern = 0;
- ri . r_symbolnum = symbolP -> sy_type & N_TYPE;
- }
- /*
- The 68k machines assign bit-fields from higher bits to
- lower bits ("left-to-right") within the int. VAXen assign
- bit-fields from lower bits to higher bits ("right-to-left").
- Both handle multi-byte numbers in their usual fashion
- (Big-endian and little-endian stuff).
- Thus we need a machine dependent routine to make
- sure the structure is written out correctly. FUN!
- */
- md_ri_to_chars((char *) &ri, ri);
- append (&next_object_file_charP, (char *)& ri, (unsigned long)sizeof(ri));
- }
- }
- }
- #endif
- int
- isdnrange(f1,f2)
- struct frag *f1,*f2;
- {
- while(f1) {
- if(f1->fr_next==f2)
- return 1;
- f1=f1->fr_next;
- }
- return 0;
- }
- /* End: as-write.c */
|