1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039 |
- {
- $Id$
- Copyright (c) 1998-2000 by Peter Vreman
- This unit implements an binary assembler output class
- 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.
- 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., 675 Mass Ave, Cambridge, MA 02139, USA.
- ****************************************************************************
- }
- {$ifdef TP}
- {$N+,E+}
- {$endif}
- unit ag386bin;
- {$define MULTIPASS}
- {$define EXTERNALBSS}
- interface
- uses
- cpubase,cobjects,aasm,files,assemble;
- type
- togtype=(og_none,og_dbg,og_coff,og_pecoff);
- pi386binasmlist=^ti386binasmlist;
- ti386binasmlist=object
- SmartAsm : boolean;
- constructor init(t:togtype;smart:boolean);
- destructor done;
- procedure WriteBin;
- private
- { the aasmoutput lists that need to be processed }
- lists : byte;
- list : array[1..maxoutputlists] of paasmoutput;
- { current processing }
- currlistidx : byte;
- currlist : paasmoutput;
- currpass : byte;
- {$ifdef GDB}
- n_line : byte; { different types of source lines }
- linecount,
- includecount : longint;
- funcname : pasmsymbol;
- stabslastfileinfo : tfileposinfo;
- procedure convertstabs(p:pchar);
- {$ifdef unused}
- procedure emitsymbolstabs(s : string;nidx,nother,line : longint;firstasm,secondasm : pasmsymbol);
- {$endif}
- procedure emitlineinfostabs(nidx,line : longint);
- procedure emitstabs(s:string);
- procedure WriteFileLineInfo(var fileinfo : tfileposinfo);
- procedure StartFileLineInfo;
- procedure EndFileLineInfo;
- {$endif}
- function MaybeNextList(var hp:pai):boolean;
- function TreePass0(hp:pai):pai;
- function TreePass1(hp:pai):pai;
- function TreePass2(hp:pai):pai;
- procedure writetree;
- procedure writetreesmart;
- end;
- implementation
- uses
- strings,
- globtype,globals,systems,verbose,
- cpuasm,
- {$ifdef GDB}
- gdb,
- {$endif}
- og386,og386dbg,og386cff;
- {$ifdef GDB}
- procedure ti386binasmlist.convertstabs(p:pchar);
- var
- ofs,
- nidx,nother,ii,i,line,j : longint;
- code : integer;
- hp : pchar;
- reloc : boolean;
- sec : tsection;
- ps : pasmsymbol;
- s : string;
- begin
- ofs:=0;
- reloc:=true;
- ps:=nil;
- sec:=sec_none;
- if p[0]='"' then
- begin
- i:=1;
- { we can have \" inside the string !! PM }
- while not ((p[i]='"') and (p[i-1]<>'\')) do
- inc(i);
- p[i]:=#0;
- ii:=i;
- hp:=@p[1];
- s:=StrPas(@P[i+2]);
- end
- else
- begin
- hp:=nil;
- s:=StrPas(P);
- i:=-2; {needed below (PM) }
- end;
- { When in pass 1 then only alloc and leave }
- if currpass=1 then
- begin
- objectalloc^.staballoc(hp);
- if assigned(hp) then
- p[i]:='"';
- exit;
- end;
- { Parse the rest of the stabs }
- if s='' then
- internalerror(33000);
- j:=pos(',',s);
- if j=0 then
- internalerror(33001);
- Val(Copy(s,1,j-1),nidx,code);
- if code<>0 then
- internalerror(33002);
- i:=i+2+j;
- Delete(s,1,j);
- j:=pos(',',s);
- if (j=0) then
- internalerror(33003);
- Val(Copy(s,1,j-1),nother,code);
- if code<>0 then
- internalerror(33004);
- i:=i+j;
- Delete(s,1,j);
- j:=pos(',',s);
- if j=0 then
- begin
- j:=256;
- ofs:=-1;
- end;
- Val(Copy(s,1,j-1),line,code);
- if code<>0 then
- internalerror(33005);
- if ofs=0 then
- begin
- Delete(s,1,j);
- i:=i+j;
- Val(s,ofs,code);
- if code=0 then
- reloc:=false
- else
- begin
- ofs:=0;
- s:=strpas(@p[i]);
- { handle asmsymbol or
- asmsymbol - asmsymbol }
- j:=pos(' ',s);
- if j=0 then
- j:=pos('-',s);
- { single asmsymbol }
- if j=0 then
- j:=256;
- { the symbol can be external
- so we must use newasmsymbol and
- not getasmsymbol !! PM }
- ps:=newasmsymbol(copy(s,1,j-1));
- if not assigned(ps) then
- internalerror(33006)
- else
- begin
- sec:=ps^.section;
- ofs:=ps^.address;
- reloc:=true;
- end;
- if j<256 then
- begin
- i:=i+j;
- s:=strpas(@p[i]);
- if (s<>'') and (s[1]=' ') then
- begin
- j:=0;
- while (s[j+1]=' ') do
- inc(j);
- i:=i+j;
- s:=strpas(@p[i]);
- end;
- ps:=getasmsymbol(s);
- if not assigned(ps) then
- internalerror(33007)
- else
- begin
- if ps^.section<>sec then
- internalerror(33008);
- ofs:=ofs-ps^.address;
- reloc:=false;
- end;
- end;
- end;
- end;
- { external bss need speical handling (PM) }
- if assigned(ps) and (ps^.section=sec_none) then
- objectoutput^.WriteSymStabs(sec,ofs,hp,ps,nidx,nother,line,reloc)
- else
- objectoutput^.WriteStabs(sec,ofs,hp,nidx,nother,line,reloc);
- if assigned(hp) then
- p[ii]:='"';
- end;
- {$ifdef unused}
- procedure ti386binasmlist.emitsymbolstabs(s : string;nidx,nother,line : longint;
- firstasm,secondasm : pasmsymbol);
- var
- hp : pchar;
- begin
- if s='' then
- hp:=nil
- else
- begin
- s:=s+#0;
- hp:=@s[1];
- end;
- if not assigned(secondasm) then
- begin
- if not assigned(firstasm) then
- internalerror(33009);
- objectoutput^.WriteStabs(firstasm^.section,firstasm^.address,hp,nidx,nother,line,true);
- end
- else
- begin
- if firstasm^.section<>secondasm^.section then
- internalerror(33010);
- objectoutput^.WriteStabs(firstasm^.section,firstasm^.address-secondasm^.address,
- hp,nidx,nother,line,false);
- end;
- end;
- {$endif}
- procedure ti386binasmlist.emitlineinfostabs(nidx,line : longint);
- var
- sec : tsection;
- begin
- if currpass=1 then
- begin
- objectalloc^.staballoc(nil);
- exit;
- end;
- if (nidx=n_textline) and assigned(funcname) and
- (target_os.use_function_relative_addresses) then
- objectoutput^.WriteStabs(sec_code,pgenericcoffoutput(objectoutput)^.sects[sec_code]^.len-funcname^.address,
- nil,nidx,0,line,false)
- else
- begin
- if nidx=n_textline then
- sec:=sec_code
- else if nidx=n_dataline then
- sec:=sec_data
- else
- sec:=sec_bss;
- objectoutput^.WriteStabs(sec,pgenericcoffoutput(objectoutput)^.sects[sec]^.len,
- nil,nidx,0,line,true);
- end;
- end;
- procedure ti386binasmlist.emitstabs(s:string);
- begin
- s:=s+#0;
- ConvertStabs(@s[1]);
- end;
- procedure ti386binasmlist.WriteFileLineInfo(var fileinfo : tfileposinfo);
- var
- curr_n : byte;
- hp : pasmsymbol;
- infile : pinputfile;
- begin
- if not (cs_debuginfo in aktmoduleswitches) then
- exit;
- { file changed ? (must be before line info) }
- if (fileinfo.fileindex<>0) and
- (stabslastfileinfo.fileindex<>fileinfo.fileindex) then
- begin
- infile:=current_module^.sourcefiles^.get_file(fileinfo.fileindex);
- if includecount=0 then
- curr_n:=n_sourcefile
- else
- curr_n:=n_includefile;
- { get symbol for this includefile }
- hp:=newasmsymbol('Ltext'+ToStr(IncludeCount));
- if currpass=1 then
- begin
- hp^.settyp(AS_LOCAL);
- hp^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- end
- else
- objectoutput^.writesymbol(hp);
- { emit stabs }
- if (infile^.path^<>'') then
- EmitStabs('"'+lower(BsToSlash(FixPath(infile^.path^,false)))+'",'+tostr(curr_n)+
- ',0,0,Ltext'+ToStr(IncludeCount));
- EmitStabs('"'+lower(FixFileName(infile^.name^))+'",'+tostr(curr_n)+
- ',0,0,Ltext'+ToStr(IncludeCount));
- inc(includecount);
- end;
- { line changed ? }
- if (stabslastfileinfo.line<>fileinfo.line) and (fileinfo.line<>0) then
- emitlineinfostabs(n_line,fileinfo.line);
- stabslastfileinfo:=fileinfo;
- end;
- procedure ti386binasmlist.StartFileLineInfo;
- var
- fileinfo : tfileposinfo;
- begin
- FillChar(stabslastfileinfo,sizeof(stabslastfileinfo),0);
- n_line:=n_textline;
- funcname:=nil;
- linecount:=1;
- includecount:=0;
- fileinfo.fileindex:=1;
- fileinfo.line:=1;
- WriteFileLineInfo(fileinfo);
- end;
- procedure ti386binasmlist.EndFileLineInfo;
- var
- hp : pasmsymbol;
- begin
- if not (cs_debuginfo in aktmoduleswitches) then
- exit;
- objectalloc^.setsection(sec_code);
- hp:=newasmsymbol('Letext');
- if currpass=1 then
- begin
- hp^.settyp(AS_LOCAL);
- hp^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- end
- else
- objectoutput^.writesymbol(hp);
- EmitStabs('"",'+tostr(n_sourcefile)+
- ',0,0,Letext');
- end;
- {$endif GDB}
- function ti386binasmlist.MaybeNextList(var hp:pai):boolean;
- begin
- { maybe end of list }
- while not assigned(hp) do
- begin
- if currlistidx<lists then
- begin
- inc(currlistidx);
- currlist:=list[currlistidx];
- hp:=pai(currlist^.first);
- end
- else
- begin
- MaybeNextList:=false;
- exit;
- end;
- end;
- MaybeNextList:=true;
- end;
- function ti386binasmlist.TreePass0(hp:pai):pai;
- var
- l : longint;
- begin
- while assigned(hp) do
- begin
- case hp^.typ of
- ait_align :
- begin
- { always use the maximum fillsize in this pass to avoid possible
- short jumps to become out of range }
- pai_align(hp)^.fillsize:=pai_align(hp)^.aligntype;
- objectalloc^.sectionalloc(pai_align(hp)^.fillsize);
- end;
- ait_datablock :
- begin
- {$ifdef EXTERNALBSS}
- if not SmartAsm then
- begin
- if not pai_datablock(hp)^.is_global then
- begin
- l:=pai_datablock(hp)^.size;
- if l>2 then
- objectalloc^.sectionalign(4)
- else if l>1 then
- objectalloc^.sectionalign(2);
- objectalloc^.sectionalloc(pai_datablock(hp)^.size);
- end;
- end
- else
- begin
- {$endif}
- l:=pai_datablock(hp)^.size;
- if l>2 then
- objectalloc^.sectionalign(4)
- else if l>1 then
- objectalloc^.sectionalign(2);
- objectalloc^.sectionalloc(pai_datablock(hp)^.size);
- end;
- end;
- ait_const_32bit :
- objectalloc^.sectionalloc(4);
- ait_const_16bit :
- objectalloc^.sectionalloc(2);
- ait_const_8bit :
- objectalloc^.sectionalloc(1);
- ait_real_80bit :
- objectalloc^.sectionalloc(10);
- ait_real_64bit :
- objectalloc^.sectionalloc(8);
- ait_real_32bit :
- objectalloc^.sectionalloc(4);
- ait_comp_64bit :
- objectalloc^.sectionalloc(8);
- ait_const_rva,
- ait_const_symbol :
- objectalloc^.sectionalloc(4);
- ait_section:
- objectalloc^.setsection(pai_section(hp)^.sec);
- ait_symbol :
- pai_symbol(hp)^.sym^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- ait_label :
- pai_label(hp)^.l^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- ait_string :
- objectalloc^.sectionalloc(pai_string(hp)^.len);
- ait_instruction :
- begin
- { reset instructions which could change in pass 2 }
- paicpu(hp)^.resetpass2;
- objectalloc^.sectionalloc(paicpu(hp)^.Pass1(objectalloc^.sectionsize));
- end;
- ait_cut :
- if SmartAsm then
- break;
- end;
- hp:=pai(hp^.next);
- end;
- TreePass0:=hp;
- end;
- function ti386binasmlist.TreePass1(hp:pai):pai;
- var
- l : longint;
- begin
- while assigned(hp) do
- begin
- {$ifdef GDB}
- { write stabs }
- if (cs_debuginfo in aktmoduleswitches) then
- begin
- if (objectalloc^.currsec<>sec_none) and
- not(hp^.typ in [
- ait_label,
- ait_regalloc,ait_tempalloc,
- ait_stabn,ait_stabs,ait_section,
- ait_cut,ait_marker,ait_align,ait_stab_function_name]) then
- WriteFileLineInfo(hp^.fileinfo);
- end;
- {$endif GDB}
- case hp^.typ of
- ait_align :
- begin
- { here we must determine the fillsize which is used in pass2 }
- pai_align(hp)^.fillsize:=align(objectalloc^.sectionsize,pai_align(hp)^.aligntype)-
- objectalloc^.sectionsize;
- objectalloc^.sectionalloc(pai_align(hp)^.fillsize);
- end;
- ait_datablock :
- begin
- if objectalloc^.currsec<>sec_bss then
- Message(asmw_e_alloc_data_only_in_bss);
- {$ifdef EXTERNALBSS}
- if not SmartAsm then
- begin
- if pai_datablock(hp)^.is_global then
- begin
- pai_datablock(hp)^.sym^.settyp(AS_EXTERNAL);
- pai_datablock(hp)^.sym^.setaddress(sec_none,pai_datablock(hp)^.size,pai_datablock(hp)^.size);
- end
- else
- begin
- l:=pai_datablock(hp)^.size;
- if l>2 then
- objectalloc^.sectionalign(4)
- else if l>1 then
- objectalloc^.sectionalign(2);
- pai_datablock(hp)^.sym^.settyp(AS_LOCAL);
- pai_datablock(hp)^.sym^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,
- pai_datablock(hp)^.size);
- objectalloc^.sectionalloc(pai_datablock(hp)^.size);
- end;
- end
- else
- {$endif}
- begin
- if pai_datablock(hp)^.is_global then
- pai_datablock(hp)^.sym^.settyp(AS_GLOBAL)
- else
- pai_datablock(hp)^.sym^.settyp(AS_LOCAL);
- l:=pai_datablock(hp)^.size;
- if l>2 then
- objectalloc^.sectionalign(4)
- else if l>1 then
- objectalloc^.sectionalign(2);
- pai_datablock(hp)^.sym^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,pai_datablock(hp)^.size);
- objectalloc^.sectionalloc(pai_datablock(hp)^.size);
- end;
- end;
- ait_const_32bit :
- objectalloc^.sectionalloc(4);
- ait_const_16bit :
- objectalloc^.sectionalloc(2);
- ait_const_8bit :
- objectalloc^.sectionalloc(1);
- ait_real_80bit :
- objectalloc^.sectionalloc(10);
- ait_real_64bit :
- objectalloc^.sectionalloc(8);
- ait_real_32bit :
- objectalloc^.sectionalloc(4);
- ait_comp_64bit :
- objectalloc^.sectionalloc(8);
- ait_const_rva,
- ait_const_symbol :
- objectalloc^.sectionalloc(4);
- ait_section:
- begin
- objectalloc^.setsection(pai_section(hp)^.sec);
- {$ifdef GDB}
- case pai_section(hp)^.sec of
- sec_code : n_line:=n_textline;
- sec_data : n_line:=n_dataline;
- sec_bss : n_line:=n_bssline;
- else
- n_line:=n_dataline;
- end;
- stabslastfileinfo.line:=-1;
- {$endif GDB}
- end;
- {$ifdef GDB}
- ait_stabn :
- convertstabs(pai_stabn(hp)^.str);
- ait_stabs :
- convertstabs(pai_stabs(hp)^.str);
- ait_stab_function_name :
- if assigned(pai_stab_function_name(hp)^.str) then
- funcname:=getasmsymbol(strpas(pai_stab_function_name(hp)^.str))
- else
- funcname:=nil;
- ait_force_line :
- stabslastfileinfo.line:=0;
- {$endif}
- ait_symbol :
- begin
- if pai_symbol(hp)^.is_global then
- pai_symbol(hp)^.sym^.settyp(AS_GLOBAL)
- else
- pai_symbol(hp)^.sym^.settyp(AS_LOCAL);
- pai_symbol(hp)^.sym^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- end;
- ait_label :
- begin
- if pai_label(hp)^.is_global then
- pai_label(hp)^.l^.settyp(AS_GLOBAL)
- else
- pai_label(hp)^.l^.settyp(AS_LOCAL);
- pai_label(hp)^.l^.setaddress(objectalloc^.currsec,objectalloc^.sectionsize,0);
- end;
- ait_string :
- objectalloc^.sectionalloc(pai_string(hp)^.len);
- ait_instruction :
- objectalloc^.sectionalloc(paicpu(hp)^.Pass1(objectalloc^.sectionsize));
- ait_direct :
- Message(asmw_f_direct_not_supported);
- ait_cut :
- if SmartAsm then
- break;
- end;
- hp:=pai(hp^.next);
- end;
- TreePass1:=hp;
- end;
- function ti386binasmlist.TreePass2(hp:pai):pai;
- var
- l : longint;
- {$ifdef I386}
- co : comp;
- {$endif I386}
- begin
- { main loop }
- while assigned(hp) do
- begin
- {$ifdef GDB}
- { write stabs }
- if cs_debuginfo in aktmoduleswitches then
- begin
- if (objectoutput^.currsec<>sec_none) and
- not(hp^.typ in [
- ait_label,
- ait_regalloc,ait_tempalloc,
- ait_stabn,ait_stabs,ait_section,
- ait_cut,ait_marker,ait_align,ait_stab_function_name]) then
- WriteFileLineInfo(hp^.fileinfo);
- end;
- {$endif GDB}
- case hp^.typ of
- ait_align :
- objectoutput^.writebytes(pai_align(hp)^.getfillbuf^,pai_align(hp)^.fillsize);
- ait_section :
- begin
- objectoutput^.defaultsection(pai_section(hp)^.sec);
- {$ifdef GDB}
- case pai_section(hp)^.sec of
- sec_code : n_line:=n_textline;
- sec_data : n_line:=n_dataline;
- sec_bss : n_line:=n_bssline;
- else
- n_line:=n_dataline;
- end;
- stabslastfileinfo.line:=-1;
- {$endif GDB}
- end;
- ait_symbol :
- objectoutput^.writesymbol(pai_symbol(hp)^.sym);
- ait_datablock :
- begin
- objectoutput^.writesymbol(pai_datablock(hp)^.sym);
- if SmartAsm
- {$ifdef EXTERNALBSS}
- or (not pai_datablock(hp)^.is_global)
- {$endif}
- then
- begin
- l:=pai_datablock(hp)^.size;
- if l>2 then
- objectoutput^.writealign(4)
- else if l>1 then
- objectoutput^.writealign(2);
- objectoutput^.writealloc(pai_datablock(hp)^.size);
- end;
- end;
- ait_const_32bit :
- objectoutput^.writebytes(pai_const(hp)^.value,4);
- ait_const_16bit :
- objectoutput^.writebytes(pai_const(hp)^.value,2);
- ait_const_8bit :
- objectoutput^.writebytes(pai_const(hp)^.value,1);
- ait_real_80bit :
- objectoutput^.writebytes(pai_real_80bit(hp)^.value,10);
- ait_real_64bit :
- objectoutput^.writebytes(pai_real_64bit(hp)^.value,8);
- ait_real_32bit :
- objectoutput^.writebytes(pai_real_32bit(hp)^.value,4);
- ait_comp_64bit :
- begin
- {$ifdef FPC}
- co:=comp(pai_comp_64bit(hp)^.value);
- {$else}
- co:=pai_comp_64bit(hp)^.value;
- {$endif}
- objectoutput^.writebytes(co,8);
- end;
- ait_string :
- objectoutput^.writebytes(pai_string(hp)^.str^,pai_string(hp)^.len);
- ait_const_rva :
- objectoutput^.writereloc(pai_const_symbol(hp)^.offset,4,
- pai_const_symbol(hp)^.sym,relative_rva);
- ait_const_symbol :
- objectoutput^.writereloc(pai_const_symbol(hp)^.offset,4,
- pai_const_symbol(hp)^.sym,relative_false);
- ait_label :
- objectoutput^.writesymbol(pai_label(hp)^.l);
- ait_instruction :
- paicpu(hp)^.Pass2;
- {$ifdef GDB}
- ait_stabn :
- convertstabs(pai_stabn(hp)^.str);
- ait_stabs :
- convertstabs(pai_stabs(hp)^.str);
- ait_stab_function_name :
- if assigned(pai_stab_function_name(hp)^.str) then
- funcname:=getasmsymbol(strpas(pai_stab_function_name(hp)^.str))
- else
- funcname:=nil;
- ait_force_line :
- stabslastfileinfo.line:=0;
- {$endif}
- ait_cut :
- if SmartAsm then
- break;
- end;
- hp:=pai(hp^.next);
- end;
- TreePass2:=hp;
- end;
- procedure ti386binasmlist.writetree;
- var
- hp : pai;
- begin
- { reset the asmsymbol list }
- ResetAsmsymbolList;
- objectoutput^.defaultsection(sec_code);
- {$ifdef MULTIPASS}
- { Pass 0 }
- currpass:=0;
- objectalloc^.setsection(sec_code);
- { start with list 1 }
- currlistidx:=1;
- currlist:=list[currlistidx];
- hp:=pai(currlist^.first);
- while assigned(hp) do
- begin
- hp:=TreePass0(hp);
- MaybeNextList(hp);
- end;
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- {$endif}
- { Pass 1 }
- currpass:=1;
- objectalloc^.resetsections;
- objectalloc^.setsection(sec_code);
- {$ifdef GDB}
- StartFileLineInfo;
- {$endif GDB}
- { start with list 1 }
- currlistidx:=1;
- currlist:=list[currlistidx];
- hp:=pai(currlist^.first);
- while assigned(hp) do
- begin
- hp:=TreePass1(hp);
- MaybeNextList(hp);
- end;
- {$ifdef GDB}
- EndFileLineInfo;
- {$endif GDB}
- { check for undefined labels }
- CheckAsmSymbolListUndefined;
- { set section sizes }
- objectoutput^.setsectionsizes(objectalloc^.secsize);
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- { Pass 2 }
- currpass:=2;
- {$ifdef GDB}
- StartFileLineInfo;
- {$endif GDB}
- { start with list 1 }
- currlistidx:=1;
- currlist:=list[currlistidx];
- hp:=pai(currlist^.first);
- while assigned(hp) do
- begin
- hp:=TreePass2(hp);
- MaybeNextList(hp);
- end;
- {$ifdef GDB}
- EndFileLineInfo;
- {$endif GDB}
- end;
- procedure ti386binasmlist.writetreesmart;
- var
- hp : pai;
- startsec : tsection;
- begin
- startsec:=sec_code;
- { start with list 1 }
- currlistidx:=1;
- currlist:=list[currlistidx];
- hp:=pai(currlist^.first);
- while assigned(hp) do
- begin
- { reset the asmsymbol list }
- ResetAsmsymbolList;
- {$ifdef MULTIPASS}
- { Pass 0 }
- currpass:=0;
- objectalloc^.resetsections;
- objectalloc^.setsection(startsec);
- TreePass0(hp);
- {$endif}
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- { Pass 1 }
- currpass:=1;
- objectalloc^.resetsections;
- objectalloc^.setsection(startsec);
- {$ifdef GDB}
- StartFileLineInfo;
- {$endif GDB}
- TreePass1(hp);
- {$ifdef GDB}
- EndFileLineInfo;
- {$endif GDB}
- { check for undefined labels }
- CheckAsmSymbolListUndefined;
- { set section sizes }
- objectoutput^.setsectionsizes(objectalloc^.secsize);
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- { Pass 2 }
- currpass:=2;
- objectoutput^.defaultsection(startsec);
- {$ifdef GDB}
- StartFileLineInfo;
- {$endif GDB}
- hp:=TreePass2(hp);
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- { end of lists? }
- if not MaybeNextList(hp) then
- break;
- { if not end then write the current objectfile }
- objectoutput^.donewriting;
- { save section for next loop }
- { this leads to a problem if startsec is sec_none !! PM }
- startsec:=objectalloc^.currsec;
- { we will start a new objectfile so reset everything }
- if (hp^.typ=ait_cut) then
- objectoutput^.initwriting(pai_cut(hp)^.place)
- else
- objectoutput^.initwriting(cut_normal);
- { avoid empty files }
- while assigned(hp^.next) and
- (pai(hp^.next)^.typ in [ait_marker,ait_comment,ait_section,ait_cut]) do
- begin
- if pai(hp^.next)^.typ=ait_section then
- startsec:=pai_section(hp^.next)^.sec;
- hp:=pai(hp^.next);
- end;
- hp:=pai(hp^.next);
- { there is a problem if startsec is sec_none !! PM }
- if startsec=sec_none then
- startsec:=sec_code;
- if not MaybeNextList(hp) then
- break;
- {$ifdef GDB}
- EndFileLineInfo;
- {$endif GDB}
- end;
- end;
- procedure ti386binasmlist.writebin;
- procedure addlist(p:paasmoutput);
- begin
- inc(lists);
- list[lists]:=p;
- end;
- begin
- objectalloc^.resetsections;
- objectalloc^.setsection(sec_code);
- objectoutput^.initwriting(cut_normal);
- objectoutput^.defaultsection(sec_code);
- if cs_debuginfo in aktmoduleswitches then
- addlist(debuglist);
- addlist(codesegment);
- addlist(datasegment);
- addlist(consts);
- addlist(rttilist);
- if assigned(resourcestringlist) then
- addlist(resourcestringlist);
- addlist(bsssegment);
- if assigned(importssection) then
- addlist(importssection);
- if assigned(exportssection) and not UseDeffileForExport then
- addlist(exportssection);
- if assigned(resourcesection) then
- addlist(resourcesection);
- if SmartAsm then
- writetreesmart
- else
- writetree;
- { leave if errors have occured }
- if errorcount>0 then
- exit;
- { write last objectfile }
- objectoutput^.donewriting;
- end;
- constructor ti386binasmlist.init(t:togtype;smart:boolean);
- begin
- case t of
- og_none :
- Message(asmw_f_no_binary_writer_selected);
- og_dbg :
- objectoutput:=new(pdbgoutput,init(smart));
- og_coff :
- objectoutput:=new(pdjgppcoffoutput,init(smart));
- og_pecoff :
- objectoutput:=new(pwin32coffoutput,init(smart));
- end;
- objectalloc:=new(pobjectalloc,init);
- SmartAsm:=smart;
- currpass:=0;
- end;
- destructor ti386binasmlist.done;
- begin
- dispose(objectoutput,done);
- dispose(objectalloc,done);
- end;
- end.
- {
- $Log$
- Revision 1.38 2000-02-18 21:54:07 pierre
- * avoid LeText if no stabs info
- Revision 1.37 2000/02/18 12:31:07 pierre
- * Reset file name to empty at end of code section
- Revision 1.36 2000/02/09 13:22:43 peter
- * log truncated
- Revision 1.35 2000/01/20 00:21:49 pierre
- * avoid startsec=sec_none
- Revision 1.34 2000/01/12 10:38:17 peter
- * smartlinking fixes for binary writer
- * release alignreg code and moved instruction writing align to cpuasm,
- but it doesn't use the specified register yet
- Revision 1.33 2000/01/07 01:14:18 peter
- * updated copyright to 2000
- Revision 1.32 1999/12/24 15:22:52 peter
- * reset insentry/lastinsoffset so writing smartlink works correct for
- short jmps
- Revision 1.31 1999/12/22 01:01:46 peter
- - removed freelabel()
- * added undefined label detection in internal assembler, this prevents
- a lot of ld crashes and wrong .o files
- * .o files aren't written anymore if errors have occured
- * inlining of assembler labels is now correct
- Revision 1.30 1999/12/08 10:39:59 pierre
- + allow use of unit var in exports of DLL for win32
- by using direct export writing by default instead of use of DEFFILE
- that does not allow assembler labels that do not
- start with an underscore.
- Use -WD to force use of Deffile for Win32 DLL
- Revision 1.29 1999/12/01 22:05:13 pierre
- * problem with unused external symbol in stabs solved
- Revision 1.28 1999/11/30 10:40:42 peter
- + ttype, tsymlist
- Revision 1.27 1999/11/06 14:34:16 peter
- * truncated log to 20 revs
- Revision 1.26 1999/11/02 15:06:56 peter
- * import library fixes for win32
- * alignment works again
- Revision 1.25 1999/09/26 21:13:40 peter
- * short jmp with alignment problems fixed
- Revision 1.24 1999/08/25 11:59:33 jonas
- * changed pai386, paippc and paiapha (same for tai*) to paicpu (taicpu)
- Revision 1.23 1999/08/10 12:26:21 pierre
- * avoid double .edata section if using DLLTOOL
- Revision 1.22 1999/08/04 00:22:35 florian
- * renamed i386asm and i386base to cpuasm and cpubase
- Revision 1.21 1999/08/01 18:28:09 florian
- * modifications for the new code generator
- Revision 1.20 1999/07/31 12:33:11 peter
- * fixed smartlinking
- Revision 1.19 1999/07/22 09:37:30 florian
- + resourcestring implemented
- + start of longstring support
- }
|