Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Delphi Datei stückchenweise mit TFileStream einlesen (https://www.delphipraxis.net/9779-datei-stueckchenweise-mit-tfilestream-einlesen.html)

Luckie 3. Okt 2003 23:34


Datei stückchenweise mit TFileStream einlesen
 
Ich versuche eien Datei stückchenweise mit ReadBuffer einzulesen, den Buffer zu verarbeiten und dann das Ganze wieder zu schreiben.
Delphi-Quellcode:
var
  SrcBuffer, DestBuffer: cardinal;
begin
  while SrcStream.Position < SrcStream.Size do
  begin
    SrcStream.ReadBuffer(SrcBuffer, 1024);
    EncodeBuffer(SrcBuffer, DestBuffer, 1024);
    DestStream.WriteBuffer(DestBuffer, 1024);
  end;
Aber das will nicht so recht. Ich bekomme da immer einen Stream.Lesefehler. Es hadelt sich um alle möglichen Dateien, binär, Text, typisierte, ...

So geht es auch nicht:
Delphi-Quellcode:
var
  SrcBuffer, DestBuffer: Pointer;
begin
  GetMem(SrcBuffer, 1024);
  GetMem(DestBuffer, 1024);
  while DestStream.Size < SrcStream.Size + sizeof(TFileHeader) do
  begin
    SrcStream.ReadBuffer(SrcBuffer, 1024);
    EncodeBuffer(SrcBuffer, DestBuffer, 1024);
    DestStream.WriteBuffer(DestBuffer, 1024);
  end;
Gleiche Fehler.

Kamil 3. Okt 2003 23:43

Re: Datei stückchenweise mit TFileStream einlesen
 
In eine cardinal Variable passen nur 32 Bit...
Nimm ein Array:
Code:
var
  SrcBuffer, DestBuffer: array[0..1023] of byte;
begin
  while SrcStream.Position < SrcStream.Size do
  begin
    SrcStream.ReadBuffer(SrcBuffer[0], 1024);
    EncodeBuffer(SrcBuffer[0], DestBuffer[0], 1024);
    DestStream.WriteBuffer(DestBuffer[0], 1024);
  end;
Du kannst auch mit pointer arbeiten:
Code:
SrcBuffer, DestBuffer: pointer;
begin
  getmem(SrcBuffer, 1024);
  getmem(DeasBuffer, 1024);
  try
    while SrcStream.Position < SrcStream.Size do
    begin
      SrcStream.ReadBuffer(SrcBuffer^, 1024);
      EncodeBuffer(SrcBuffer^, DestBuffer^, 1024);
      DestStream.WriteBuffer(DestBuffer^, 1024);
    end;
  finally
    freemem(SrcBuffer);
    freemem(DestBuffer);
  end;

Luckie 3. Okt 2003 23:49

Re: Datei stückchenweise mit TFileStream einlesen
 
Ha, da war ich doch gar nicht soweit weg. Blos die Abbruchbedingung stimmt noch nicht ganz. Ich schreibe in den ZielStream erst einen Header und dann den QuellStream. Nur mit
Delphi-Quellcode:
while SrcStream.Position < SrcStream.Size do
Bekomme ich zum Schluß wieder einen Stream-Lesefehler.

Kamil 3. Okt 2003 23:57

Re: Datei stückchenweise mit TFileStream einlesen
 
Code:
var
  Len: integer;
  ...
begin
...
  while SrcStream.Position < SrcStream.Size do
  begin
    Len:=(SrcStream.Size-SrcStream.Position) mod 1025;
    SrcStream.ReadBuffer(SrcBuffer^, Len);
    EncodeBuffer(SrcBuffer^, DestBuffer^, Len);
    DestStream.WriteBuffer(DestBuffer^, Len);
  end;
...

Luckie 4. Okt 2003 00:03

Re: Datei stückchenweise mit TFileStream einlesen
 
Der Schuß ins Blaue ging daneben. Len ist immer 0. Er liest und schreibt also gar nichts.

Kamil 4. Okt 2003 00:10

Re: Datei stückchenweise mit TFileStream einlesen
 
Code:
if SrcStream.Size-SrcStream.Position>1024 then
  Len=1024
else
  Len:=SrcStream.Size-SrcStream.Position;
Bin schon ein wenig müde...

Luckie 4. Okt 2003 00:14

Re: Datei stückchenweise mit TFileStream einlesen
 
Perfekt, der saß genau im Bulleseye. Besten Dank.

Kamil 4. Okt 2003 00:33

Re: Datei stückchenweise mit TFileStream einlesen
 
Da fällt mir noch ein:
Falls du statt ReadBuffer Read nimmst, dann kannst du es so machen:

Code:
var
  Len: integer;
  ...
begin
...
  while SrcStream.Position < SrcStream.Size do
  begin
    Len:=SrcStream.Read(SrcBuffer^, 1024);
    EncodeBuffer(SrcBuffer^, DestBuffer^, Len);
    DestStream.WriteBuffer(DestBuffer^, Len);
  end;
...

Luckie 4. Okt 2003 02:49

Re: Datei stückchenweise mit TFileStream einlesen
 
Hm, Jetzt hab eich es so, aber die zweite Lösung wäre wohl performanter, da die if-Abfrage wegfällt. Und da waren sie wieder meine drei Problem... Stream-Lesefehler.

Christian Seehase 4. Okt 2003 03:00

Re: Datei stückchenweise mit TFileStream einlesen
 
Moin Luckie,

komisch.
An welcher Stelle denn?

Im Prinzip müsste es auch so gehen:

Delphi-Quellcode:
  iLen := fsIN.Read(sBuf[1],iBlockSize);
  while iLen > 0 do begin
    fsOUT.Write(sBuf[1],iLen);
    iLen := fsIN.Read(sBuf[1],iBlockSize);
  end;
Statt eines Strings wie bei mir kannst Du natürlich auch einen anderen Buffer (array of char) nehmen.
Lesefehler dürftest Du nur bei ReadBuffer bekommen, nicht bei Read.

Luckie 4. Okt 2003 03:03

Re: Datei stückchenweise mit TFileStream einlesen
 
Und genau das sind meine zwei andern Probleme: Es müßte gehen und ich verstehe nicht, warum es nicht geht. :roll:

Christian Seehase 4. Okt 2003 03:06

Re: Datei stückchenweise mit TFileStream einlesen
 
Moin Luckie,

bekommst Du wirklich bei SrcStream.Read den Fehler? :shock:

Wie sieht die Routine denn bei Dir aus.
Am Besten incl. Datendeklaration, und Dateigrösse (genau)

Luckie 4. Okt 2003 03:16

Re: Datei stückchenweise mit TFileStream einlesen
 
Jupp, ich bekomme einen. Deklarationen wie gehabt (siehe oben). UInd ausgesehen hat sie wie von Kamil gepostet. Dateigröße: 49.955.924 Bytes. GetMem habe ich auch aufgerufen.

Keinen Plan. Habe es aber schon wieder rückgängig gemacht.

Wenn ich die if-Konstruktion nehme und statt 1024 2048 oder 4096 einsetze, bekomme ich eine AccessViolation zum Schluß.

negaH 4. Okt 2003 11:31

Re: Datei stückchenweise mit TFileStream einlesen
 
Delphi-Quellcode:
var
  Len: integer;
  Buffer: array[0..1023] of Byte;
  ...
begin
...
  while SrcStream.Position < SrcStream.Size do
  begin
    Len := SrcStream.Size - SrcStream.Position;
    if Len > SizeOf(Buffer) then Len := SizeOf(Buffer);
    SrcStream.Read(Buffer, Len);
    EncodeBuffer(Buffer, Buffer, Len);
    DestStream.WriteBuffer(Buffer, Len);
  end;
...
Wichtig! EncodeBuffer() sollte nach Möglichkeit inplaced arbeiten.

oder
Delphi-Quellcode:
var
  Len: integer;
  Buffer: array[0..1023] of Byte;
  ...
begin
...
  repeat
    Len := SrcStream.Read(Buffer, SizeOf(Buffer)); // .Read statt .ReadBuffer um Exceptions zu vermeiden
    EncodeBuffer(Buffer, Buffer, Len);
    DestStream.WriteBuffer(Buffer, Len);
  until Len = 0;
...
end;
Dies vermeidet die Seek-Aufrufe in den Dateistream durch das OS um .Position und .Size zu berechnen. Dadurch wird also der Dateistream nur sequentiell gelesen und dies ist erheblich schneller !! Probier es aus mit großen Files.

Gruß Hagen

Luckie 4. Okt 2003 12:01

Re: Datei stückchenweise mit TFileStream einlesen
 
Geil. Werde ich gleich ausprobieren. Besten Dank.

PS: Steht das im DEC irgendwo, dass EncodeBuffer inplaced arbeiten sollte? Oder ist das allgemeines Kryptologiewissen?

Luckie 4. Okt 2003 12:14

Re: Datei stückchenweise mit TFileStream einlesen
 
So jetzt haben wir den Salat, jetzt bekomme ich unterschiedliche Hash-Werte nach dem Entschlüsseln und die entschlüsselte Datei ist dann natürlich nicht zu gebrauchen.

Luckie 4. Okt 2003 12:18

Re: Datei stückchenweise mit TFileStream einlesen
 
Entschlüsseln:
Delphi-Quellcode:
          // ignore the file header
          SrcStream.Seek(sizeof(TFileHeader), soFromBeginning);
          // assign the Cipherclass
          CipherClass := Cipher;
          // prepare for decoding
          with CipherClass.Create(PW, nil) do
          begin
            Mode := cmCFB;
            InitKey(PW, nil);
            // as long as we have not reached the end of the source stream
            repeat
              Len := SrcStream.Read(Buffer, SizeOf(Buffer));
              EncodeBuffer(Buffer, Buffer, Len);
              DestStream.WriteBuffer(Buffer, Len);
            until Len = 0;
          end;
Verschlüsseln:
Delphi-Quellcode:
          // write the head into the stream
          DestStream.Write(FileHeader, sizeof(TFileHeader));
          //  prpare vor encoding
          with CipherClass.Create(PW, nil) do
          begin
            Mode := cmCFB;
            InitKey(PW, nil);
            // as long as Len has something to read
            repeat
              Len := SrcStream.Read(Buffer, SizeOf(Buffer));
              EncodeBuffer(Buffer, Buffer, Len);
              DestStream.WriteBuffer(Buffer, Len);
            until Len = 0;
          end;
Was habe ich übersehen?

negaH 4. Okt 2003 12:38

Re: Datei stückchenweise mit TFileStream einlesen
 
Zitat:

Steht das im DEC irgendwo, dass EncodeBuffer inplaced arbeiten sollte? Oder ist das allgemeines Kryptologiewissen?
Das steht nicht im DEC, du weisst DEC ist NUR eine Sammlung von effizient programmierten Algorithmen.

Wird inplaced gearbeitet so werden die Daten im Buffer implizit mit deren verschlüsseltem Produkt überschrieben. Da .EncodeBuffer() nicht wissen kann ob der Anwender seine Daten geschützt haben will darf .EncodeBuffer() auch nicht den Sourcebuffer von sich aus mit Müll füllen. Dies müsste dann schon deine Funktion machen damit eben keine sicherheitsrelevanten Daten im Speicher verbleiben. Übrigens: dies ist auch der Grund warum ich den Buffer NICHT dynamisch alloziere sondern im Stack ablege. Bei dynamischen Speicher müsstest du wiederum dafür Sorge tragen das er wiped wird. Im Stack wird mit enorm hoher Wahrscheinlichkeit der Stackspeicher durch nachfolgende Funktionen nochmals überschrieben. Geht man davon aus das es heute Techniken gibt die aus benutzten Speicherzellen denoch deren Inhalt vor 100 Schreibänderungen ermitteln können, so wird klar das der Stack zB. einer der Speicherbereiche ist der die meisten Schreibänderungen hat.
Die Lebenszeit der sichertsrelevanten Daten bei inplaced Codierungen ist die kürzt mögliche.
Das Tracen des Stackes ist eine der schwierigsten Operationen eines guten Debuggers.
All diese Überlegungen kann ein Programmierer ohne viel Aufwand umsetzen um seine Kryptofunktionen sicherer zu machen.

Zitat:

So jetzt haben wir den Salat, jetzt bekomme ich unterschiedliche Hash-Werte nach dem Entschlüsseln und die entschlüsselte Datei ist dann natürlich nicht zu gebrauchen.
Ich habe deinen Source noch nicht genauer analysiert. Es ist aber wichtig das der Stream-Dateizeiger korrekt repositioniert wird. Also, nach Hash.CalcStream(Stream) muß Stream.Position := 0 gesetzt werden. Dies ist ein häufig gemachter Anfängerfehler den ich hier nur erwähne weil auch Profi's wie du ihn von Zeit zu Zeit mal machen :)


Gruß hagen

negaH 4. Okt 2003 12:43

Re: Datei stückchenweise mit TFileStream einlesen
 
Zitat:

Was habe ich übersehen?
Zitat:

Verschlüsseln ..... EncodeBuffer()
Zitat:

Entschlüsseln ..... EncodeBuffer()
Hm, irgendwas stimmt da aber nicht ? ;)

Gruß Hagen


Alle Zeitangaben in WEZ +1. Es ist jetzt 07:44 Uhr.

Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz