1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270 |
- {
- 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
- 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;
- 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: String;
- 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(Bytes));
- 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);
- 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;
- {****************************************************************************}
- {* 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}
|