Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi WM_COPYDATA mit Records und Arrays (https://www.delphipraxis.net/101598-wm_copydata-mit-records-und-arrays.html)

Tyrael Y. 16. Okt 2007 08:37


WM_COPYDATA mit Records und Arrays
 
Hallo zusammen,

irgendwie seh ich nicht den Fehler den ich grad mache,
vielleicht weiss einer von euch Rat.

Ich sende mit WM_COPYDATA einen Record, beim Empfang ist mein Array nicht mehr gültig und ich frag mich grad warum. Glaube ich bin grad Codeblind.

Auf eine Form einen Button, einen Edit und ein Memo anbringen zum Testen.


Sender:
Delphi-Quellcode:
type
TBytes = array of Byte;
PBytes = ^TBytes;


PMyData = ^TMyData;
TMyData = packed record
  Data1   : DWord;
  Data2   : Word;
  ArrayData : PBytes;
  end;

procedure TForm1.Button1Click(Sender: TObject);
var LCopyDataStruct: TCopyDataStruct;
    LArray: PBytes;
    i: Integer;

    LStr: string;
    FensterH: HWND;
    LAnzahl: Integer;
begin
  LCopyDataStruct.cbData := StrToInt(Edit1.Text); //Anzahl
  LCopyDataStruct.dwData := self.Handle;
  LAnzahl := LCopyDataStruct.cbData;

  if assigned(LCopyDataStruct.lpdata) then
    ReallocMem(LCopyDataStruct.lpData, SizeOf(DWord) + SizeOf(Word) + LAnzahl * SizeOf(Byte))
  else
    LCopyDataStruct.lpData := AllocMem(SizeOf(DWord) + SizeOf(Word) + LAnzahl * SizeOf(Byte));

  Memo1.lines.add('');
  Memo1.lines.add('Sende:');
  Memo1.lines.add('cbData = ' + Edit1.Text);
  Memo1.lines.Add('dwData = ' + IntToStr(self.Handle));

  if assigned(LArray) then
    ReAllocMem(LArray, 4)
  else
    LArray := AllocMem(4);

  SetLength(LArray^, LCopyDataStruct.cbData);

  for i := 0 to LCopyDataStruct.cbData -1 do
  begin
    LArray^[i] := Random(200);
    LStr := LStr + ' ' + IntToStr(LArray^[i]);
  end;
  Memo1.Lines.Add('Daten:');
  Memo1.Lines.Add(LStr);


  PMyData(LCopydataStruct.lpData)^.ArrayData := Allocmem(4);
  SetLength(PMyData(LCopydataStruct.lpData)^.ArrayData^, LCopyDataStruct.cbData);
  move(LArray^[0], PMyData(LCopydataStruct.lpData)^.ArrayData^[0], LCopyDataStruct.cbData);

  FensterH := FindWindow(nil, PChar('DataSend2'));
  SendMessage(FensterH, WM_COPYDATA, 0, LongInt(@LCopyDataStruct));
end;

procedure TForm1.FormCreate(Sender: TObject);
begin
  Randomize();
end;

procedure TForm1.CopyData(var AMsg: TWMCopyData);
var i: Integer;
    LStr: string;
    LArray: PBytes;
    LAnzahl: integer;
begin
   Memo1.lines.add('');
   Memo1.Lines.Add('Empfangen :');

   Memo1.Lines.add('cbData : '+  IntToStr(AMsg.copyDatastruct^.cbData)); //Anzahl
   Memo1.Lines.Add('dwData : '+  IntToStr(Amsg.CopyDataStruct^.dwData)); //Fensterhandle


   LAnzahl := Amsg.CopyDataStruct^.dwData;

   Memo1.Lines.add('Daten:');

   if assigned(LArray) then
     ReallocMem(LArray, 4)
   else
     LArray := AllocMem(4);

   SetLength(LArray^, LAnzahl);

   move(PMyData(Amsg.copydatastruct^.lpdata)^.ArrayData^[0], LArray^[0], LAnzahl);

   for i := 0 to AMsg.CopydataStruct^.cbData -1 do
   begin
     LStr := LStr + IntToStr(LArray^[i]);
   end;
end;

Empfänger:

Delphi-Quellcode:
type
TBytes = array of Byte;
PBytes = ^TBytes;


PMyData = ^TMyData;
TMyData = packed record
  Data1   : DWord;
  Data2   : Word;
  ArrayData : PBytes;
  end;

procedure TForm1.Button1Click(Sender: TObject);
var LCopyDataStruct: TCopyDataStruct;
    LArray: PBytes;
    i: Integer;

    LStr: string;
    FensterH: HWND;
    LAnzahl: integer;
begin
  LCopyDataStruct.cbData := StrToInt(Edit1.Text); //Anzahl
  LCopyDataStruct.dwData := self.Handle;


  LAnzahl := LCopyDataStruct.cbData;

  if assigned(LCopyDataStruct.lpdata) then
    ReallocMem(LCopyDataStruct.lpData, 6 + LAnzahl)
  else
    LCopyDataStruct.lpData := AllocMem(6 + LAnzahl);

  Memo1.lines.add('');
  Memo1.lines.add('Sende:');
  Memo1.lines.add('cbData = ' + Edit1.Text);
  Memo1.lines.Add('dwData = ' + IntToStr(self.Handle));

  if assigned(LArray) then
    ReAllocMem(LArray, 4)
  else
    LArray := AllocMem(4);
  SetLength(LArray^, LCopyDataStruct.cbData);

  for i := 0 to LCopyDataStruct.cbData -1 do
  begin
    LArray^[i] := Random(200);
    LStr := LStr + ' ' + IntToStr(LArray^[i]);
  end;
  Memo1.Lines.Add('Daten:');
  Memo1.Lines.Add(LStr);

  PMyData(LCopydataStruct.lpData)^.ArrayData := AllocMem(4);
  SetLength(PMyData(LCopydataStruct.lpData)^.ArrayData^, LCopyDataStruct.cbData);
  move(LArray^[0], PMyData(LCopydataStruct.lpData)^.ArrayData^[0], LCopyDataStruct.cbData);

  FensterH := FindWindow(nil, PChar('DataSend1'));
  SendMessage(FensterH, WM_COPYDATA, 0, LongInt(@LCopyDataStruct));
end;

procedure TForm1.FormCreate(Sender: TObject);
begin
  Randomize();
end;

procedure TForm1.CopyData(var AMsg: TWMCopyData);
var i: Integer;
    LStr: string;
    LAnzahl: integer;
    LArray: PBytes;
begin
   Memo1.lines.add('');
   Memo1.Lines.Add('Empfangen :');

   Memo1.Lines.add('cbData : '+  IntToStr(AMsg.copyDatastruct^.cbData)); //Anzahl
   Memo1.Lines.Add('dwData : '+  IntToStr(Amsg.CopyDataStruct^.dwData));



   Memo1.Lines.add('Daten:');

   LAnzahl := Amsg.CopyDataStruct^.cbData;

   if assigned(LArray) then
     ReallocMem(LArray, 4)
   else
     LArray := AllocMem(4);

   SetLength(LArray^, LAnzahl);

   move(PMyData(Amsg.copydatastruct^.lpdata)^.Arraydata^[0], LArray^[0], LAnzahl); //<- hier ist das Array nicht mehr gültig und es gibt eine AccessViolation

   for i := 0 to AMsg.CopydataStruct^.cbData -1 do
   begin
     LStr := LStr + IntToStr(LArray^[i]);
   end;
end;

OregonGhost 16. Okt 2007 10:00

Re: WM_COPYDATA mit Records und Arrays
 
Habe das jetzt nur überflogen, daher nur, was mir auf Anhieb komisch vorkam:
  1. Bist du sicher, dass das Array im Record inline liegt? Ich habe dunkel in Erinnerung, das Array für diesen Fall mit array[0..0] deklarieren zu müssen (aber vielleicht war das auch bei älteren Delphiversionen). Edit: Mir kommt es so gesehen seltsam vor, dass du in deinem Record einen weiteren Zeiger auf einen neuen Speicherblock anforderst. Wird der überhaupt mitkopiert?
  2. Edit: Zurückgezogen. Das war Blödsinn :)
  3. Was gibt sizeof(deinRecord) zurück, wenn du es fertig gefüllt hast? Nur so interessehalber.

