mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 06:12:35 +01:00 
			
		
		
		
	 639a59df92
			
		
	
	
		639a59df92
		
	
	
	
	
		
			
			For partial specialization only the declaration is reparsed, but not method bodies.
The way generic parameters are passed around inside the compiler is changed: instead of creating new type symbols we keep a (name,def) pair so that the code in insert_generic_parameter_types can decide whether it needs to add a type symbol (for new undefined defs) or not (for real types and undefined defs that were passed on from the parent generic). This required the tfpobjectlist type of the genericlist variables/parameters to be changed to tfphashobjectlist.
For correctly parsing Delphi specializations as parameters in functions of records (or objects) the relationship between the def and its typesym must already be established during the parsing. For this the checks for forcing a "type is not completely defined" message needed to be adjusted to correctly handle nested types as well. This should as a sideeffect also allow the usage of nested constants, etc like was fixed for classes some months ago.
ToDo: 
  - if a generic is specialized with only fully defined types then we could generate the in the unit where it's used. This is not yet done.
  - currently we don't specialize generics that are currently parsed; maybe this could be improved in the future for better type compatibility checks
  - check whether the pausing of token recording for partial specializations works correct in context of hint modifiers
pgenutil.pas:
  * parse_generic_parameters: return a tfphashobjectlist instead of a tfpobjectlist (requires a few type adjustments in various other declarations)
  * maybe_insert_generic_rename_symbol, insert_generic_parameter_types: change genericlist from tfpobjectlist to tfphashobjectlist
  * parse_generic_specialization_types_internal: use is_generic instead of checking for df_generic
  * generate_specialization:
      + add a nested function to disable the requirement to check for method bodies
      * use the "simple" parameter parsing only for error recovery
      * instead of already creating a new type symbol for a parameter we use the found symbol's name and its def and maybe create it later on (therefor the type of tfpobjectlist was changed to tfphashobjectlist)
      * a partial specialization is specialized into the symtable of the def it is specialized in instead of one of the two global symtables
      * for now we handle partial specializations of generics we are currently parsing like before
      * don't continue recording generic tokens while we do a partial specialization
      * use the new unset_forwarddef function on the newly created defs
  * insert_generic_parameter_types: only create a new type symbol if the found type symbol does not yet have an owner (thus was freshly created for this generic declaration)
pdecobj.pas, object_dec:
  * change type of genericlist from tfpobjectlist to tfphashobjectlist
  * set the type sym for all object types that can be generic or inside a generic (needed for correctly parsing Delphi style generic declarations)
pdecsub.pas, parse_proc_head:
  * consume_generic_interface: always generate the specialization name as now all generics are "specialized" inside a generic
  * the assumption that the def index numbers are the same is no longer true as the genericdef might contain the defs of partial specializations which are not generated for full specializations
pdecvar.pas, read_record_fields:
  * we also need to check nested types whether they contain a not yet completely parsed record or object
ptype.pas:
  * read_named_type: 
      * change genericlist from tfpobjectlist to tfphashobjectlist
      * pass the typesymbol along to record_dec
  * resolve_forward_types: use is_generic instead of checking for df_generic
  * single_type: 
      * use is_generic instead of checking for df_generic
      * no need to check generic parameters
  * parse_record_members:
      + add parameter for the record's type symbol
      * setup the typesym <=> def relationship
  + record_dec: add parameter for the type symbol and pass it to parse_record_members
  * read_named_type, expr_type: use is_generic instead of checking for df_generic
  * array_dec & procvar_dec: change genericlist from tfpobjectlist to tfphashobjectlist
symdef.pas, tstoreddef:
  * improve the checks used in is_generic and is_specialization to really only work on true generics and true (and partial) specializations respectively
  * don't search the type parameters in the symtable, but store them in the PPU and load them from there
  - remove fillgenericparas method (including the calls in the descendants tarraydef, tprocvardef, tobjectdef and trecorddef)
defcmp.pas, compare_defs_ext:
  * handle partial specializations: specializations with only undefineddefs are compatible to generic defs
pdecl.pas, types_dec:
  * switch generictypelist from tfpobjectlist to tfphashobjectlist
ppu.pas:
  * increase PPU version
+ added tests that ensure that "not completely defined" checks for records (and objects) still work correctly
git-svn-id: trunk@27861 -
		
	
			
		
			
				
	
	
		
			1383 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			1383 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     Copyright (c) 1998-2002 by Florian Klaempfl
 | |
| 
 | |
|     Routines to read/write ppu files
 | |
| 
 | |
|     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 ppu;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
|   uses
 | |
|     systems,globtype,constexp,cstreams;
 | |
| 
 | |
| { Also write the ppu if only crc if done, this can be used with ppudump to
 | |
|   see the differences between the intf and implementation }
 | |
| { define INTFPPU}
 | |
| 
 | |
| {$ifdef Test_Double_checksum}
 | |
| var
 | |
|   CRCFile : text;
 | |
| const
 | |
|   CRC_array_Size = 200000;
 | |
| type
 | |
|   tcrc_array = array[0..crc_array_size] of dword;
 | |
|   pcrc_array = ^tcrc_array;
 | |
| {$endif Test_Double_checksum}
 | |
| 
 | |
| const
 | |
|   CurrentPPUVersion = 170;
 | |
| 
 | |
| { buffer sizes }
 | |
|   maxentrysize = 1024;
 | |
|   ppubufsize   = 16384;
 | |
| 
 | |
| {ppu entries}
 | |
|   mainentryid         = 1;
 | |
|   subentryid          = 2;
 | |
|   {special}
 | |
|   iberror             = 0;
 | |
|   ibstartdefs         = 248;
 | |
|   ibenddefs           = 249;
 | |
|   ibstartsyms         = 250;
 | |
|   ibendsyms           = 251;
 | |
|   ibendinterface      = 252;
 | |
|   ibendimplementation = 253;
 | |
| //  ibendbrowser        = 254;
 | |
|   ibend               = 255;
 | |
|   {general}
 | |
|   ibmodulename           = 1;
 | |
|   ibsourcefiles          = 2;
 | |
|   ibloadunit             = 3;
 | |
|   ibinitunit             = 4;
 | |
|   iblinkunitofiles       = 5;
 | |
|   iblinkunitstaticlibs   = 6;
 | |
|   iblinkunitsharedlibs   = 7;
 | |
|   iblinkotherofiles      = 8;
 | |
|   iblinkotherstaticlibs  = 9;
 | |
|   iblinkothersharedlibs  = 10;
 | |
|   ibImportSymbols        = 11;
 | |
|   ibsymref               = 12;
 | |
|   ibdefref               = 13;
 | |
| //  ibendsymtablebrowser   = 14;
 | |
| //  ibbeginsymtablebrowser = 15;
 | |
| {$IFDEF MACRO_DIFF_HINT}
 | |
|   ibusedmacros           = 16;
 | |
| {$ENDIF}
 | |
|   ibderefdata            = 17;
 | |
|   ibexportedmacros       = 18;
 | |
|   ibderefmap             = 19;
 | |
|   {syms}
 | |
|   ibtypesym        = 20;
 | |
|   ibprocsym        = 21;
 | |
|   ibstaticvarsym   = 22;
 | |
|   ibconstsym       = 23;
 | |
|   ibenumsym        = 24;
 | |
| //  ibtypedconstsym  = 25;
 | |
|   ibabsolutevarsym = 26;
 | |
|   ibpropertysym    = 27;
 | |
|   ibfieldvarsym    = 28;
 | |
|   ibunitsym        = 29;
 | |
|   iblabelsym       = 30;
 | |
|   ibsyssym         = 31;
 | |
|   ibnamespacesym   = 32;
 | |
|   iblocalvarsym    = 33;
 | |
|   ibparavarsym     = 34;
 | |
|   ibmacrosym       = 35;
 | |
|   {definitions}
 | |
|   iborddef         = 40;
 | |
|   ibpointerdef     = 41;
 | |
|   ibarraydef       = 42;
 | |
|   ibprocdef        = 43;
 | |
|   ibshortstringdef = 44;
 | |
|   ibrecorddef      = 45;
 | |
|   ibfiledef        = 46;
 | |
|   ibformaldef      = 47;
 | |
|   ibobjectdef      = 48;
 | |
|   ibenumdef        = 49;
 | |
|   ibsetdef         = 50;
 | |
|   ibprocvardef     = 51;
 | |
|   ibfloatdef       = 52;
 | |
|   ibclassrefdef    = 53;
 | |
|   iblongstringdef  = 54;
 | |
|   ibansistringdef  = 55;
 | |
|   ibwidestringdef  = 56;
 | |
|   ibvariantdef     = 57;
 | |
|   ibundefineddef   = 58;
 | |
|   ibunicodestringdef = 59;
 | |
|   {implementation/ObjData}
 | |
|   ibnodetree       = 80;
 | |
|   ibasmsymbols     = 81;
 | |
|   ibresources      = 82;
 | |
|   ibcreatedobjtypes = 83;
 | |
|   ibwpofile         = 84;
 | |
|   ibmoduleoptions   = 85;
 | |
| 
 | |
|   ibmainname       = 90;
 | |
|   ibsymtableoptions = 91;
 | |
|   ibrecsymtableoptions = 91;
 | |
|   { target-specific things }
 | |
|   iblinkotherframeworks = 100;
 | |
|   ibjvmnamespace = 101;
 | |
| 
 | |
| { unit flags }
 | |
|   uf_init                = $000001; { unit has initialization section }
 | |
|   uf_finalize            = $000002; { unit has finalization section   }
 | |
|   uf_big_endian          = $000004;
 | |
| //uf_has_browser         = $000010;
 | |
|   uf_in_library          = $000020; { is the file in another file than <ppufile>.* ? }
 | |
|   uf_smart_linked        = $000040; { the ppu can be smartlinked }
 | |
|   uf_static_linked       = $000080; { the ppu can be linked static }
 | |
|   uf_shared_linked       = $000100; { the ppu can be linked shared }
 | |
| //uf_local_browser       = $000200;
 | |
|   uf_no_link             = $000400; { unit has no .o generated, but can still have external linking! }
 | |
|   uf_has_resourcestrings = $000800; { unit has resource string section }
 | |
|   uf_little_endian       = $001000;
 | |
|   uf_release             = $002000; { unit was compiled with -Ur option }
 | |
|   uf_threadvars          = $004000; { unit has threadvars }
 | |
|   uf_fpu_emulation       = $008000; { this unit was compiled with fpu emulation on }
 | |
|   uf_has_stabs_debuginfo = $010000; { this unit has stabs debuginfo generated }
 | |
|   uf_local_symtable      = $020000; { this unit has a local symtable stored }
 | |
|   uf_uses_variants       = $040000; { this unit uses variants }
 | |
|   uf_has_resourcefiles   = $080000; { this unit has external resources (using $R directive)}
 | |
|   uf_has_exports         = $100000; { this module or a used unit has exports }
 | |
|   uf_has_dwarf_debuginfo = $200000; { this unit has dwarf debuginfo generated }
 | |
|   uf_wideinits           = $400000; { this unit has winlike widestring typed constants }
 | |
|   uf_classinits          = $800000; { this unit has class constructors/destructors }
 | |
|   uf_resstrinits        = $1000000; { this unit has string consts referencing resourcestrings }
 | |
|   uf_i8086_far_code     = $2000000; { this unit uses an i8086 memory model with far code (i.e. medium, large or huge) }
 | |
|   uf_i8086_far_data     = $4000000; { this unit uses an i8086 memory model with far data (i.e. compact or large) }
 | |
|   uf_i8086_huge_data    = $8000000; { this unit uses an i8086 memory model with huge data (i.e. huge) }
 | |
|   uf_i8086_cs_equals_ds = $10000000; { this unit uses an i8086 memory model with CS=DS (i.e. tiny) }
 | |
| 
 | |
| {$ifdef generic_cpu}
 | |
| { We need to use the correct size of aint and pint for
 | |
|   the target CPU }
 | |
| const
 | |
|   CpuAddrBitSize : array[tsystemcpu] of longint =
 | |
|     (
 | |
|     {  0 } 32 {'none'},
 | |
|     {  1 } 32 {'i386'},
 | |
|     {  2 } 32 {'m68k'},
 | |
|     {  3 } 32 {'alpha'},
 | |
|     {  4 } 32 {'powerpc'},
 | |
|     {  5 } 32 {'sparc'},
 | |
|     {  6 } 32 {'vis'},
 | |
|     {  7 } 64 {'ia64'},
 | |
|     {  8 } 64 {'x86_64'},
 | |
|     {  9 } 32 {'mipseb'},
 | |
|     { 10 } 32 {'arm'},
 | |
|     { 11 } 64 {'powerpc64'},
 | |
|     { 12 } 16 {'avr'},
 | |
|     { 13 } 32 {'mipsel'},
 | |
|     { 14 } 32 {'jvm'},
 | |
|     { 15 } 16 {'i8086'}
 | |
|     );
 | |
