| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495 | {    Copyright (c) 1998-2002 by Florian Klaempfl    Handles the resource files handling    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 comprsrc;{$i fpcdefs.inc}interface  uses    Systems, cstreams;type   tresoutput = (roRES, roOBJ);   tresourcefile = class(TAbstractResourceFile)   private      fname : ansistring;   public      constructor Create(const fn : ansistring);override;      procedure Compile(output: tresoutput; const OutName: ansistring);virtual;      procedure PostProcessResourcefile(const s : ansistring);virtual;      function IsCompiled(const fn : ansistring) : boolean;virtual;      procedure Collect(const fn : ansistring);virtual;   end;      TWinLikeResourceFile = class(tresourcefile)   private      FOut: TCFileStream;      FLastIconID: longint;      FLastCursorID: longint;   public      function IsCompiled(const fn : ansistring) : boolean;override;      procedure Collect(const fn : ansistring);override;   end;procedure CompileResourceFiles;procedure CollectResourceFiles;Var  ResCompiler : String;  RCCompiler  : String;implementationuses  SysUtils,  cutils,cfileutils,cclasses,  Globtype,Globals,Verbose,Fmodule,  Script;  const  GlobalResName = 'fpc-res';{****************************************************************************                              TRESOURCEFILE****************************************************************************}constructor tresourcefile.create(const fn : ansistring);begin  fname:=fn;end;procedure tresourcefile.PostProcessResourcefile(const s : ansistring);beginend;function tresourcefile.IsCompiled(const fn: ansistring): boolean;begin  Result:=CompareText(ExtractFileExt(fn), target_info.resobjext) = 0;end;procedure tresourcefile.Collect(const fn: ansistring);begin  if fn='' then    exit;  fname:=fn;  Compile(roOBJ, ChangeFileExt(fn, target_info.resobjext));end;procedure tresourcefile.compile(output: tresoutput; const OutName: ansistring);  Function SelectBin(Const Bin1,Bin2 : String) : String;    begin    If (Bin1<>'') then      SelectBin:=Bin1    else      SelectBin:=Bin2;    end;  var  respath,  srcfilepath,  preprocessorbin,  s,  bin,  resbin   : TCmdStr;  resfound,  objused  : boolean;begin  if output=roRES then    Bin:=SelectBin(RCCompiler,target_res.rcbin)  else    Bin:=SelectBin(ResCompiler,target_res.resbin);  if bin='' then    exit;  resfound:=false;  if utilsdirectory<>'' then    resfound:=FindFile(utilsprefix+bin+source_info.exeext,utilsdirectory,false,resbin);  if not resfound then    resfound:=FindExe(utilsprefix+bin,false,resbin);  { get also the path to be searched for the windres.h }  respath:=ExtractFilePath(resbin);  if (not resfound) and not(cs_link_nolink in current_settings.globalswitches) then   begin     Message(exec_e_res_not_found);     current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];   end;  srcfilepath:=ExtractFilePath(current_module.mainsource^);  if output=roRES then    begin      s:=target_res.rccmd;      Replace(s,'$RES',maybequoted(OutName));      Replace(s,'$RC',maybequoted(fname));      ObjUsed:=False;    end  else    begin      s:=target_res.rescmd;      ObjUsed:=(pos('$OBJ',s)>0);      Replace(s,'$OBJ',maybequoted(OutName));      Replace(s,'$RES',maybequoted(fname));    end;  { windres doesn't like empty include paths }  if respath='' then    respath:='.';  Replace(s,'$INC',maybequoted(respath));  if (target_res.resbin='windres') then   begin     if (srcfilepath<>'') then       s:=s+' --include '+maybequoted(srcfilepath);     { try to find a preprocessor }     preprocessorbin := respath+'cpp'+source_info.exeext;     if FileExists(preprocessorbin,true) then       s:=s+' --preprocessor='+preprocessorbin;   end;{ Execute the command }  if not (cs_link_nolink in current_settings.globalswitches) then   begin     Message1(exec_i_compilingresource,fname);     Message2(exec_d_resbin_params,resbin,s);     FlushOutput;     try       if ExecuteProcess(resbin,s) <> 0 then       begin         Message(exec_e_error_while_linking);         current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];       end;     except       on E:EOSError do       begin         Message(exec_e_cant_call_linker);         current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];       end     end;    end;  if output=roOBJ then    PostProcessResourcefile(OutName);  { Update asmres when externmode is set }  if cs_link_nolink in current_settings.globalswitches then    AsmRes.AddLinkCommand(resbin,s,'');  if (output=roOBJ) and ObjUsed then    current_module.linkunitofiles.add(OutName,link_always);end;function TWinLikeResourceFile.IsCompiled(const fn: ansistring): boolean;const  ResSignature : array [1..32] of byte =  ($00,$00,$00,$00,$20,$00,$00,$00,$FF,$FF,$00,$00,$FF,$FF,$00,$00,   $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00);var  f : file;  oldfmode : byte;  buf: array[1..32] of byte;  i: longint;begin  Result:=CompareText(ExtractFileExt(fn), target_info.resext) = 0;  if Result or not FileExists(fn, False) then exit;  oldfmode:=Filemode;  Filemode:=0;  assign(f,fn);  reset(f,1);  BlockRead(f, buf, SizeOf(buf), i);  close(f);  Filemode:=oldfmode;    if i<>SizeOf(buf) then    exit;  for i:=1 to 32 do    if buf[i]<>ResSignature[i] then      exit;        Result:=True;end;procedure TWinLikeResourceFile.Collect(const fn: ansistring);const  zeroes: array[1..3] of byte = (0,0,0);type  TResHeader = packed record    DataSize: dword;    HeaderSize: dword;    ResTypeFlag: word;    ResTypeID: word;  end;    PIconHeader = ^TIconHeader;  TIconHeader = packed record    Reserved: word;    wType: word;    wCount: word;  end;    PIconDir = ^TIconDir;  TIconDir = packed record    bWidth: byte;    bHeight: byte;    bColorCount: byte;    bReserved: byte;    wPlanes: word;    wBitCount: word;    lBytesInRes: dword;    wNameOrdinal: word;  end;var  fs: TCFileStream;  i, sz, rsz, MaxIconID, MaxCursorID: longint;  hdr: TResHeader;  P: pointer;  PData: PIconHeader;  PDir: PIconDir;  ResNameBuf: array[0..1] of word;begin  if fn='' then    begin      if FOut<>nil then        begin          FOut.Free;          Compile(roOBJ,ChangeFileExt(fname,target_info.resobjext));        end;    end  else    try      fs:=TCFileStream.Create(fn,fmOpenRead or fmShareDenyNone);      if CStreamError<>0 then        begin          fs.Free;          Comment(V_Error,'Can''t open resource file: '+fn);          Include(current_settings.globalswitches, cs_link_nolink);          exit;        end;      if FOut=nil then        begin          FOut:=TCFileStream.Create(fname,fmCreate);          { writing res signature }          FOut.CopyFrom(fs, 32);        end      else        fs.Seek(32, soFromBeginning);      sz:=fs.Size;      MaxIconID := 0;      MaxCursorID := 0;      repeat        fs.ReadBuffer(hdr, SizeOf(hdr));        FOut.WriteBuffer(hdr, SizeOf(hdr));        rsz:=hdr.HeaderSize + hdr.DataSize - SizeOf(hdr);        if fs.Position + rsz > sz then          begin            Comment(V_Error,'Invalid resource file: '+fn);            Include(current_settings.globalswitches, cs_link_nolink);            fs.Free;            exit;          end;        { Adjusting cursor and icon IDs }        if hdr.ResTypeFlag = $FFFF then       { resource type is ordinal }          case hdr.ResTypeID of            1, 3:              { cursor or icon resource }              begin                fs.ReadBuffer(ResNameBuf, SizeOf(ResNameBuf));                if ResNameBuf[0] = $FFFF then   { resource name is ordinal }                  if hdr.ResTypeID = 1 then                    begin                      if ResNameBuf[1] > MaxCursorID then                        MaxCursorID:=ResNameBuf[1];                      Inc(ResNameBuf[1], FLastCursorID);                    end                  else                    begin                      if ResNameBuf[1] > MaxIconID then                        MaxIconID:=ResNameBuf[1];                      Inc(ResNameBuf[1], FLastIconID);                    end;                FOut.WriteBuffer(ResNameBuf, SizeOf(ResNameBuf));                Dec(rsz, SizeOf(ResNameBuf));              end;            12, 14:              { cursor or icon group resource }              begin                GetMem(P, rsz);                fs.ReadBuffer(P^, rsz);                PData := PIconHeader(P + hdr.HeaderSize - sizeof(hdr));                PDir := PIconDir(Pointer(PData) + sizeof(TIconHeader));                for i := 0 to PData^.wCount-1 do                  begin                    if hdr.ResTypeID = 12 then                      Inc(PDir^.wNameOrdinal, FLastCursorID)                    else                      Inc(PDir^.wNameOrdinal, FLastIconID);                    Inc(PDir);                  end;                FOut.WriteBuffer(P^, rsz);                rsz:=0;                FreeMem(P);              end;          end;        { copy rest of the resource data }        FOut.CopyFrom(fs, rsz);        { align resource to dword }        i:=4 - FOut.Position mod 4;        if i<4 then          FOut.WriteBuffer(zeroes, i);        { position to the next resource }        i:=4 - fs.Position mod 4;        if i<4 then          fs.Seek(i, soFromCurrent);      until fs.Position + SizeOf(hdr) >= sz;      fs.Free;      Inc(FLastCursorID, MaxCursorID);      Inc(FLastIconID, MaxIconID);    except      on E:EOSError do begin        Comment(V_Error,'Error processing resource file: '+fn+': '+E.Message);        Include(current_settings.globalswitches, cs_link_nolink);      end;    end;end;procedure CompileResourceFiles;var  resourcefile : tresourcefile;  res: TCmdStrListItem;  p,s : TCmdStr;  src,dst : TCFileStream;  outfmt : tresoutput;begin  { OS/2 (EMX) must be processed elsewhere (in the linking/binding stage).    same with MacOS}  if target_info.system in [system_i386_os2,system_i386_emx,system_powerpc_macos] then exit;  p:=ExtractFilePath(current_module.mainsource^);  res:=TCmdStrListItem(current_module.ResourceFiles.First);  while res<>nil do    begin      if target_info.res=res_none then        Message(scan_e_resourcefiles_not_supported);      s:=res.FPStr;      if not path_absolute(s) then        s:=p+s;      resourcefile:=TResourceFile(resinfos[target_info.res]^.resourcefileclass.create(s));      if resourcefile.IsCompiled(s) then        begin          resourcefile.free;          if CompareText(current_module.outputpath^, p) <> 0 then            begin              { Copy .res file to units output dir }              res.FPStr:=ExtractFileName(res.FPStr);              src:=TCFileStream.Create(s,fmOpenRead or fmShareDenyNone);              if CStreamError<>0 then                begin                  Comment(V_Error,'Can''t open resource file: '+src.FileName);                  Include(current_settings.globalswitches, cs_link_nolink);                  exit;                end;              dst:=TCFileStream.Create(current_module.outputpath^+res.FPStr,fmCreate);              if CStreamError<>0 then                begin                  Comment(V_Error,'Can''t create resource file: '+dst.FileName);                  Include(current_settings.globalswitches, cs_link_nolink);                  exit;                end;              dst.CopyFrom(src,src.Size);              dst.Free;              src.Free;            end;        end      else        begin          res.FPStr:=ExtractFileName(res.FPStr);          if target_res.rcbin='' then            begin              { if target does not have .rc to .res compiler, create obj }              outfmt:=roOBJ;              res.FPStr:=ChangeFileExt(res.FPStr,target_info.resobjext);            end          else            begin              outfmt:=roRES;              res.FPStr:=ChangeFileExt(res.FPStr,target_info.resext);            end;          resourcefile.compile(outfmt, current_module.outputpath^+res.FPStr);          resourcefile.free;        end;      res:=TCmdStrListItem(res.Next);    end;end;procedure CollectResourceFiles;var  resourcefile : tresourcefile;    procedure ProcessModule(u : tmodule);  var    res : TCmdStrListItem;    s   : TCmdStr;  begin    res:=TCmdStrListItem(u.ResourceFiles.First);    while assigned(res) do      begin        if path_absolute(res.FPStr) then          s:=res.FPStr        else          begin            s:=u.path^+res.FPStr;            if not FileExists(s,True) then              s:=u.outputpath^+res.FPStr;          end;        resourcefile.Collect(s);        res:=TCmdStrListItem(res.Next);      end;  end;  var  hp : tused_unit;  s : TCmdStr;begin  if (target_info.res=res_none) or (target_res.rcbin='') then    exit;  if cs_link_nolink in current_settings.globalswitches then    exit;  s:=main_module.outputpath^+GlobalResName+target_info.resext;  resourcefile:=TResourceFile(resinfos[target_info.res]^.resourcefileclass.create(s));  hp:=tused_unit(usedunits.first);  while assigned(hp) do    begin      ProcessModule(hp.u);      hp:=tused_unit(hp.next);    end;  ProcessModule(current_module);  { Finish collection }  resourcefile.Collect('');  resourcefile.free;end;end.
 |