sirius 16. Okt 2007 10:06

Re: WM_COPYDATA mit Records und Arrays
 
Hmmm, da muss man ganz schön suchen.
Vermutung:
Copydatastruct.cbData muss die Länge von lpData beinhalten. Dein Record ist immer 10 Bytes groß, also muss in CBData eine 10 stehen. Ist da nur eine 5 drin, wird das Array nicht mit kopiert.

Du schickst nur den Pointer zu deinem ByteArray. Das kannst du auch über eine "normale Message" tun insofern, der Sender und Empfänger im selben Process sind.

Tyrael Y. 16. Okt 2007 10:10

Re: WM_COPYDATA mit Records und Arrays
 
Hm,

seeehr merkwürdig

SizeOf(meinRecord.data1) = 4
SizeOf(meinRecord.data2) = 2
SizeOf(meinRecord.DataArray) = 4

SizeOf(meinRecord) = 12 oO

...woher kommen die 2 Bytes?


Edit: es sind zwei verschiedene Applikationen, die miteinander kommunizieren müssen,
zwingend ist hierbei WM_COPYDATA, da der Empfänger feststeht und genau diese Infos über WM_COPDATA erwartet

sirius 16. Okt 2007 10:33

Re: WM_COPYDATA mit Records und Arrays
 