|   CpuAluBitSize : array[tsystemcpu] of longint =
 | |
|     (
 | |
|     {  0 } 32 {'none'},
 | |
|     {  1 } 32 {'i386'},
 | |
|     {  2 } 32 {'m68k'},
 | |
|     {  3 } 32 {'alpha'},
 | |
|     {  4 } 32 {'powerpc'},
 | |
|     {  5 } 32 {'sparc'},
 | |
|     {  6 } 32 {'vis'},
 | |
|     {  7 } 64 {'ia64'},
 | |
|     {  8 } 64 {'x86_64'},
 | |
|     {  9 } 32 {'mipseb'},
 | |
|     { 10 } 32 {'arm'},
 | |
|     { 11 } 64 {'powerpc64'},
 | |
|     { 12 }  8 {'avr'},
 | |
|     { 13 } 32 {'mipsel'},
 | |
|     { 14 } 64 {'jvm'},
 | |
|     { 15 } 16 {'i8086'}
 | |
|     );
 | |
| {$endif generic_cpu}
 | |
| 
 | |
| type
 | |
|   { bestreal is defined based on the target architecture }
 | |
|   ppureal=bestreal;
 | |
| 
 | |
|   tppuerror=(ppuentrytoobig,ppuentryerror);
 | |
| 
 | |
|   tppuheader=record
 | |
|     id       : array[1..3] of char; { = 'PPU' }
 | |
|     ver      : array[1..3] of char;
 | |
|     compiler : word;
 | |
|     cpu      : word;
 | |
|     target   : word;
 | |
|     flags    : longint;
 | |
|     size     : longint; { size of the ppufile without header }
 | |
|     checksum : cardinal; { checksum for this ppufile }
 | |
|     interface_checksum : cardinal;
 | |
|     deflistsize,
 | |
|     symlistsize : longint;
 | |
|     indirect_checksum: cardinal;
 | |
|   end;
 | |
| 
 | |
|   tppuentry=packed record
 | |
|     size : longint;
 | |
|     id   : byte;
 | |
|     nr   : byte;
 | |
|   end;
 | |
| 
 | |
|   { tppufile }
 | |
| 
 | |
|   tppufile=class
 | |
|   private
 | |
|     f        : TCCustomFileStream;
 | |
|     mode     : byte; {0 - Closed, 1 - Reading, 2 - Writing}
 | |
|     fname    : string;
 | |
|     fsize    : integer;
 | |
| {$ifdef Test_Double_checksum}
 | |
|   public
 | |
|     crcindex,
 | |
|     crc_index,
 | |
|     crcindex2,
 | |
|     crc_index2 : cardinal;
 | |
|     crc_test,
 | |
|     crc_test2  : pcrc_array;
 | |
|   private
 | |
| {$endif def Test_Double_checksum}
 | |
|     buf      : pchar;
 | |
|     bufstart,
 | |
|     bufsize,
 | |
|     bufidx   : integer;
 | |
|     entrybufstart,
 | |
|     entrystart,
 | |
|     entryidx : integer;
 | |
|     entry    : tppuentry;
 | |
|     closed,
 | |
|     tempclosed : boolean;
 | |
|     closepos : integer;
 | |
|   public
 | |
|     entrytyp : byte;
 | |
|     header           : tppuheader;
 | |
|     size             : integer;
 | |
|     change_endian    : boolean; { Used in ppudump util }
 | |
|     { crc for the entire unit }
 | |
|     crc,
 | |
|     { crc for the interface definitions in this unit }
 | |
|     interface_crc,
 | |
|     { crc of all object/class definitions in the interface of this unit, xor'ed
 | |
|       by the crc's of all object/class definitions in the interfaces of units
 | |
|       used by this unit. Reason: see mantis #13840 }
 | |
|     indirect_crc     : cardinal;
 | |
|     error,
 | |
| {$ifdef generic_cpu}
 | |
|     has_more,
 | |
| {$endif not generic_cpu}
 | |
|     do_crc,
 | |
|     do_interface_crc,
 | |
|     do_indirect_crc  : boolean;
 | |
|     crc_only         : boolean;    { used to calculate interface_crc before implementation }
 | |
|     constructor Create(const fn:string);
 | |
|     destructor  Destroy;override;
 | |
|     procedure flush;
 | |
|     procedure closefile;
 | |
|     function  CheckPPUId:boolean;
 | |
|     function  GetPPUVersion:integer;
 | |
|     procedure NewHeader;
 | |
|     procedure NewEntry;
 | |
|   {read}
 | |
|     function  openfile:boolean;
 | |
|     procedure reloadbuf;
 | |
|     procedure readdata(out b;len:integer);
 | |
|     procedure skipdata(len:integer);
 | |
|     function  readentry:byte;
 | |
|     function  EndOfEntry:boolean;
 | |
|     function  entrysize:longint;
 | |
|     function  entryleft:longint;
 | |
|     procedure getdatabuf(out b;len:integer;out res:integer);
 | |
|     procedure getdata(out b;len:integer);
 | |
|     function  getbyte:byte;
 | |
|     function  getword:word;
 | |
|     function  getdword:dword;
 | |
|     function  getlongint:longint;
 | |
|     function getint64:int64;
 | |
|     function  getqword:qword;
 | |
|     function getaint:aint;
 | |
|     function getasizeint:asizeint;
 | |
|     function getaword:aword;
 | |
|     function  getreal:ppureal;
 | |
|     function  getrealsize(sizeofreal : longint):ppureal;
 | |
|     function  getstring:string;
 | |
|     function  getansistring:ansistring;
 | |
|     procedure getnormalset(out b);
 | |
|     procedure getsmallset(out b);
 | |
|     function  skipuntilentry(untilb:byte):boolean;
 | |
|   {write}
 | |
|     function  createfile:boolean;
 | |
|     procedure writeheader;
 | |
|     procedure writebuf;
 | |
|     procedure writedata(const b;len:integer);
 | |
|     procedure writeentry(ibnr:byte);
 | |
|     procedure putdata(const b;len:integer);
 | |
|     procedure putbyte(b:byte);
 | |
|     procedure putword(w:word);
 | |
|     procedure putdword(w:dword);
 | |
|     procedure putlongint(l:longint);
 | |
|     procedure putint64(i:int64);
 | |
|     procedure putqword(q:qword);
 | |
|     procedure putaint(i:aint);
 | |
|     procedure putasizeint(i:asizeint);
 | |
|     procedure putaword(i:aword);
 | |
|     procedure putreal(d:ppureal);
 | |
|     procedure putstring(const s:string);
 | |
|     procedure putansistring(const s:ansistring);
 | |
|     procedure putnormalset(const b);
 | |
|     procedure putsmallset(const b);
 | |
|     procedure tempclose;        // MG: not used, obsolete?
 | |
|     function  tempopen:boolean; // MG: not used, obsolete?
 | |
|   end;
 | |
| 
 | |
| implementation
 | |
| 
 | |
|   uses
 | |
| {$ifdef Test_Double_checksum}
 | |
|     comphook,
 | |
| {$endif def Test_Double_checksum}
 | |
|     fpccrc,
 | |
|     cutils;
 | |
| 
 | |
| function swapendian_ppureal(d:ppureal):ppureal;
 | |
| 
 | |
| type ppureal_bytes=array[0..sizeof(d)-1] of byte;
 | |
| 
 | |
| var i:0..sizeof(d)-1;
 | |
| 
 | |
| begin
 | |
|   for i:=low(ppureal_bytes) to high(ppureal_bytes) do
 | |
|     ppureal_bytes(swapendian_ppureal)[i]:=ppureal_bytes(d)[high(ppureal_bytes)-i];
 | |
| end;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                                   TPPUFile
 | |
| *****************************************************************************}
 | |
