123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332 |
- {
- 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), get_caller_frame(get_frame);
- end;
- procedure TStream.WriteNotImplemented;
- begin
- raise EStreamError.CreateFmt(SStreamNoWriting, [ClassName]) at get_caller_addr(get_frame), get_caller_frame(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), get_caller_frame(get_frame);
- end;
- procedure TStream.FakeSeekForward(Offset: Int64; const Origin: TSeekOrigin; const Pos: Int64);
- 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);
- Var
- r,t : longint;
- begin
- t:=0;
- repeat
- r:=Read(PByte(@Buffer)[t],Count-t);
- inc(t,r);
- until (t=Count) or (r<=0);
- if (t<Count) then
- Raise EReadError.Create(SReadError);
- end;
- procedure TStream.WriteBuffer(const Buffer; Count: Longint);
- var
- r,t : Longint;
- begin
- T:=0;
- Repeat
- r:=Write(PByte(@Buffer)[t],Count-t);
- inc(t,r);
- Until (t=count) or (r<=0);
- if (t<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: Longint;
- 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: Longint);
- 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: Longint);
- var
- ResSize,TmpResSize : Longint;
- 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
- // We set the position afterwards, because the size can also be larger.
- if not FileTruncate(FHandle,NewSize) then
- Raise EInOutError.Create(SStreamSetSize);
- Position:=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
- begin
- {$if declared(GetLastOSError)}
- raise EFCreateError.createfmt(SFCreateErrorEx,[AFileName, SysErrorMessage(GetLastOSError)])
- {$else}
- raise EFCreateError.createfmt(SFCreateError,[AFileName])
- {$endif}
- end
- else
- begin
- {$if declared(GetLastOSError)}
- raise EFOpenError.Createfmt(SFOpenErrorEx,[AFilename, SysErrorMessage(GetLastOSError)]);
- {$else}
- raise EFOpenError.Createfmt(SFOpenError,[AFilename]);
- {$endif}
- end;
- 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;
- Var
- GC : PtrInt;
- begin
- If NewCapacity<0 Then
- NewCapacity:=0
- else
- begin
- GC:=FCapacity + (FCapacity div 4);
- // if growing, grow at least a quarter
- if (NewCapacity>FCapacity) and (NewCapacity < GC) then
- NewCapacity := GC;
- // 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 NewSize: Int64{$else}NewSize: LongInt{$endif});
- 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.GetDataString: string;
- begin
- Result:=FEncoding.GetAnsiString(Bytes,0,Size);
- end;
- function TStringStream.GetUnicodeDataString: UnicodeString;
- begin
- Result:=FEncoding.GetString(Bytes, 0, Size);
- end;
- constructor TStringStream.Create(const AString: string = '');
- begin
- Create(AString,TEncoding.Default, False);
- end;
- constructor TStringStream.Create(const ABytes: TBytes);
- begin
- inherited Create(ABytes);
- FEncoding:=TEncoding.Default;
- FOwnsEncoding:=False;
- end;
- constructor TStringStream.CreateRaw(const AString: RawByteString);
- var
- CP: TSystemCodePage;
- begin
- CP:=StringCodePage(AString);
- if (CP=CP_ACP) or (CP=TEncoding.Default.CodePage) then
- begin
- FEncoding:=TEncoding.Default;
- FOwnsEncoding:=False;
- end
- else
- begin
- FEncoding:=TEncoding.GetEncoding(CP);
- FOwnsEncoding:=True;
- end;
- inherited Create(BytesOf(AString));
- end;
- constructor TStringStream.Create(const AString: string; AEncoding: TEncoding; AOwnsEncoding: Boolean);
- begin
- FOwnsEncoding:=AOwnsEncoding and not TEncoding.IsStandardEncoding(AEncoding);
- FEncoding:=AEncoding;
- Inherited Create(AEncoding.GetAnsiBytes(AString));
- end;
- constructor TStringStream.Create(const AString: string; ACodePage: Integer);
- begin
- Create(AString,TEncoding.GetEncoding(ACodePage),true);
- end;
- constructor TStringStream.Create(const AString: UnicodeString);
- begin
- Create(AString,TEncoding.Unicode,false);
- end;
- constructor TStringStream.Create(const AString: UnicodeString; AEncoding: TEncoding; AOwnsEncoding: Boolean);
- begin
- FOwnsEncoding:=AOwnsEncoding and not TEncoding.IsStandardEncoding(AEncoding);
- FEncoding:=AEncoding;
- Inherited Create(AEncoding.GetBytes(AString));
- end;
- constructor TStringStream.Create(const AString: UnicodeString; ACodePage: Integer);
- begin
- Create(AString,TEncoding.GetEncoding(ACodePage),true);
- end;
- destructor TStringStream.Destroy;
- begin
- If FOwnsEncoding then
- FreeAndNil(FEncoding);
- inherited Destroy;
- end;
- function TStringStream.ReadString(Count: Longint): string;
- begin
- Result:=ReadAnsiString(Count);
- end;
- function TStringStream.ReadUnicodeString(Count: Longint): UnicodeString;
- Var
- NewLen,SLen : Longint;
- begin
- NewLen:=Size-FPosition;
- If NewLen>Count then NewLen:=Count;
- Result:=FEncoding.GetString(FBytes,FPosition,NewLen);
- end;
- procedure TStringStream.WriteString(const AString: string);
- begin
- WriteAnsiString(AString);
- end;
- procedure TStringStream.WriteUnicodeString(const AString: UnicodeString);
- Var
- B: TBytes;
- begin
- B:=FEncoding.GetBytes(AString);
- if Length(B)>0 then
- WriteBuffer(B[0],Length(B));
- end;
- function TStringStream.ReadAnsiString(Count: Longint): AnsiString;
- Var
- NewLen : Longint;
- begin
- NewLen:=Size-FPosition;
- If NewLen>Count then NewLen:=Count;
- Result:=FEncoding.GetAnsiString(FBytes,FPosition,NewLen);
- Inc(FPosition,NewLen);
- end;
- procedure TStringStream.WriteAnsiString(const AString: AnsiString);
- Var
- B: TBytes;
- begin
- B:=FEncoding.GetAnsiBytes(AString);
- if Length(B)>0 then
- WriteBuffer(B[0],Length(B));
- end;
- {****************************************************************************}
- {* TRawByteStringStream *}
- {****************************************************************************}
- constructor TRawByteStringStream.Create(const aData: RawByteString);
- begin
- Inherited Create;
- If Length(aData)>0 then
- begin
- WriteBuffer(aData[1],Length(aData));
- Position:=0;
- end;
- end;
- function TRawByteStringStream.DataString: RawByteString;
- begin
- Result:='';
- SetLength(Result,Size);
- if Size>0 then
- Move(Memory^, Result[1], Size);
- end;
- function TRawByteStringStream.ReadString(Count: Longint): RawByteString;
- Var
- NewLen : Longint;
- begin
- NewLen:=Size-FPosition;
- If NewLen>Count then NewLen:=Count;
- Result:='';
- if NewLen>0 then
- begin
- SetLength(Result, NewLen);
- Move(FBytes[FPosition],Result[1],NewLen);
- inc(FPosition,Newlen);
- end;
- end;
- procedure TRawByteStringStream.WriteString(const AString: RawByteString);
- begin
- if Length(AString)>0 then
- WriteBuffer(AString[1],Length(AString));
- end;
- {****************************************************************************}
- {* TResourceStream *}
- {****************************************************************************}
- {$ifdef FPC_OS_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 FPC_OS_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 FPC_OS_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/TPJIStreamWrapper
- end;
- 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: DWORD; out libNewPosition: LargeUint): HResult; stdcall;
- var
- newpos: QWord;
- 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: LargeUint): 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: LargeUint; out cbRead: LargeUint; out cbWritten: Largeuint): 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: DWORD): 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: LargeUint; cb: LargeUint; dwLockType: DWORD): HResult; stdcall;
- begin
- Result := STG_E_INVALIDFUNCTION;
- end;
- function TStreamAdapter.UnlockRegion(libOffset: LargeUint; cb: LargeUint; dwLockType: DWORD): HResult; stdcall;
- begin
- Result := STG_E_INVALIDFUNCTION;
- end;
- function TStreamAdapter.Stat(out statstg: TStatStg; grfStatFlag: DWORD): 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_INVALIDFLAG
- end;
- 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), QWord(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}
|