Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Win32/Win64 API (native code) (https://www.delphipraxis.net/17-win32-win64-api-native-code/)
-   -   ReadProcessMemory -> Problem (https://www.delphipraxis.net/155265-readprocessmemory-problem.html)

Xerogon 15. Okt 2010 12:09

ReadProcessMemory -> Problem
 
Hi, ich hoffe ich bin hier in der richtigen Sektion :?
Also mein Problem ist, das ich bei Verwendung von ReadProcessMemory eine Exception bekomme.

Delphi-Quellcode:
var
  ThreadID, ProcessID: Integer;
  WindowHWnd: HWnd;
  write: Cardinal;
  WindowHandle: THandle;
  P: PChar; //Variable für ReadProcessMemory

...

Procedure OpenMemory;
begin
  While WindowHWnd = 0 do
  begin
    WindowHWnd := FindWindow(nil, 'Spider Solitär'); //FensterHandle suchen
    sleep(100);
  end;

  ThreadID := GetWindowThreadProcessID(WindowHWnd, @ProcessID);     //processID bestimmen
  WindowHandle := OpenProcess(PROCESS_ALL_ACCESS, false, ProcessID); //WindowHandle suchen
  Writeln('ThreadID: ' + IntToStr(ThreadID) + ' /  ' + 'ProcessID: ' + IntToStr(ProcessID) + ' /  ' + 'WindowHandle: ' + IntToStr(WindowHandle)); //sollte klar sein ;)
end;

Function ReadMemory: PChar;
begin
try
  //Speicher reservieren
  GetMem(p, SizeOf(P));
  //Speicher auslesen --> Error
  ReadProcessMemory(WindowHandle, Ptr($01324480), @P, SizeOf(P), write); //Aus gelesener Wert befindet sich nun in P, oder?
  Result := P;
  //Speicher freigeben + Handle schließen
finally
  FreeMem(P);
  CloseHandle(WindowHandle);
end;

begin
  OpenMemory;
  Writeln(ReadMemory); //--> geht nicht
  Readln;
end.
so, also bei der Funktion "Readmemory" stürzt er ab. Warum?
Mit WriteProcessMemory kann ich ohne Probleme arbeiten.

Danke :wink:

himitsu 15. Okt 2010 12:36

AW: ReadProcessMemory -> Problem
 
Zitat:

Zitat von Xerogon (Beitrag 1055913)
Also mein Problem ist, das ich bei Verwendung von ReadProcessMemory eine Exception bekomme.

Und die allseits beliebte Frage "Welche Exception, bzw. welche Meldung wird denn angezeigt?"


Sicher daß $01324480 stimmt?
(Speicheradressen sind selten statisch)


Zitat:

Aus gelesener Wert befindet sich nun in P, oder?
Joar, eventuell.
Ob was ausgelesen wurde erfährt man über das Result von ReadProcessMemory und wenn, dann das Wieviel über deine "write"-Variable.
Und falls nichts ausgelesen wurde, dann bekommt man über GetLastError den Grund dafür.
(daß auch ständig Leute das Result in die Fehlerbehandlung (GetLastError) bei den WinAPIs vergessen)

Aber SizeOf(P) ließt nur genau 4 Byte und nicht einen ganzen Text.
Und wenn nun innerhalb dieser 4 Byte kein #0, als Ende-Markierung vorkommt, dann geht der Zugriff als PChar schief, da das auslesen so lange ließt, bis es eine #0 findet und wenn dieses fehlt, dann knallt's.

Xerogon 15. Okt 2010 12:37

AW: ReadProcessMemory -> Problem
 
Kann ich nicht lesen. Die Konsole geht zu schnell zu. :(

Zitat:

Ja klar,
du ließt irgendeinen Wert (eventuell eine Adresse, aber jedenfalls keinen "Text") aus dem Speicher einer fremden Anwendung.
(Speicheradressen sind nur innerhalb der Anwendung gültig, von wo sie kommen)
Und willst dann über diese Adresse im Adressraum deiner Anwendung als PChar auf (eventuell nicht existierenden) Speicher deiner Anwendung zugreifen.
Ja, aber als Übergabewert wird ja ein Pointer verlangt.

Wie sähe denn die Zeile richtig aus, oder welchen Datentyp sollte ich dafür nehmen?

Edit: Danke, es klappt. Habe nun Integer statt PChar verwendet

himitsu 15. Okt 2010 12:46

AW: ReadProcessMemory -> Problem
 
Zitat:

Zitat von Xerogon (Beitrag 1055918)
Kann ich nicht lesen. Die Konsole geht zu schnell zu. :(

Debuggen?

Kennst du schon die F7-Taste und seine Freunde?

Luckie 15. Okt 2010 13:00

AW: ReadProcessMemory -> Problem
 
Füge am Ende des Programm ein Readln ein. Dann musst du mit einem Tastendruck die Konsole schließen. Oder starte das Programm aus der Konsole.

Und es wird ein Zeiger auf einen Speicherbereich erwartet,m der natürlich groß genug sein muss.

Sir Rufo 15. Okt 2010 13:32

AW: ReadProcessMemory -> Problem
 
Zitat:

Zitat von Luckie (Beitrag 1055923)
Füge am Ende des Programm ein Readln ein. Dann musst du mit einem Tastendruck die Konsole schließen. Oder starte das Programm aus der Konsole.

Und es wird ein Zeiger auf einen Speicherbereich erwartet,m der natürlich groß genug sein muss.

Das ReadLn hat er ja am Ende schon drin, bringt aber nix, wenn das Prog vorher abkackt ;)
Sonst einfach mal ein try..except Block der die exception ausgibt

himitsu 15. Okt 2010 13:42

AW: ReadProcessMemory -> Problem
 
Das haben die "neueren" Projektvorlagen zwar schon drin
und ein bissl abgewandelt, isses auch nutzbar. :angle:
Delphi-Quellcode:
program Project1;

{$APPTYPE CONSOLE}

uses
  SysUtils;

begin
  try
    { TODO -oEntwickler -cKonsole Main : Hier Code einfügen }
  except
    on E: Exception do
    begin
      WriteLn(E.ClassName, ': ', E.Message);
      ReadLn;
    end;
  end;
end.

Aund ansonsten kann dir keiner sagen wie das nun richtig ausgelesen wird,
da wir noch nicht erfahren haben, was genau dort ausgelesen werden soll, bzw. in welchem Format diese Daten vorliegen.

Luckie 15. Okt 2010 13:49

AW: ReadProcessMemory -> Problem
 
Seit wann werfen API-Funktionen Exceptions?

Sir Rufo 15. Okt 2010 13:54

AW: ReadProcessMemory -> Problem
 
Wenn keine Exception geworfen wird, warum wird dann sein ReadLn am Ende nicht ausgeführt? :gruebel:

Xerogon 15. Okt 2010 14:00

AW: ReadProcessMemory -> Problem
 
Es klappt nun.
Ich habe einfach PChar mit integer ausgetauscht.

Quellcode, der jetzt geht:

Delphi-Quellcode:
program Tut;

{$APPTYPE CONSOLE}

uses
  SysUtils,
  Windows;

var
  ThreadID, ProcessID: Integer;
  WindowHWnd: HWnd;
  write: Cardinal;
  WindowHandle: THandle;
  P: Integer; //Variable für ReadProcessMemory

Procedure OpenMemory;
begin
  While WindowHWnd = 0 do
  begin
    WindowHWnd := FindWindow(nil, 'Spider Solitär');
    sleep(100);
  end;

  ThreadID := GetWindowThreadProcessID(WindowHWnd, @ProcessID);
  WindowHandle := OpenProcess(PROCESS_ALL_ACCESS, false, ProcessID);
  Writeln('ThreadID: ' + IntToStr(ThreadID) + ' / ' + 'ProcessID: ' + IntToStr(ProcessID) + ' / ' + 'WindowHandle: ' + IntToStr(WindowHandle));
end;

Function ReadMemory: Integer;
var New: Integer;
begin
try

  ReadProcessMemory(WindowHandle, Ptr($04D4480), @P, SizeOf(P), write);
  New := P+$10;
  ReadProcessMemory(WindowHandle, Ptr(New), @P, SizeOf(P), write);
  New := P+$8c;
  ReadProcessMemory(WindowHandle, Ptr(New), @P, SizeOf(P), write);

  Result := P;

finally
  CloseHandle(WindowHandle);
end;
end;

begin
  OpenMemory;
  Writeln(ReadMemory);
  Readln;
end.


Alle Zeitangaben in WEZ +1. Es ist jetzt 10:45 Uhr.
Seite 1 von 2  1 2      

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 by Thomas Breitkreuz