| 
 | |
| constructor tppufile.Create(const fn:string);
 | |
| begin
 | |
|   fname:=fn;
 | |
|   change_endian:=false;
 | |
|   crc_only:=false;
 | |
|   Mode:=0;
 | |
|   NewHeader;
 | |
|   Error:=false;
 | |
|   closed:=true;
 | |
|   tempclosed:=false;
 | |
|   getmem(buf,ppubufsize);
 | |
| end;
 | |
| 
 | |
| 
 | |
| destructor tppufile.destroy;
 | |
| begin
 | |
|   closefile;
 | |
|   if assigned(buf) then
 | |
|     freemem(buf,ppubufsize);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.flush;
 | |
| begin
 | |
|   if Mode=2 then
 | |
|    writebuf;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.closefile;
 | |
| begin
 | |
| {$ifdef Test_Double_checksum}
 | |
|   if mode=2 then
 | |
|    begin
 | |
|      if assigned(crc_test) then
 | |
|       dispose(crc_test);
 | |
|      if assigned(crc_test2) then
 | |
|       dispose(crc_test2);
 | |
|    end;
 | |
| {$endif Test_Double_checksum}
 | |
|   if Mode<>0 then
 | |
|    begin
 | |
|      Flush;
 | |
|      f.Free;
 | |
|      Mode:=0;
 | |
|      closed:=true;
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.CheckPPUId:boolean;
 | |
| begin
 | |
|   CheckPPUId:=((Header.Id[1]='P') and (Header.Id[2]='P') and (Header.Id[3]='U'));
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.GetPPUVersion:integer;
 | |
| var
 | |
|   l    : integer;
 | |
|   code : integer;
 | |
| begin
 | |
|   Val(header.ver[1]+header.ver[2]+header.ver[3],l,code);
 | |
|   if code=0 then
 | |
|    GetPPUVersion:=l
 | |
|   else
 | |
|    GetPPUVersion:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.NewHeader;
 | |
| var
 | |
|   s : string;
 | |
| begin
 | |
|   fillchar(header,sizeof(tppuheader),0);
 | |
|   str(currentppuversion,s);
 | |
|   while length(s)<3 do
 | |
|    s:='0'+s;
 | |
|   with header do
 | |
|    begin
 | |
|      Id[1]:='P';
 | |
|      Id[2]:='P';
 | |
|      Id[3]:='U';
 | |
|      Ver[1]:=s[1];
 | |
|      Ver[2]:=s[2];
 | |
|      Ver[3]:=s[3];
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                                 TPPUFile Reading
 | |
| *****************************************************************************}
 | |
| 
 | |
| function tppufile.openfile:boolean;
 | |
| var
 | |
|   i      : integer;
 | |
| begin
 | |
|   openfile:=false;
 | |
|   try
 | |
|     f:=CFileStreamClass.Create(fname,fmOpenRead)
 | |
|   except
 | |
|     exit;
 | |
|   end;
 | |
|   closed:=false;
 | |
| {read ppuheader}
 | |
|   fsize:=f.Size;
 | |
|   if fsize<sizeof(tppuheader) then
 | |
|    exit;
 | |
|   i:=f.Read(header,sizeof(tppuheader));
 | |
|   { The header is always stored in little endian order }
 | |
|   { therefore swap if on a big endian machine          }
 | |
| {$IFDEF ENDIAN_BIG}
 | |
|   header.compiler := swapendian(header.compiler);
 | |
|   header.cpu := swapendian(header.cpu);
 | |
|   header.target := swapendian(header.target);
 | |
|   header.flags := swapendian(header.flags);
 | |
|   header.size := swapendian(header.size);
 | |
|   header.checksum := swapendian(header.checksum);
 | |
|   header.interface_checksum := swapendian(header.interface_checksum);
 | |
|   header.indirect_checksum := swapendian(header.indirect_checksum);
 | |
|   header.deflistsize:=swapendian(header.deflistsize);
 | |
|   header.symlistsize:=swapendian(header.symlistsize);
 | |
| {$ENDIF}
 | |
|   { the PPU DATA is stored in native order }
 | |
|   if (header.flags and uf_big_endian) = uf_big_endian then
 | |
|    Begin
 | |
| {$IFDEF ENDIAN_LITTLE}
 | |
|      change_endian := TRUE;
 | |
| {$ELSE}
 | |
|      change_endian := FALSE;
 | |
| {$ENDIF}
 | |
|    End
 | |
|   else if (header.flags and uf_little_endian) = uf_little_endian then
 | |
|    Begin
 | |
| {$IFDEF ENDIAN_BIG}
 | |
|      change_endian := TRUE;
 | |
| {$ELSE}
 | |
|      change_endian := FALSE;
 | |
| {$ENDIF}
 | |
|    End;
 | |
| {reset buffer}
 | |
|   bufstart:=i;
 | |
|   bufsize:=0;
 | |
|   bufidx:=0;
 | |
|   Mode:=1;
 | |
|   FillChar(entry,sizeof(tppuentry),0);
 | |
