123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419 |
- {
- Copyright (c) 2020 by Free Pascal Development Team
- This unit implements support import, export, link routines
- for the m68k Sinclair QL target
- 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.
- ****************************************************************************
- }
- unit t_sinclairql;
- {$i fpcdefs.inc}
- interface
- uses
- rescmn, comprsrc, link;
- type
- PLinkerSinclairQL = ^TLinkerSinclairQL;
- TLinkerSinclairQL = class(texternallinker)
- private
- Origin: DWord;
- UseVLink: boolean;
- function WriteResponseFile(isdll: boolean): boolean;
- procedure SetSinclairQLInfo;
- function MakeSinclairQLExe: boolean;
- public
- constructor Create; override;
- procedure SetDefaultInfo; override;
- procedure InitSysInitUnitName; override;
- function MakeExecutable: boolean; override;
- end;
- implementation
- uses
- sysutils,cutils,cfileutl,cclasses,aasmbase,
- globtype,globals,systems,verbose,cscript,fmodule,i_sinclairql;
- type
- TQLHeader = packed record
- hdr_id: array[0..17] of char;
- hdr_reserved: byte;
- hdr_length: byte;
- hdr_access: byte;
- hdr_type: byte;
- hdr_data: dword;
- hdr_extra: dword;
- end;
- TXTccData = packed record
- xtcc_id: array[0..3] of char;
- xtcc_data: dword;
- end;
- const
- DefaultQLHeader: TQLHeader = (
- hdr_id: ']!QDOS File Header';
- hdr_reserved: 0;
- hdr_length: $f;
- hdr_access: 0;
- hdr_type: 1;
- hdr_data: 0;
- hdr_extra: 0;
- );
- DefaultXTccData: TXTCCData = (
- xtcc_id: 'XTcc';
- xtcc_data: 0;
- );
- const
- DefaultOrigin = $0;
- ProgramHeaderName = 'main';
- constructor TLinkerSinclairQL.Create;
- begin
- UseVLink:=(cs_link_vlink in current_settings.globalswitches);
- Inherited Create;
- { allow duplicated libs (PM) }
- SharedLibFiles.doubles:=true;
- StaticLibFiles.doubles:=true;
- end;
- procedure TLinkerSinclairQL.SetSinclairQLInfo;
- begin
- if ImageBaseSetExplicity then
- Origin:=ImageBase
- else
- Origin:=DefaultOrigin;
- with Info do
- begin
- if not UseVLink then
- begin
- ExeCmd[1]:='ld $DYNLINK $OPT -d -n -o $EXE $RES';
- end
- else
- begin
- ExeCmd[1]:='vlink $QLFLAGS $FLAGS $GCSECTIONS $OPT $STRIP $MAP -o $EXE -T $RES';
- end;
- end;
- end;
- procedure TLinkerSinclairQL.SetDefaultInfo;
- begin
- if target_info.system = system_m68k_sinclairql then
- SetSinclairQLInfo;
- end;
- procedure TLinkerSinclairQL.InitSysInitUnitName;
- begin
- sysinitunit:='si_prc';
- end;
- function TLinkerSinclairQL.WriteResponseFile(isdll: boolean): boolean;
- var
- linkres : TLinkRes;
- HPath : TCmdStrListItem;
- s : string;
- begin
- WriteResponseFile:=False;
- { Open link.res file }
- LinkRes:=TLinkRes.Create(outputexedir+Info.ResName,true);
- if UseVLink and (source_info.dirsep <> '/') then
- LinkRes.fForceUseForwardSlash:=true;
- { Write path to search libraries }
- HPath:=TCmdStrListItem(current_module.locallibrarysearchpath.First);
- while assigned(HPath) do
- begin
- s:=HPath.Str;
- if (cs_link_on_target in current_settings.globalswitches) then
- s:=ScriptFixFileName(s);
- LinkRes.Add('-L'+s);
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- HPath:=TCmdStrListItem(LibrarySearchPath.First);
- while assigned(HPath) do
- begin
- s:=HPath.Str;
- if s<>'' then
- LinkRes.Add('SEARCH_DIR("'+s+'")');
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- LinkRes.Add('INPUT (');
- { add objectfiles, start with prt0 always }
- if not (target_info.system in systems_internal_sysinit) then
- begin
- s:=FindObjectFile('prt0','',false);
- LinkRes.AddFileName(maybequoted(s));
- end;
- while not ObjectFiles.Empty do
- begin
- s:=ObjectFiles.GetFirst;
- if s<>'' then
- begin
- { vlink doesn't use SEARCH_DIR for object files }
- if UseVLink then
- s:=FindObjectFile(s,'',false);
- LinkRes.AddFileName(maybequoted(s));
- end;
- end;
- { Write staticlibraries }
- if not StaticLibFiles.Empty then
- begin
- { vlink doesn't need, and doesn't support GROUP }
- if not UseVLink then
- begin
- LinkRes.Add(')');
- LinkRes.Add('GROUP(');
- end;
- while not StaticLibFiles.Empty do
- begin
- S:=StaticLibFiles.GetFirst;
- LinkRes.AddFileName(maybequoted(s));
- end;
- end;
- LinkRes.Add(')');
- with LinkRes do
- begin
- Add('');
- Add('PHDRS {');
- Add(' '+ProgramHeaderName+' PT_LOAD;');
- Add('}');
- Add('SECTIONS');
- Add('{');
- Add(' . = 0x'+hexstr(Origin,8)+';');
- Add(' .text : {');
- Add(' _stext = .;');
- Add(' *(.text .text.* )');
- Add(' *(.data .data.* .rodata .rodata.* .fpc.* )');
- Add(' *(.stack .stack.*)');
- { force the end of section to be word aligned }
- Add(' . = ALIGN(2); SHORT(0x514C);');
- Add(' _etext = .;');
- Add(' } :'+ProgramHeaderName);
- Add(' .bss (NOLOAD): {');
- Add(' _sbss = .;');
- Add(' *(.bss .bss.*)');
- Add(' . = ALIGN(2); SHORT(0x0000);');
- Add(' _ebss = .;');
- Add(' } :'+ProgramHeaderName);
- Add('}');
- end;
- { Write and Close response }
- linkres.writetodisk;
- linkres.free;
- WriteResponseFile:=True;
- end;
- function TLinkerSinclairQL.MakeSinclairQLExe: boolean;
- var
- BinStr,
- CmdStr : TCmdStr;
- StripStr: string[40];
- DynLinkStr : ansistring;
- GCSectionsStr : string;
- FlagsStr : string;
- QLFlagsStr: string;
- MapStr : string;
- ExeName: string;
- fd,fs: file;
- fhdr: text;
- buf: pointer;
- bufread,bufsize: longint;
- HdrName: string;
- HeaderLine: string;
- HeaderSize: longint;
- code: word;
- QLHeader: TQLHeader;
- XTccData: TXTccData;
- BinSize: longint;
- RelocSize: longint;
- DataSpace: DWord;
- begin
- StripStr:='';
- GCSectionsStr:='';
- DynLinkStr:='';
- FlagsStr:='';
- QLFlagsStr:='';
- MapStr:='';
- if (cs_link_map in current_settings.globalswitches) then
- MapStr:='-M'+maybequoted(ScriptFixFilename(current_module.mapfilename));
- if (cs_link_strip in current_settings.globalswitches) then
- StripStr:='-s';
- if rlinkpath<>'' then
- DynLinkStr:='--rpath-link '+rlinkpath;
- if UseVLink then
- begin
- if create_smartlink_sections then
- GCSectionsStr:='-gc-all';
- if sinclairql_vlink_experimental then
- QLFlagsStr:='-b sinclairql -q -'+lower(sinclairql_metadata_format)+' -stack='+tostr(StackSize)
- else
- QLFlagsStr:='-b rawseg -q';
- end;
- ExeName:=current_module.exefilename;
- HdrName:=ExeName+'.hdr';
- { Call linker }
- SplitBinCmd(Info.ExeCmd[1],BinStr,CmdStr);
- binstr:=FindUtil(utilsprefix+BinStr);
- Replace(cmdstr,'$OPT',Info.ExtraOptions);
- Replace(cmdstr,'$EXE',maybequoted(ScriptFixFileName(ExeName)));
- Replace(cmdstr,'$RES',maybequoted(ScriptFixFileName(outputexedir+Info.ResName)));
- Replace(cmdstr,'$MAP',MapStr);
- Replace(cmdstr,'$FLAGS',FlagsStr);
- Replace(cmdstr,'$STRIP',StripStr);
- Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
- Replace(cmdstr,'$DYNLINK',DynLinkStr);
- Replace(cmdstr,'$QLFLAGS',QLFlagsStr);
- MakeSinclairQLExe:=DoExec(BinStr,CmdStr,true,false);
- { Kludge:
- With the above linker script, vlink will produce two files. The main binary
- and the relocation info. Here we copy the two together. (KB) }
- if MakeSinclairQLExe and not sinclairql_vlink_experimental then
- begin
- QLHeader:=DefaultQLHeader;
- XTccData:=DefaultXTccData;
- BinSize:=0;
- RelocSize:=0;
- bufsize:=16384;
- {$push}
- {$i-}
- { Rename vlink's output file into the header file it is, then parse the
- expected length from it. Later we use either this size or the final binary
- size in the BASIC loader, depending on which one is bigger. (KB) }
- RenameFile(ExeName,HdrName);
- assign(fhdr,HdrName);
- reset(fhdr);
- readln(fhdr,HeaderLine);
- Val(Copy(HeaderLine,RPos('0x',HeaderLine),Length(HeaderLine)),HeaderSize,code);
- close(fhdr);
- buf:=GetMem(bufsize);
- assign(fd,ExeName);
- rewrite(fd,1);
- assign(fs,ExeName+'.'+ProgramHeaderName+'.rel'+ProgramHeaderName);
- reset(fs,1);
- RelocSize := FileSize(fs);
- close(fs);
- assign(fs,ExeName+'.'+ProgramHeaderName);
- reset(fs,1);
- BinSize := FileSize(fs);
- { We assume .bss size is total size indicated by linker minus emmited binary.
- DataSpace size is .bss + stack space }
- DataSpace := NToBE(DWord(max((HeaderSize - BinSize) - RelocSize + StackSize,0)));
- { Option: prepend QEmuLator and QPC2 v5 compatible header to EXE }
- if sinclairql_metadata_format='QHDR' then
- begin
- QLHeader.hdr_data:=DataSpace;
- blockwrite(fd, QLHeader, sizeof(QLHeader));
- end;
- repeat
- blockread(fs,buf^,bufsize,bufread);
- blockwrite(fd,buf^,bufread);
- until eof(fs);
- close(fs);
- // erase(fs);
- assign(fs,ExeName+'.'+ProgramHeaderName+'.rel'+ProgramHeaderName);
- reset(fs,1);
- repeat
- blockread(fs,buf^,bufsize,bufread);
- blockwrite(fd,buf^,bufread);
- until eof(fs);
- close(fs);
- // erase(fs);
- { Option: append cross compilation data space marker, this can be picked up by
- a special version of InfoZIP (compiled with -DQLZIP and option -Q) or by any
- of the XTcc unpack utilities }
- if sinclairql_metadata_format='XTCC' then
- begin
- XTccData.xtcc_data:=DataSpace;
- blockwrite(fd, XTccData, sizeof(XTccData));
- end;
- close(fd);
- {$pop}
- FreeMem(buf);
- MakeSinclairQLExe:=(code = 0) and not (BinSize = 0) and (IOResult = 0);
- end;
- end;
- function TLinkerSinclairQL.MakeExecutable:boolean;
- var
- success : boolean;
- bootfile : TScript;
- ExeName: String;
- begin
- if not(cs_link_nolink in current_settings.globalswitches) then
- Message1(exec_i_linking,current_module.exefilename);
- { Write used files and libraries }
- WriteResponseFile(false);
- success:=MakeSinclairQLExe;
- { Remove ReponseFile }
- if (success) and not(cs_link_nolink in current_settings.globalswitches) then
- DeleteFile(outputexedir+Info.ResName);
- MakeExecutable:=success; { otherwise a recursive call to link method }
- end;
- {*****************************************************************************
- Initialize
- *****************************************************************************}
- initialization
- RegisterLinker(ld_sinclairql,TLinkerSinclairQL);
- RegisterTarget(system_m68k_sinclairql_info);
- end.
|