| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199 | {    This file is part of the Free Component Library (FCL)    Copyright (c) 1999-2000 by the Free Pascal development team    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    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. **********************************************************************}{****************************************************************************}{*                             TStream                                      *}{****************************************************************************}procedure TStream.ReadNotImplemented;begin  raise EStreamError.CreateFmt(SStreamNoReading, [ClassName]) at get_caller_addr(get_frame);end;procedure TStream.WriteNotImplemented;begin  raise EStreamError.CreateFmt(SStreamNoWriting, [ClassName]) at get_caller_addr(get_frame);end;function TStream.Read(var Buffer; Count: Longint): Longint;begin  ReadNotImplemented;  Result := 0;end;function TStream.Write(const Buffer; Count: Longint): Longint;begin  WriteNotImplemented;  Result := 0;end;  function TStream.GetPosition: Int64;    begin       Result:=Seek(0,soCurrent);    end;  procedure TStream.SetPosition(const Pos: Int64);    begin       Seek(pos,soBeginning);    end;  procedure TStream.SetSize64(const NewSize: Int64);    begin      // Required because can't use overloaded functions in properties      SetSize(NewSize);    end;  function TStream.GetSize: Int64;    var       p : int64;    begin       p:=Seek(0,soCurrent);       GetSize:=Seek(0,soEnd);       Seek(p,soBeginning);    end;  procedure TStream.SetSize(NewSize: Longint);    begin    // We do nothing. Pipe streams don't support this    // As wel as possible read-ony streams !!    end;  procedure TStream.SetSize(const NewSize: Int64);    begin      // Backwards compatibility that calls the longint SetSize      if (NewSize<Low(longint)) or         (NewSize>High(longint)) then        raise ERangeError.Create(SRangeError);      SetSize(longint(NewSize));    end;  function TStream.Seek(Offset: Longint; Origin: Word): Longint;    type      TSeek64 = function(const offset:Int64;Origin:TSeekorigin):Int64 of object;    var      CurrSeek,      TStreamSeek : TSeek64;      CurrClass   : TClass;    begin      // Redirect calls to 64bit Seek, but we can't call the 64bit Seek      // from TStream, because then we end up in an infinite loop      CurrSeek:=nil;      CurrClass:=Classtype;      while (CurrClass<>nil) and            (CurrClass<>TStream) do       CurrClass:=CurrClass.Classparent;      if CurrClass<>nil then       begin         CurrSeek:[email protected];         TStreamSeek:=@TStream(@CurrClass).Seek;         if TMethod(TStreamSeek).Code=TMethod(CurrSeek).Code then          CurrSeek:=nil;       end;      if CurrSeek<>nil then       Result:=Seek(Int64(offset),TSeekOrigin(origin))      else       raise EStreamError.CreateFmt(SSeekNotImplemented,[ClassName]);    end;  procedure TStream.Discard(const Count: Int64);  const    CSmallSize      =255;    CLargeMaxBuffer =32*1024; // 32 KiB  var    Buffer: array[1..CSmallSize] of Byte;  begin    if Count=0 then      Exit;    if Count<=SizeOf(Buffer) then      ReadBuffer(Buffer,Count)    else      DiscardLarge(Count,CLargeMaxBuffer);  end;  procedure TStream.DiscardLarge(Count: int64; const MaxBufferSize: Longint);  var    Buffer: array of Byte;  begin    if Count=0 then       Exit;    if Count>MaxBufferSize then      SetLength(Buffer,MaxBufferSize)    else      SetLength(Buffer,Count);    while (Count>=Length(Buffer)) do      begin      ReadBuffer(Buffer[0],Length(Buffer));      Dec(Count,Length(Buffer));      end;    if Count>0 then      ReadBuffer(Buffer[0],Count);  end;  procedure TStream.InvalidSeek;  begin    raise EStreamError.CreateFmt(SStreamInvalidSeek, [ClassName]) at get_caller_addr(get_frame);  end;  procedure TStream.FakeSeekForward(Offset: Int64;  const Origin: TSeekOrigin; const Pos: Int64);  var    Buffer: Pointer;    BufferSize, i: LongInt;  begin    if Origin=soBeginning then       Dec(Offset,Pos);    if (Offset<0) or (Origin=soEnd) then      InvalidSeek;    if Offset>0 then      Discard(Offset);   end; function TStream.Seek(const Offset: Int64; Origin: TSeekorigin): Int64;    begin      // Backwards compatibility that calls the longint Seek      if (Offset<Low(longint)) or         (Offset>High(longint)) then        raise ERangeError.Create(SRangeError);      Result:=Seek(longint(Offset),ord(Origin));    end;  procedure TStream.ReadBuffer(var Buffer; Count: Longint);    begin       if Read(Buffer,Count)<Count then         Raise EReadError.Create(SReadError);    end;  procedure TStream.WriteBuffer(const Buffer; Count: Longint);    begin       if Write(Buffer,Count)<Count then         Raise EWriteError.Create(SWriteError);    end;  function TStream.CopyFrom(Source: TStream; Count: Int64): Int64;    var       Buffer: Pointer;       BufferSize, i: LongInt;    const       MaxSize = $20000;    begin       Result:=0;       if Count=0 then         Source.Position:=0;   // This WILL fail for non-seekable streams...       BufferSize:=MaxSize;       if (Count>0) and (Count<BufferSize) then         BufferSize:=Count;    // do not allocate more than needed       GetMem(Buffer,BufferSize);       try         if Count=0 then         repeat           i:=Source.Read(buffer^,BufferSize);           if i>0 then             WriteBuffer(buffer^,i);           Inc(Result,i);         until i<BufferSize         else         while Count>0 do         begin           if Count>BufferSize then             i:=BufferSize           else             i:=Count;           Source.ReadBuffer(buffer^,i);           WriteBuffer(buffer^,i);           Dec(count,i);           Inc(Result,i);         end;       finally         FreeMem(Buffer);       end;    end;  function TStream.ReadComponent(Instance: TComponent): TComponent;    var      Reader: TReader;    begin      Reader := TReader.Create(Self, 4096);      try        Result := Reader.ReadRootComponent(Instance);      finally        Reader.Free;      end;    end;  function TStream.ReadComponentRes(Instance: TComponent): TComponent;    begin      ReadResHeader;      Result := ReadComponent(Instance);    end;  procedure TStream.WriteComponent(Instance: TComponent);    begin      WriteDescendent(Instance, nil);    end;  procedure TStream.WriteComponentRes(const ResName: string; Instance: TComponent);    begin      WriteDescendentRes(ResName, Instance, nil);    end;  procedure TStream.WriteDescendent(Instance, Ancestor: TComponent);    var       Driver : TAbstractObjectWriter;       Writer : TWriter;    begin       Driver := TBinaryObjectWriter.Create(Self, 4096);       Try         Writer := TWriter.Create(Driver);         Try           Writer.WriteDescendent(Instance, Ancestor);         Finally           Writer.Destroy;         end;       Finally         Driver.Free;       end;    end;  procedure TStream.WriteDescendentRes(const ResName: string; Instance, Ancestor: TComponent);    var      FixupInfo: Integer;    begin      { Write a resource header }      WriteResourceHeader(ResName, FixupInfo);      { Write the instance itself }      WriteDescendent(Instance, Ancestor);      { Insert the correct resource size into the resource header }      FixupResourceHeader(FixupInfo);    end;  procedure TStream.WriteResourceHeader(const ResName: string; {!!!: out} var FixupInfo: Integer);    var      ResType, Flags : word;    begin       ResType:=NtoLE(word($000A));       Flags:=NtoLE(word($1030));       { Note: This is a Windows 16 bit resource }       { Numeric resource type }       WriteByte($ff);       { Application defined data }       WriteWord(ResType);       { write the name as asciiz }       WriteBuffer(ResName[1],length(ResName));       WriteByte(0);       { Movable, Pure and Discardable }       WriteWord(Flags);       { Placeholder for the resource size }       WriteDWord(0);       { Return current stream position so that the resource size can be         inserted later }       FixupInfo := Position;    end;  procedure TStream.FixupResourceHeader(FixupInfo: Integer);    var       ResSize,TmpResSize : Integer;    begin      ResSize := Position - FixupInfo;      TmpResSize := NtoLE(longword(ResSize));      { Insert the correct resource size into the placeholder written by        WriteResourceHeader }      Position := FixupInfo - 4;      WriteDWord(TmpResSize);      { Seek back to the end of the resource }      Position := FixupInfo + ResSize;    end;  procedure TStream.ReadResHeader;    var      ResType, Flags : word;    begin       try         { Note: This is a Windows 16 bit resource }         { application specific resource ? }         if ReadByte<>$ff then           raise EInvalidImage.Create(SInvalidImage);         ResType:=LEtoN(ReadWord);         if ResType<>$000a then           raise EInvalidImage.Create(SInvalidImage);         { read name }         while ReadByte<>0 do           ;         { check the access specifier }         Flags:=LEtoN(ReadWord);         if Flags<>$1030 then           raise EInvalidImage.Create(SInvalidImage);         { ignore the size }         ReadDWord;       except         on EInvalidImage do           raise;         else           raise EInvalidImage.create(SInvalidImage);       end;    end;  function TStream.ReadByte : Byte;    var       b : Byte;    begin       ReadBuffer(b,1);       ReadByte:=b;    end;  function TStream.ReadWord : Word;    var       w : Word;    begin       ReadBuffer(w,2);       ReadWord:=w;    end;  function TStream.ReadDWord : Cardinal;    var       d : Cardinal;    begin       ReadBuffer(d,4);       ReadDWord:=d;    end;  function TStream.ReadQWord: QWord;    var       q: QWord;    begin      ReadBuffer(q,8);      ReadQWord:=q;    end;  Function TStream.ReadAnsiString : String;  Var    TheSize : Longint;    P : PByte ;  begin    ReadBuffer (TheSize,SizeOf(TheSize));    SetLength(Result,TheSize);    // Illegal typecast if no AnsiStrings defined.    if TheSize>0 then     begin       ReadBuffer (Pointer(Result)^,TheSize);       P:=Pointer(Result)+TheSize;       p^:=0;     end;   end;  Procedure TStream.WriteAnsiString (const S : String);  Var L : Longint;  begin    L:=Length(S);    WriteBuffer (L,SizeOf(L));    WriteBuffer (Pointer(S)^,L);  end;  procedure TStream.WriteByte(b : Byte);    begin       WriteBuffer(b,1);    end;  procedure TStream.WriteWord(w : Word);    begin       WriteBuffer(w,2);    end;  procedure TStream.WriteDWord(d : Cardinal);    begin       WriteBuffer(d,4);    end;  procedure TStream.WriteQWord(q: QWord);    begin      WriteBuffer(q,8);    end;{****************************************************************************}{*                             THandleStream                                *}{****************************************************************************}Constructor THandleStream.Create(AHandle: THandle);begin  Inherited Create;  FHandle:=AHandle;end;function THandleStream.Read(var Buffer; Count: Longint): Longint;begin  Result:=FileRead(FHandle,Buffer,Count);  If Result=-1 then Result:=0;end;function THandleStream.Write(const Buffer; Count: Longint): Longint;begin  Result:=FileWrite (FHandle,Buffer,Count);  If Result=-1 then Result:=0;end;Procedure THandleStream.SetSize(NewSize: Longint);begin  SetSize(Int64(NewSize));end;Procedure THandleStream.SetSize(const NewSize: Int64);begin  FileTruncate(FHandle,NewSize);end;function THandleStream.Seek(const Offset: Int64; Origin: TSeekOrigin): Int64;begin  Result:=FileSeek(FHandle,Offset,ord(Origin));end;{****************************************************************************}{*                             TFileStream                                  *}{****************************************************************************}constructor TFileStream.Create(const AFileName: string; Mode: Word);begin  Create(AFileName,Mode,438);end;constructor TFileStream.Create(const AFileName: string; Mode: Word; Rights: Cardinal);begin  FFileName:=AFileName;  If (Mode and fmCreate) > 0 then    FHandle:=FileCreate(AFileName,Mode,Rights)  else    FHAndle:=FileOpen(AFileName,Mode);  If (THandle(FHandle)=feInvalidHandle) then    If Mode=fmcreate then      raise EFCreateError.createfmt(SFCreateError,[AFileName])    else      raise EFOpenError.Createfmt(SFOpenError,[AFilename]);end;destructor TFileStream.Destroy;begin  FileClose(FHandle);end;{****************************************************************************}{*                             TCustomMemoryStream                          *}{****************************************************************************}procedure TCustomMemoryStream.SetPointer(Ptr: Pointer; ASize: PtrInt);begin  FMemory:=Ptr;  FSize:=ASize;end;function TCustomMemoryStream.GetSize: Int64;begin  Result:=FSize;end;function TCustomMemoryStream.GetPosition: Int64;begin  Result:=FPosition;end;function TCustomMemoryStream.Read(var Buffer; Count: LongInt): LongInt;begin  Result:=0;  If (FSize>0) and (FPosition<Fsize) and (FPosition>=0) then    begin    Result:=Count;    If (Result>(FSize-FPosition)) then      Result:=(FSize-FPosition);    Move ((FMemory+FPosition)^,Buffer,Result);    FPosition:=Fposition+Result;    end;end;function TCustomMemoryStream.Seek(const Offset: Int64; Origin: TSeekOrigin): Int64;begin  Case Word(Origin) of    soFromBeginning : FPosition:=Offset;    soFromEnd       : FPosition:=FSize+Offset;    soFromCurrent   : FPosition:=FPosition+Offset;  end;  Result:=FPosition;  {$IFDEF DEBUG}  if Result < 0 then    raise Exception.Create('TCustomMemoryStream');  {$ENDIF}end;procedure TCustomMemoryStream.SaveToStream(Stream: TStream);begin  if FSize>0 then Stream.WriteBuffer (FMemory^,FSize);end;procedure TCustomMemoryStream.SaveToFile(const FileName: string);Var S : TFileStream;begin  S:=TFileStream.Create (FileName,fmCreate);  Try    SaveToStream(S);  finally    S.free;  end;end;{****************************************************************************}{*                             TMemoryStream                                *}{****************************************************************************}Const TMSGrow = 4096; { Use 4k blocks. }procedure TMemoryStream.SetCapacity(NewCapacity: PtrInt);begin  SetPointer (Realloc(NewCapacity),Fsize);  FCapacity:=NewCapacity;end;function TMemoryStream.Realloc(var NewCapacity: PtrInt): Pointer;begin  If NewCapacity<0 Then    NewCapacity:=0  else    begin      // if growing, grow at least a quarter      if (NewCapacity>FCapacity) and (NewCapacity < (5*FCapacity) div 4) then        NewCapacity := (5*FCapacity) div 4;      // round off to block size.      NewCapacity := (NewCapacity + (TMSGrow-1)) and not (TMSGROW-1);    end;  // Only now check !  If NewCapacity=FCapacity then    Result:=FMemory  else    begin      Result:=Reallocmem(FMemory,Newcapacity);      If (Result=Nil) and (Newcapacity>0) then        Raise EStreamError.Create(SMemoryStreamError);    end;end;destructor TMemoryStream.Destroy;begin  Clear;  Inherited Destroy;end;procedure TMemoryStream.Clear;begin  FSize:=0;  FPosition:=0;  SetCapacity (0);end;procedure TMemoryStream.LoadFromStream(Stream: TStream);begin  Stream.Position:=0;  SetSize(Stream.Size);  If FSize>0 then Stream.ReadBuffer(FMemory^,FSize);end;procedure TMemoryStream.LoadFromFile(const FileName: string);Var S : TFileStream;begin  S:=TFileStream.Create (FileName,fmOpenRead or fmShareDenyWrite);  Try    LoadFromStream(S);  finally    S.free;  end;end;procedure TMemoryStream.SetSize({$ifdef CPU64}const{$endif CPU64} NewSize: PtrInt);begin  SetCapacity (NewSize);  FSize:=NewSize;  IF FPosition>FSize then    FPosition:=FSize;end;function TMemoryStream.Write(const Buffer; Count: LongInt): LongInt;Var NewPos : PtrInt;begin  If (Count=0) or (FPosition<0) then    exit(0);  NewPos:=FPosition+Count;  If NewPos>Fsize then    begin    IF NewPos>FCapacity then      SetCapacity (NewPos);    FSize:=Newpos;    end;  System.Move (Buffer,(FMemory+FPosition)^,Count);  FPosition:=NewPos;  Result:=Count;end;{****************************************************************************}{*                              TBytesStream                                *}{****************************************************************************}constructor TBytesStream.Create(const ABytes: TBytes);begin  inherited Create;  FBytes:=ABytes;  SetPointer(Pointer(FBytes),Length(FBytes));  FCapacity:=Length(FBytes);end;function TBytesStream.Realloc(var NewCapacity: PtrInt): Pointer;begin  // adapt TMemoryStream code to use with dynamic array  if NewCapacity<0 Then    NewCapacity:=0  else    begin      if (NewCapacity>Capacity) and (NewCapacity < (5*Capacity) div 4) then        NewCapacity := (5*Capacity) div 4;      NewCapacity := (NewCapacity + (TMSGrow-1)) and not (TMSGROW-1);    end;  if NewCapacity=Capacity then    Result:=Pointer(FBytes)  else    begin      SetLength(FBytes,Newcapacity);      Result:=Pointer(FBytes);      if (Result=nil) and (Newcapacity>0) then        raise EStreamError.Create(SMemoryStreamError);    end;end;{****************************************************************************}{*                             TStringStream                                *}{****************************************************************************}function TStringStream.GetSize: Int64;begin  Result:=Length(FDataString);end;function TStringStream.GetPosition: Int64;begin  Result:=FPosition;end;procedure TStringStream.SetSize(NewSize: Longint);begin Setlength(FDataString,NewSize); If FPosition>NewSize then FPosition:=NewSize;end;constructor TStringStream.Create(const AString: string);begin  Inherited create;  FDataString:=AString;end;function TStringStream.Read(var Buffer; Count: Longint): Longint;begin  Result:=Length(FDataString)-FPosition;  If Result>Count then Result:=Count;  // This supposes FDataString to be of type AnsiString !  Move (Pchar(FDataString)[FPosition],Buffer,Result);  FPosition:=FPosition+Result;end;function TStringStream.ReadString(Count: Longint): string;Var NewLen : Longint;begin  NewLen:=Length(FDataString)-FPosition;  If NewLen>Count then NewLen:=Count;  SetLength(Result,NewLen);  Read (Pointer(Result)^,NewLen);end;function TStringStream.Seek(Offset: Longint; Origin: Word): Longint;begin  Case Origin of    soFromBeginning : FPosition:=Offset;    soFromEnd       : FPosition:=Length(FDataString)+Offset;    soFromCurrent   : FPosition:=FPosition+Offset;  end;  If FPosition>Length(FDataString) then    FPosition:=Length(FDataString)  else If FPosition<0 then    FPosition:=0;  Result:=FPosition;end;function TStringStream.Write(const Buffer; Count: Longint): Longint;begin  Result:=Count;  SetSize(FPosition+Count);  // This supposes that FDataString is of type AnsiString)  Move (Buffer,PChar(FDataString)[Fposition],Count);  FPosition:=FPosition+Count;end;procedure TStringStream.WriteString(const AString: string);begin  Write (PChar(Astring)[0],Length(AString));end;{****************************************************************************}{*                             TResourceStream                              *}{****************************************************************************}{$ifdef UNICODE}procedure TResourceStream.Initialize(Instance: TFPResourceHMODULE; Name, ResType: PWideChar; NameIsID: Boolean);  begin    Res:=FindResource(Instance, Name, ResType);    if Res=0 then      if NameIsID then        raise EResNotFound.CreateFmt(SResNotFound,[IntToStr(PtrInt(Name))])      else        raise EResNotFound.CreateFmt(SResNotFound,[Name]);    Handle:=LoadResource(Instance,Res);    if Handle=0 then      if NameIsID then        raise EResNotFound.CreateFmt(SResNotFound,[IntToStr(PtrInt(Name))])      else        raise EResNotFound.CreateFmt(SResNotFound,[Name]);    SetPointer(LockResource(Handle),SizeOfResource(Instance,Res));  end;constructor TResourceStream.Create(Instance: TFPResourceHMODULE; const ResName: WideString; ResType: PWideChar);  begin    inherited create;    Initialize(Instance,PWideChar(ResName),ResType,False);  end;constructor TResourceStream.CreateFromID(Instance: TFPResourceHMODULE; ResID: Integer; ResType: PWideChar);  begin    inherited create;    Initialize(Instance,PWideChar(ResID),ResType,True);  end;{$else UNICODE}procedure TResourceStream.Initialize(Instance: TFPResourceHMODULE; Name, ResType: PChar; NameIsID: Boolean);  begin    Res:=FindResource(Instance, Name, ResType);    if Res=0 then      if NameIsID then        raise EResNotFound.CreateFmt(SResNotFound,[IntToStr(PtrInt(Name))])      else        raise EResNotFound.CreateFmt(SResNotFound,[Name]);    Handle:=LoadResource(Instance,Res);    if Handle=0 then      if NameIsID then        raise EResNotFound.CreateFmt(SResNotFound,[IntToStr(PtrInt(Name))])      else        raise EResNotFound.CreateFmt(SResNotFound,[Name]);    SetPointer(LockResource(Handle),SizeOfResource(Instance,Res));  end;constructor TResourceStream.Create(Instance: TFPResourceHMODULE; const ResName: string; ResType: PChar);  begin    inherited create;    Initialize(Instance,pchar(ResName),ResType,False);  end;constructor TResourceStream.CreateFromID(Instance: TFPResourceHMODULE; ResID: Integer; ResType: PChar);  begin    inherited create;    Initialize(Instance,pchar(PtrInt(ResID)),ResType,True);  end;{$endif UNICODE}destructor TResourceStream.Destroy;  begin    UnlockResource(Handle);    FreeResource(Handle);    inherited destroy;  end;{****************************************************************************}{*                             TOwnerStream                                 *}{****************************************************************************}constructor TOwnerStream.Create(ASource: TStream);begin  FSource:=ASource;end;destructor TOwnerStream.Destroy;begin  If FOwner then    FreeAndNil(FSource);  inherited Destroy;end;{****************************************************************************}{*                             TStreamAdapter                               *}{****************************************************************************}constructor TStreamAdapter.Create(Stream: TStream; Ownership: TStreamOwnership = soReference);begin  inherited Create;  FStream:=Stream;  FOwnership:=Ownership;  m_bReverted:=false;   // mantis 15003			// http://www.tech-archive.net/Archive/German/microsoft.public.de.vc/2005-08/msg00791.html			// http://code.google.com/p/ddab-lib/wiki/TPJIStreamWrapperend;destructor TStreamAdapter.Destroy;begin  if StreamOwnership=soOwned then    FreeAndNil(FStream);  inherited Destroy;end;{$warnings off}function TStreamAdapter.Read(pv: Pointer; cb: DWORD; pcbRead: PDWORD): HResult; stdcall;var  readcount: Longint;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  if pv = nil then    begin      Result := STG_E_INVALIDPOINTER;      Exit;    end;  readcount := FStream.Read(pv^, cb);  if pcbRead <> nil then pcbRead^ := readcount;  Result := S_OK;end;function TStreamAdapter.Write(pv: Pointer; cb: DWORD; pcbWritten: PDWORD): HResult; stdcall;var  writecount: Longint;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  if pv = nil then    begin      Result := STG_E_INVALIDPOINTER;      Exit;    end;  writecount := FStream.Write(pv^, cb);  if pcbWritten <> nil then pcbWritten^ := writecount;  Result := S_OK;end;function TStreamAdapter.Seek(dlibMove: Largeint; dwOrigin: Longint; out libNewPosition: Largeint): HResult; stdcall;var  newpos: Int64;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  case dwOrigin of    STREAM_SEEK_SET: newpos := FStream.Seek(dlibMove, soBeginning);    STREAM_SEEK_CUR: newpos := FStream.Seek(dlibMove, soCurrent);    STREAM_SEEK_END: newpos := FStream.Seek(dlibMove, soEnd);    else      begin        Result := STG_E_INVALIDFUNCTION;        Exit;      end;  end;  if @libNewPosition <> nil then    libNewPosition := newpos;  Result := S_OK;end;function TStreamAdapter.SetSize(libNewSize: Largeint): HResult; stdcall;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  if libNewSize<0 then    begin      Result := STG_E_INVALIDFUNCTION;      Exit;    end;  try    FStream.Size := libNewSize;    Result := S_OK;  except    // TODO: return different error value according to exception like STG_E_MEDIUMFULL    Result := E_FAIL;  end;end;function TStreamAdapter.CopyTo(stm: IStream; cb: Largeint; out cbRead: Largeint; out cbWritten: Largeint): HResult; stdcall;var  sz: dword;  buffer : array[0..1023] of byte;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  // the method is similar to TStream.CopyFrom => use CopyFrom implementation  cbWritten := 0;  cbRead := 0;  while cb > 0 do    begin      if (cb > sizeof(buffer)) then        sz := sizeof(Buffer)      else        sz := cb;      sz := FStream.Read(buffer, sz);      inc(cbRead, sz);      stm.Write(@buffer[0], sz, @sz);      inc(cbWritten, sz);      if sz = 0 then        begin          Result := E_FAIL;          Exit;        end;      dec(cb, sz);    end;  Result := S_OK;end;function TStreamAdapter.Commit(grfCommitFlags: Longint): HResult; stdcall;begin  if m_bReverted then    Result := STG_E_REVERTED  else    Result := S_OK;end;function TStreamAdapter.Revert: HResult; stdcall;begin  m_bReverted := True;  Result := S_OK;end;function TStreamAdapter.LockRegion(libOffset: Largeint; cb: Largeint; dwLockType: Longint): HResult; stdcall;begin  Result := STG_E_INVALIDFUNCTION;end;function TStreamAdapter.UnlockRegion(libOffset: Largeint; cb: Largeint; dwLockType: Longint): HResult; stdcall;begin  Result := STG_E_INVALIDFUNCTION;end;function TStreamAdapter.Stat(out statstg: TStatStg; grfStatFlag: Longint): HResult; stdcall;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  if grfStatFlag in [STATFLAG_DEFAULT,STATFLAG_NOOPEN,STATFLAG_NONAME] then  begin    if @statstg <> nil then    begin      fillchar(statstg, sizeof(TStatStg),#0);      { //TODO handle pwcsName        if grfStatFlag = STATFLAG_DEFAULT then          runerror(217) //Result :={$ifdef windows} STG_E_INVALIDFLAG{$else}E_INVALID_FLAG{$endif}      }      statstg.dwType := STGTY_STREAM;      statstg.cbSize := FStream.Size;      statstg.grfLocksSupported := LOCK_WRITE;    end;    Result := S_OK;  end else    Result := STG_E_INVALIDFLAGend;function TStreamAdapter.Clone(out stm: IStream): HResult; stdcall;begin  if m_bReverted then    begin      Result := STG_E_REVERTED;      Exit;    end;  // don't raise an exception here return error value that function is not implemented  // to implement this we need a clone method for TStream class  Result := STG_E_UNIMPLEMENTEDFUNCTION;end;constructor TProxyStream.Create(const Stream: IStream);begin  FStream := Stream;end;function TProxyStream.Read(var Buffer; Count: Longint): Longint;begin  Check(FStream.Read(@Buffer, Count, @Result));end;function TProxyStream. Seek(const Offset: int64; Origin: TSeekOrigin): int64;begin  Check(FStream.Seek(Offset, ord(Origin), result));end;function TProxyStream.Write(const Buffer; Count: Longint): Longint;begin  Check(FStream.Write(@Buffer, Count, @Result));end;function TProxyStream.GetIStream: IStream;begin  Result := FStream;end;procedure TProxyStream.Check(err:integer);var e : EInOutError;begin  e:= EInOutError.Create('Proxystream.Check');  e.Errorcode:=err;  raise e;end;{$warnings on}
 |