|   entryidx:=0;
 | |
|   entrystart:=0;
 | |
|   entrybufstart:=0;
 | |
|   Error:=false;
 | |
|   openfile:=true;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.reloadbuf;
 | |
| begin
 | |
|   inc(bufstart,bufsize);
 | |
|   bufsize:=f.Read(buf^,ppubufsize);
 | |
|   bufidx:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.readdata(out b;len:integer);
 | |
| var
 | |
|   p,pbuf : pchar;
 | |
|   left : integer;
 | |
| begin
 | |
|   p:=pchar(@b);
 | |
|   pbuf:=@buf[bufidx];
 | |
|   repeat
 | |
|     left:=bufsize-bufidx;
 | |
|     if len<left then
 | |
|       break;
 | |
|     move(pbuf^,p^,left);
 | |
|     dec(len,left);
 | |
|     inc(p,left);
 | |
|     reloadbuf;
 | |
|     pbuf:=@buf[bufidx];
 | |
|     if bufsize=0 then
 | |
|       exit;
 | |
|   until false;
 | |
|   move(pbuf^,p^,len);
 | |
|   inc(bufidx,len);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.skipdata(len:integer);
 | |
| var
 | |
|   left : integer;
 | |
| begin
 | |
|   while len>0 do
 | |
|    begin
 | |
|      left:=bufsize-bufidx;
 | |
|      if len>left then
 | |
|       begin
 | |
|         dec(len,left);
 | |
|         reloadbuf;
 | |
|         if bufsize=0 then
 | |
|          exit;
 | |
|       end
 | |
|      else
 | |
|       begin
 | |
|         inc(bufidx,len);
 | |
|         exit;
 | |
|       end;
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.readentry:byte;
 | |
| begin
 | |
|   if entryidx<entry.size then
 | |
|     begin
 | |
| {$ifdef generic_cpu}
 | |
|      has_more:=true;
 | |
| {$endif not generic_cpu}
 | |
|      skipdata(entry.size-entryidx);
 | |
|     end;
 | |
|   readdata(entry,sizeof(tppuentry));
 | |
|   if change_endian then
 | |
|     entry.size:=swapendian(entry.size);
 | |
|   entrystart:=bufstart+bufidx;
 | |
|   entryidx:=0;
 | |
| {$ifdef generic_cpu}
 | |
|   has_more:=false;
 | |
| {$endif not generic_cpu}
 | |
|   if not(entry.id in [mainentryid,subentryid]) then
 | |
|    begin
 | |
|      readentry:=iberror;
 | |
|      error:=true;
 | |
|      exit;
 | |
|    end;
 | |
|   readentry:=entry.nr;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.endofentry:boolean;
 | |
| begin
 | |
| {$ifdef generic_cpu}
 | |
|   endofentry:=(entryidx=entry.size);
 | |
| {$else not generic_cpu}
 | |
|   endofentry:=(entryidx>=entry.size);
 | |
| {$endif not generic_cpu}
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.entrysize:longint;
 | |
| begin
 | |
|   entrysize:=entry.size;
 | |
| end;
 | |
| 
 | |
| function tppufile.entryleft:longint;
 | |
| begin
 | |
|   entryleft:=entry.size-entryidx;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.getdatabuf(out b;len:integer;out res:integer);
 | |
| begin
 | |
|   if entryidx+len>entry.size then
 | |
|    res:=entry.size-entryidx
 | |
|   else
 | |
|    res:=len;
 | |
|   readdata(b,res);
 | |
|   inc(entryidx,res);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.getdata(out b;len:integer);
 | |
| begin
 | |
|   if entryidx+len>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      exit;
 | |
|    end;
 | |
|   readdata(b,len);
 | |
|   inc(entryidx,len);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getbyte:byte;
 | |
| begin
 | |
|   if entryidx+1>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=1 then
 | |
|     begin
 | |
|       result:=pbyte(@buf[bufidx])^;
 | |
|       inc(bufidx);
 | |
|     end
 | |
|   else
 | |
|     readdata(result,1);
 | |
|   inc(entryidx);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getword:word;
 | |
| begin
 | |
|   if entryidx+2>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=sizeof(word) then
 | |
|     begin
 | |
|       result:=Unaligned(pword(@buf[bufidx])^);
 | |
|       inc(bufidx,sizeof(word));
 | |
|     end
 | |
|   else
 | |
|     readdata(result,sizeof(word));
 | |
|   if change_endian then
 | |
|    result:=swapendian(result);
 | |
|   inc(entryidx,2);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getlongint:longint;
 | |
| begin
 | |
|   if entryidx+4>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=sizeof(longint) then
 | |
|     begin
 | |
|       result:=Unaligned(plongint(@buf[bufidx])^);
 | |
|       inc(bufidx,sizeof(longint));
 | |
|     end
 | |
|   else
 | |
|     readdata(result,sizeof(longint));
 | |
|   if change_endian then
 | |
|    result:=swapendian(result);
 | |
|   inc(entryidx,4);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getdword:dword;
 | |
| begin
 | |
|   if entryidx+4>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=sizeof(dword) then
 | |
|     begin
 | |
|       result:=Unaligned(plongint(@buf[bufidx])^);
 | |
|       inc(bufidx,sizeof(longint));
 | |
|     end
 | |
|   else
 | |
|     readdata(result,sizeof(dword));
 | |
|   if change_endian then
 | |
|    result:=swapendian(result);
 | |
|   inc(entryidx,4);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getint64:int64;
 | |
| begin
 | |
|   if entryidx+8>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=sizeof(int64) then
 | |
|     begin
 | |
|       result:=Unaligned(pint64(@buf[bufidx])^);
 | |
|       inc(bufidx,sizeof(int64));
 | |
|     end
 | |
|   else
 | |
|     readdata(result,sizeof(int64));
 | |
|   if change_endian then
 | |
|    result:=swapendian(result);
 | |
|   inc(entryidx,8);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getqword:qword;
 | |
| begin
 | |
|   if entryidx+8>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:=0;
 | |
|      exit;
 | |
|    end;
 | |
|   if bufsize-bufidx>=sizeof(qword) then
 | |
|     begin
 | |
|       result:=Unaligned(pqword(@buf[bufidx])^);
 | |
|       inc(bufidx,sizeof(qword));
 | |
|     end
 | |
|   else
 | |
|     readdata(result,sizeof(qword));
 | |
|   if change_endian then
 | |
|    result:=swapendian(result);
 | |
|   inc(entryidx,8);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getaint:aint;
 | |
| begin
 | |
| {$ifdef generic_cpu}
 | |
|   if CpuAluBitSize[tsystemcpu(header.cpu)]=64 then
 | |
|     result:=getint64
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=32 then
 | |
|     result:=getlongint
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=16 then
 | |
|     result:=smallint(getword)
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=8 then
 | |
|     result:=shortint(getbyte)
 | |
|   else
 | |
|     begin
 | |
|       error:=true;
 | |
|       result:=0;
 | |
|     end;
 | |
| {$else not generic_cpu}
 | |
|   result:=4;
 | |
|   case sizeof(aint) of
 | |
|     8: result:=getint64;
 | |
|     4: result:=getlongint;
 | |
|     2: result:=smallint(getword);
 | |
|     1: result:=shortint(getbyte);
 | |
|   end;
 | |
| {$endif not generic_cpu}
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getasizeint:asizeint;
 | |
| begin
 | |
| {$ifdef generic_cpu}
 | |
|   if CpuAddrBitSize[tsystemcpu(header.cpu)]=64 then
 | |
|     result:=getint64
 | |
|   else if CpuAddrBitSize[tsystemcpu(header.cpu)]=32 then
 | |
|     result:=getlongint
 | |
|   else if CpuAddrBitSize[tsystemcpu(header.cpu)]=16 then
 | |
|     result:=smallint(getword)
 | |
|   else
 | |
|     begin
 | |
|       error:=true;
 | |
|       result:=0;
 | |
|     end;
 | |
| {$else not generic_cpu}
 | |
| {$if defined(cpu64bitaddr)}
 | |
|   result:=getint64;
 | |
| {$elseif defined(cpu32bitaddr)}
 | |
|   result:=getlongint;
 | |
| {$elseif defined(cpu16bitaddr)}
 | |
|   result:=getword;
 | |
| {$endif}
 | |
| {$endif not generic_cpu}
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getaword:aword;
 | |
| begin
 | |
| {$ifdef generic_cpu}
 | |
|   if CpuAluBitSize[tsystemcpu(header.cpu)]=64 then
 | |
|     result:=getqword
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=32 then
 | |
|     result:=getdword
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=16 then
 | |
|     result:=getword
 | |
|   else if CpuAluBitSize[tsystemcpu(header.cpu)]=8 then
 | |
|     result:=getbyte
 | |
|   else
 | |
|     begin
 | |
|       error:=true;
 | |
|       result:=0;
 | |
|     end;
 | |
| {$else not generic_cpu}
 | |
|   result:=4;
 | |
|   case sizeof(aword) of
 | |
|     8: result:=getqword;
 | |
|     4: result:=getdword;
 | |
|     2: result:=getword;
 | |
|     1: result:=getbyte;
 | |
|   end;
 | |
| {$endif not generic_cpu}
 | |
| end;
 | |
| 
 | |
| function  tppufile.getrealsize(sizeofreal : longint):ppureal;
 | |
| var
 | |
|   e : ppureal;
 | |
|   d : double;
 | |
|   s : single;
 | |
| begin
 | |
|   if sizeofreal=sizeof(e) then
 | |
|     begin
 | |
|       if entryidx+sizeof(e)>entry.size then
 | |
|        begin
 | |
|          error:=true;
 | |
|          result:=0;
 | |
|          exit;
 | |
|        end;
 | |
|       readdata(e,sizeof(e));
 | |
|       if change_endian then
 | |
|         result:=swapendian_ppureal(e)
 | |
|       else
 | |
|         result:=e;
 | |
|       inc(entryidx,sizeof(e));
 | |
|       exit;
 | |
|     end;
 | |
|   if sizeofreal=sizeof(d) then
 | |
|     begin
 | |
|       if entryidx+sizeof(d)>entry.size then
 | |
|        begin
 | |
|          error:=true;
 | |
|          result:=0;
 | |
|          exit;
 | |
|        end;
 | |
|       readdata(d,sizeof(d));
 | |
|       if change_endian then
 | |
|         result:=swapendian(pqword(@d)^)
 | |
|       else
 | |
|         result:=d;
 | |
|       inc(entryidx,sizeof(d));
 | |
|       result:=d;
 | |
|       exit;
 | |
|     end;
 | |
|   if sizeofreal=sizeof(s) then
 | |
|     begin
 | |
|       if entryidx+sizeof(s)>entry.size then
 | |
|        begin
 | |
|          error:=true;
 | |
|          result:=0;
 | |
|          exit;
 | |
|        end;
 | |
|       readdata(s,sizeof(s));
 | |
|       if change_endian then
 | |
|         result:=swapendian(pdword(@s)^)
 | |
|       else
 | |
|         result:=s;
 | |
|       inc(entryidx,sizeof(s));
 | |
|       result:=s;
 | |
|       exit;
 | |
|     end;
 | |
|   error:=true;
 | |
|   result:=0.0;
 | |
| end;
 | |
| 
 | |
| function tppufile.getreal:ppureal;
 | |
| var
 | |
|   d : ppureal;
 | |
|   hd : double;
 | |
| begin
 | |
|   if target_info.system=system_x86_64_win64 then
 | |
|     begin
 | |
|       hd:=getrealsize(sizeof(hd));
 | |
|       getreal:=hd;
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
|       d:=getrealsize(sizeof(d));
 | |
|       getreal:=d;
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getstring:string;
 | |
| begin
 | |
|   result[0]:=chr(getbyte);
 | |
|   if entryidx+length(result)>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      exit;
 | |
|    end;
 | |
|   ReadData(result[1],length(result));
 | |
|   inc(entryidx,length(result));
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.getansistring:ansistring;
 | |
| var
 | |
|   len: longint;
 | |
| begin
 | |
|   len:=getlongint;
 | |
|   if entryidx+len>entry.size then
 | |
|    begin
 | |
|      error:=true;
 | |
|      result:='';
 | |
|      exit;
 | |
|    end;
 | |
|   setlength(result,len);
 | |
|   if len>0 then
 | |
|     getdata(result[1],len);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.getsmallset(out b);
 | |
| var
 | |
|   i : longint;
 | |
| begin
 | |
|   getdata(b,4);
 | |
|   if change_endian then
 | |
|     for i:=0 to 3 do
 | |
|       Pbyte(@b)[i]:=reverse_byte(Pbyte(@b)[i]);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.getnormalset(out b);
 | |
| var
 | |
|   i : longint;
 | |
| begin
 | |