Also in D7 ist dein Record 10 Bytes groß.

OregonGhost 16. Okt 2007 10:34

Re: WM_COPYDATA mit Records und Arrays
 
Das mit den zwei Bytes ist interessant. Bei packed würde ich erwarten, dass die Padding-Bytes wegfallen. Hmm, also ist es wohl so, dass dein Array nicht mit drin liegt im Record, ja? Du könntest natürlich den schmutzigen, einfachen Weg gehen und die Daten einfach in ein Byte-Array kopieren und das per WM_COPYDATA übertragen. Da hast du auch die volle Kontrolle über das Layout.

Tyrael Y. 16. Okt 2007 11:27

Re: WM_COPYDATA mit Records und Arrays
 
Habe es mal leicht umgebaut und untersucht, die Daten in meinem Record werden richtig überstragen, daß heißt Data1 und Data2 sind richtig, das Array hat auch die richtige Adresse nach dem Empfang nur das Array hat angeblich keine Element mehr.


P.S.: Ich hatte das packed beim Record kurzfristig weggelassen und dann war es 12 Byte groß als packed record ist es 10Byte groß.



Beim folgenden Besipiel hab ich das Array der Einfachheit halber auf der Länge 3 belassen.

Sender:

Delphi-Quellcode:
procedure TForm1.Button1Click(Sender: TObject);
var LCopyDataStruct: TCopyDataStruct;
    LArray: PBytes;
    i: Integer;

    LStr: string;
    FensterH: HWND;
    LAnzahl: Integer;
begin
  LCopyDataStruct.cbData := SizeOf(TMyData);//StrToInt(Edit1.Text); //Anzahl
  LCopyDataStruct.dwData := self.Handle;
  LAnzahl := StrToInt(Edit1.Text);


  Memo1.lines.add('');
  Memo1.lines.add('Sende:');
  Memo1.lines.add('cbData = ' + Edit1.Text);
  Memo1.lines.Add('dwData = ' + IntToStr(self.Handle));

  if assigned(LArray) then
    ReAllocMem(LArray, 4)
  else
    LArray := AllocMem(4);

  SetLength(LArray^, LAnzahl);

  for i := 0 to LAnzahl -1 do
  begin
    LArray^[i] := Random(200);
    LStr := LStr + ' ' + IntToStr(LArray^[i]);
  end;
  Memo1.Lines.Add('Daten:');
  Memo1.Lines.Add(LStr);


  if assigned(LCopyDataStruct.lpdata) then
    ReallocMem(LCopyDataStruct.lpData, SizeOf(TMyData) + LAnzahl * SizeOf(Byte))
  else
    LCopyDataStruct.lpData := AllocMem(SizeOf(TMyData) + LAnzahl * SizeOf(Byte));


  PMyData(LCopydataStruct.lpData)^.Data1 := 3333;
  PMyData(LCopydataStruct.lpData)^.Data2 := 5555;
  PMyData(LCopydataStruct.lpData)^.ArrayData := Allocmem(4);
  SetLength(PMyData(LCopydataStruct.lpData)^.ArrayData^, LAnzahl);
  move(LArray^[0], PMyData(LCopydataStruct.lpData)^.ArrayData^[0], LAnzahl);


  FensterH := FindWindow(nil, PChar('DataSend7654'));
  SendMessage(FensterH, WM_COPYDATA, 0, LongInt(@LCopyDataStruct));
  Application.ProcessMessages;
end;

Empfänger:

Delphi-Quellcode:
procedure TForm1.CopyData(var AMsg: TWMCopyData);
var i: Integer;
    LStr: string;
    LAnzahl: integer;
    LArray: PBytes;
begin
   Memo1.lines.add('');
   Memo1.Lines.Add('Empfangen :');

   Memo1.Lines.add('cbData : '+  IntToStr(AMsg.copyDatastruct^.cbData)); //Anzahl
   Memo1.Lines.Add('dwData : '+  IntToStr(Amsg.CopyDataStruct^.dwData));



   Memo1.Lines.add('Daten:');

   LAnzahl := Amsg.CopyDataStruct^.cbData;

   if assigned(LArray) then
     ReallocMem(LArray, 4)
   else
     LArray := AllocMem(4);

   SetLength(LArray^, 3);

   move(PMyData(Amsg.copydatastruct^.lpdata)^.Arraydata^[0], LArray^[0], 3); //<- das array hat die richtige Adresse aber kein Elemente mehr


   for i := 0 to AMsg.CopydataStruct^.cbData -1 do
   begin
     LStr := LStr + IntToStr(LArray^[i]);
   end;