|   getdata(b,32);
 | |
|   if change_endian then
 | |
|     for i:=0 to 31 do
 | |
|       Pbyte(@b)[i]:=reverse_byte(Pbyte(@b)[i]);
 | |
| end;
 | |
| 
 | |
| 
 | |
| function tppufile.skipuntilentry(untilb:byte):boolean;
 | |
| var
 | |
|   b : byte;
 | |
| begin
 | |
|   repeat
 | |
|     b:=readentry;
 | |
|   until (b in [ibend,iberror]) or ((b=untilb) and (entry.id=mainentryid));
 | |
|   skipuntilentry:=(b=untilb);
 | |
| end;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                                 TPPUFile Writing
 | |
| *****************************************************************************}
 | |
| 
 | |
| function tppufile.createfile:boolean;
 | |
| var
 | |
|   ok: boolean;
 | |
| begin
 | |
|   createfile:=false;
 | |
| {$ifdef INTFPPU}
 | |
|   if crc_only then
 | |
|    begin
 | |
|      fname:=fname+'.intf';
 | |
|      crc_only:=false;
 | |
|    end;
 | |
| {$endif}
 | |
|   if not crc_only then
 | |
|     begin
 | |
|       {$ifdef MACOS}
 | |
|       {FPas is FreePascal's creator code on MacOS. See systems/mac_crea.txt}
 | |
|       SetDefaultMacOSCreator('FPas');
 | |
|       SetDefaultMacOSFiletype('FPPU');
 | |
|       {$endif}
 | |
|       ok:=false;
 | |
|       try
 | |
|         f:=CFileStreamClass.Create(fname,fmCreate);
 | |
|         ok:=true;
 | |
|       except
 | |
|       end;
 | |
|       {$ifdef MACOS}
 | |
|       SetDefaultMacOSCreator('MPS ');
 | |
|       SetDefaultMacOSFiletype('TEXT');
 | |
|       {$endif}
 | |
|       if not ok then
 | |
|        exit;
 | |
|       Mode:=2;
 | |
|     {write header for sure}
 | |
|       f.Write(header,sizeof(tppuheader));
 | |
|     end;
 | |
|   bufsize:=ppubufsize;
 | |
|   bufstart:=sizeof(tppuheader);
 | |
|   bufidx:=0;
 | |
| {reset}
 | |
|   crc:=0;
 | |
|   interface_crc:=0;
 | |
|   indirect_crc:=0;
 | |
|   do_interface_crc:=true;
 | |
|   do_indirect_crc:=false;
 | |
|   Error:=false;
 | |
|   do_crc:=true;
 | |
|   size:=0;
 | |
|   entrytyp:=mainentryid;
 | |
| {start}
 | |
|   NewEntry;
 | |
|   createfile:=true;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.writeheader;
 | |
| var
 | |
|   opos : integer;
 | |
| begin
 | |
|   if crc_only then
 | |
|    exit;
 | |
|   { flush buffer }
 | |
|   writebuf;
 | |
|   { update size (w/o header!) in the header }
 | |
|   header.size:=bufstart-sizeof(tppuheader);
 | |
|   { set the endian flag }
 | |
| {$ifndef FPC_BIG_ENDIAN}
 | |
|     header.flags := header.flags or uf_little_endian;
 | |
| {$else not FPC_BIG_ENDIAN}
 | |
|     header.flags := header.flags or uf_big_endian;
 | |
|     { Now swap the header in the correct endian (always little endian) }
 | |
|     header.compiler := swapendian(header.compiler);
 | |
|     header.cpu := swapendian(header.cpu);
 | |
|     header.target := swapendian(header.target);
 | |
|     header.flags := swapendian(header.flags);
 | |
|     header.size := swapendian(header.size);
 | |
|     header.checksum := swapendian(header.checksum);
 | |
|     header.interface_checksum := swapendian(header.interface_checksum);
 | |
|     header.indirect_checksum := swapendian(header.indirect_checksum);
 | |
|     header.deflistsize:=swapendian(header.deflistsize);
 | |
|     header.symlistsize:=swapendian(header.symlistsize);
 | |
| {$endif not FPC_BIG_ENDIAN}
 | |
| { write header and restore filepos after it }
 | |
|   opos:=f.Position;
 | |
|   f.Position:=0;
 | |
|   f.Write(header,sizeof(tppuheader));
 | |
|   f.Position:=opos;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.writebuf;
 | |
| begin
 | |
|   if not crc_only and
 | |
|      (bufidx <> 0) then
 | |
|     f.Write(buf^,bufidx);
 | |
|   inc(bufstart,bufidx);
 | |
|   bufidx:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.writedata(const b;len:integer);
 | |
| var
 | |
|   p   : pchar;
 | |
|   left,
 | |
|   idx : integer;
 | |
| begin
 | |
|   if crc_only then
 | |
|     exit;
 | |
|   p:=pchar(@b);
 | |
|   idx:=0;
 | |
|   while len>0 do
 | |
|    begin
 | |
|      left:=bufsize-bufidx;
 | |
|      if len>left then
 | |
|       begin
 | |
|         move(p[idx],buf[bufidx],left);
 | |
|         dec(len,left);
 | |
|         inc(idx,left);
 | |
|         inc(bufidx,left);
 | |
|         writebuf;
 | |
|       end
 | |
|      else
 | |
|       begin
 | |
|         move(p[idx],buf[bufidx],len);
 | |
|         inc(bufidx,len);
 | |
|         exit;
 | |
|       end;
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.NewEntry;
 | |
| begin
 | |
|   with entry do
 | |
|    begin
 | |
|      id:=entrytyp;
 | |
|      nr:=ibend;
 | |
|      size:=0;
 | |
|    end;
 | |
| {Reset Entry State}
 | |
|   entryidx:=0;
 | |
|   entrybufstart:=bufstart;
 | |
|   entrystart:=bufstart+bufidx;
 | |
| {Alloc in buffer}
 | |
|   writedata(entry,sizeof(tppuentry));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.writeentry(ibnr:byte);
 | |
| var
 | |
|   opos : integer;
 | |
| begin
 | |
| {create entry}
 | |
|   entry.id:=entrytyp;
 | |
|   entry.nr:=ibnr;
 | |
|   entry.size:=entryidx;
 | |
| {it's already been sent to disk ?}
 | |
|   if entrybufstart<>bufstart then
 | |
|    begin
 | |
|     if not crc_only then
 | |
|       begin
 | |
|       {flush to be sure}
 | |
|         WriteBuf;
 | |
|       {write entry}
 | |
|         opos:=f.Position;
 | |
|         f.Position:=entrystart;
 | |
|         f.write(entry,sizeof(tppuentry));
 | |
|         f.Position:=opos;
 | |
|       end;
 | |
|      entrybufstart:=bufstart;
 | |
|    end
 | |
|   else
 | |
|    move(entry,buf[entrystart-bufstart],sizeof(entry));
 | |
| {Add New Entry, which is ibend by default}
 | |
|   entrystart:=bufstart+bufidx; {next entry position}
 | |
|   NewEntry;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putdata(const b;len:integer);
 | |
| begin
 | |
|   if do_crc then
 | |
|    begin
 | |
|      crc:=UpdateCrc32(crc,b,len);
 | |
| {$ifdef Test_Double_checksum}
 | |
|      if crc_only then
 | |
|        begin
 | |
|          crc_test2^[crc_index2]:=crc;
 | |
| {$ifdef Test_Double_checksum_write}
 | |
|          Writeln(CRCFile,crc);
 | |
| {$endif Test_Double_checksum_write}
 | |
|          if crc_index2<crc_array_size then
 | |
|           inc(crc_index2);
 | |
|        end
 | |
|      else
 | |
|        begin
 | |
|          if (crcindex2<crc_array_size) and (crcindex2<crc_index2) and
 | |
|             (crc_test2^[crcindex2]<>crc) then
 | |
|            Do_comment(V_Note,'impl CRC changed');
 | |
| {$ifdef Test_Double_checksum_write}
 | |
|          Writeln(CRCFile,crc);
 | |
| {$endif Test_Double_checksum_write}
 | |
|          inc(crcindex2);
 | |
|        end;
 | |
| {$endif def Test_Double_checksum}
 | |
|      if do_interface_crc then
 | |
|        begin
 | |
|          interface_crc:=UpdateCrc32(interface_crc,b,len);
 | |
| {$ifdef Test_Double_checksum}
 | |
|         if crc_only then
 | |
|           begin
 | |
|             crc_test^[crc_index]:=interface_crc;
 | |
| {$ifdef Test_Double_checksum_write}
 | |
|             Writeln(CRCFile,interface_crc);
 | |
| {$endif Test_Double_checksum_write}
 | |
|             if crc_index<crc_array_size then
 | |
|              inc(crc_index);
 | |
|           end
 | |
|         else
 | |
|           begin
 | |
|             if (crcindex<crc_array_size) and (crcindex<crc_index) and
 | |
|                (crc_test^[crcindex]<>interface_crc) then
 | |
|               Do_comment(V_Warning,'CRC changed');
 | |
| {$ifdef Test_Double_checksum_write}
 | |
|             Writeln(CRCFile,interface_crc);
 | |
| {$endif Test_Double_checksum_write}
 | |
|             inc(crcindex);
 | |
|           end;
 | |
| {$endif def Test_Double_checksum}
 | |
|          { indirect crc must only be calculated for the interface; changes
 | |
|            to a class in the implementation cannot require another unit to
 | |
|            be recompiled }
 | |
|          if do_indirect_crc then
 | |
|            indirect_crc:=UpdateCrc32(indirect_crc,b,len);
 | |
|        end;
 | |
|     end;
 | |
|   if not crc_only then
 | |
|     writedata(b,len);
 | |
|   inc(entryidx,len);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putbyte(b:byte);
 | |
| begin
 | |
|   putdata(b,1);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putword(w:word);
 | |
| begin
 | |
|   putdata(w,2);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putdword(w:dword);
 | |
| begin
 | |
|   putdata(w,4);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putlongint(l:longint);
 | |
| begin
 | |
|   putdata(l,4);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putint64(i:int64);
 | |
| begin
 | |
|   putdata(i,8);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putqword(q:qword);
 | |
| begin
 | |
|   putdata(q,sizeof(qword));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putaint(i:aint);
 | |
| begin
 | |
|   putdata(i,sizeof(aint));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putasizeint(i: asizeint);
 | |
| begin
 | |
|   putdata(i,sizeof(asizeint));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putaword(i:aword);
 | |
| begin
 | |
|   putdata(i,sizeof(aword));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putreal(d:ppureal);
 | |
| var
 | |
|   hd : double;
 | |
| begin
 | |
|   if target_info.system=system_x86_64_win64 then
 | |
|     begin
 | |
|       hd:=d;
 | |
|       putdata(hd,sizeof(hd));
 | |
|     end
 | |
|   else
 | |
|     putdata(d,sizeof(ppureal));
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putstring(const s:string);
 | |
|   begin
 | |
|     putdata(s,length(s)+1);
 | |
|   end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putansistring(const s:ansistring);
 | |
|   var
 | |
|     len: longint;
 | |
|   begin
 | |
|     len:=length(s);
 | |
|     putlongint(len);
 | |
|     if len>0 then
 | |
|       putdata(s[1],len);
 | |
|   end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putsmallset(const b);
 | |
|   var
 | |
|     l : longint;
 | |
|   begin
 | |
|     l:=longint(b);
 | |
|     putlongint(l);
 | |
|   end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.putnormalset(const b);
 | |
|   begin
 | |
|     putdata(b,32);
 | |
|   end;
 | |
| 
 | |
| 
 | |
| procedure tppufile.tempclose;
 | |
|   begin
 | |
|     if not closed then
 | |
|      begin
 | |
|        closepos:=f.Position;
 | |
|        f.Free;
 | |
|        f:=nil;
 | |
|        closed:=true;
 | |
|        tempclosed:=true;
 | |
|      end;
 | |
|   end;
 | |
| 
 | |
| 
 | |
| function tppufile.tempopen:boolean;
 | |
|   begin
 | |
|     tempopen:=false;
 | |
|     if not closed or not tempclosed then
 | |
|      exit;
 | |
|    { MG: not sure, if this is correct
 | |
|      f.position:=0;
 | |
|        No, f was freed in tempclose above, we need to
 | |
|        recreate it.  PM 2011/06/06 }
 | |
|     try
 | |
|       f:=CFileStreamClass.Create(fname,fmOpenRead);
 | |
|     except
 | |
|       exit;
 | |
|     end;
 | |
|     closed:=false;
 | |
|     tempclosed:=false;
 | |
| 
 | |
|   { restore state }
 | |
|     f.Position:=closepos;
 | |
|     tempopen:=true;
 | |
|   end;
 | |
| 
 | |
| end.
 |