end;

Tyrael Y. 16. Okt 2007 12:11

Re: WM_COPYDATA mit Records und Arrays
 
Zitat:

Zitat von OregonGhost
Hmm, also ist es wohl so, dass dein Array nicht mit drin liegt im Record, ja? Du könntest natürlich den schmutzigen, einfachen Weg gehen und die Daten einfach in ein Byte-Array kopieren und das per WM_COPYDATA übertragen. Da hast du auch die volle Kontrolle über das Layout.

Ich habe meinen Record mal testweise folgendermassen deklariert

Delphi-Quellcode:
type
TBytes = array of Byte;

PMyData = ^TMyData;
TMyData = packed record
  Data1   : DWord;
  Data2   : Word;
  ArrayData : TBytes;
  end;
Gefüllt wird das Array richtig, nach dem Empfang hat das Array auch in diesem Fall keine Elemente mehr. Wenn ich statisches Array nehme funktioniert es problemlos.

Sobald ich ein dynamisches Array nehme habe, ist das Array nach dem Empfang leer.
Dies muss doch auch mit dynamischen Arrays möglich sein. Irgendwo ist der Wurm drin und ich seh es einfach nicht.

OregonGhost 16. Okt 2007 12:25

Re: WM_COPYDATA mit Records und Arrays
 
Da ich in Delphi nicht so drin stecke (und auch derzeit keins installiert habe): Was passiert denn, wenn du das Array tatsächlich als array[0..0] deklarierst (und dann mit SetLength die Länge veränderst)? Das ist auch in C der (hässliche) übliche Weg, ein dynamisches Array mit in eine Struktur zu packen.

Tyrael Y. 16. Okt 2007 12:41

Re: WM_COPYDATA mit Records und Arrays
 
Wenn ich es als

Delphi-Quellcode:
PMyData = ^TMyData;
TMyData = packed record
  Data1   : DWord;
  Data2   : Word;
  ArrayData : array[0..0] of Byte;
  end;
deklariere, kann ich kein SetLength benutzen, die Meldung inkompatible Typen erscheint dann.

OregonGhost 16. Okt 2007 12:45

Re: WM_COPYDATA mit Records und Arrays
 
Nanu? Ging das nicht mal? :)

Tyrael Y. 16. Okt 2007 14:50

Re: WM_COPYDATA mit Records und Arrays
 
Also meine Vermutung ist jetzt, daß es mit dynamischen Arrays nicht möglich ist.

Wie komme ich darauf?

Ich hab mir den Record zur Laufzeit angeschaut.

zB.:

Delphi-Quellcode:
type
TBytes = array of Byte;

TMyData = packed record
  Data1   : DWord;
  Data2   : Word;
  ArrayData : TBytes;
end;


Record.Data1 Adresse $983FB4
Record.Data2 Adresse $983FB8 - Abstand wie in Definition 4 Bytes
Record.Array Adresse $983FBA - Abstand wie in Definition 2 Bytes

Record.Array[0] liegt dagegen ganz wo anders zB. auf $982070,
also viel weiter unten im Adressbereich

Record.Array[1]-[n] sind die Folgeadressen von $982070.


Wenn man jetzt die Daten über WM_COPYDATA versendet, geht wogl der Bezug von Record.Array zu Record.Array[0] verloren. Das Array "weiss" nicht mehr wo das erste Element liegt.

Kann mir das ein Wissender bestätigen?

Zacherl 16. Okt 2007 15:05

Re: WM_COPYDATA mit Records und Arrays
 
Kann ich bestätigen. Daher kann man ja auch keine Strings schicken. Wenn muss man sie als ShortString oder String[255] deklarieren.
Evtl geht es aber, dass du nur die Adresse des ersten Elements, die Länge des Arrays und die eigene ProcessID sendest. Der Empfänger kann dann mittels OpenProcess, ReadProcessMemory direkt auf das Array in deinem Sender Prozess zugreifen.

hitzi 2. Nov 2007 07:17

Re: WM_COPYDATA mit Records und Arrays
 
Mit den hier angesprochenen Lösungen hat es bei mir geklappt: http://delphi.about.com/od/windowssh...m_copydata.htm


Alle Zeitangaben in WEZ +1. Es ist jetzt 14:27 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