![]() |
Delphi-Version: 10.2 Tokyo
Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Hallo,
ich bin gerade mal wieder leicht am Verzweifeln. Ich erstelle zurzeit eine Delphi-DLL mit einem ganzen Haufen von Funktionen, die sowohl in einem Delphi-Programm als auch in VBA genutzt werden sollen. Daher verwende ich in der DLL AnsiString bzw. PAnsiChar zur Übergabe von String-Parametern. Die größten Hindernisse habe ich dabei bereits umschifft, doch im Moment kämpfe ich mit einem Phänomen, das ich mir nicht erklären kann. Die DLL-Funktion sieht dabei wie folgt aus:
Delphi-Quellcode:
Im DelphiProgramm steht dann folgendes:
Function UserHatRecht(cPObj, cPRecht : PAnsiChar): Boolean; stdCall;
var cSql : String; lResult : Boolean; cObj, cRecht : String; begin cObj := String(cPObj); cRecht := String(cPRecht); Result := false; lResult := False; // mittelglroßes SQL-Statement entfernt, Result := lResult; end;
Delphi-Quellcode:
Kommentiere ich die Zeile aus, in der die Zugriffsverletzung passiert, läuft das Programm ohne Probleme weiter.
interface
Function UserHatRecht( cObj, cRecht : string): Boolean; Function DLL_UserHatRecht( cObj, cRecht : PAnsiChar): Boolean; stdCall; external 'FSGVBA.dll' name 'UserHatRecht'; implementation function HoleSpeicher( cVal : string) : PAnsiChar; var pPtr : Pointer; nLen : Integer; cAVal : AnsiString; begin cAVal := AnsiString(cVal); nLen := Length(cAVal); GetMem( pPtr, nLen); StrLCopy(pPtr, PAnsiChar(cAVal), nLen); Result := pPtr; end; Function UserHatRecht( cObj, cRecht : string): Boolean; var pO, pR : PAnsiChar; begin // Vorbereitung: pO := HoleSpeicher(cObj); pR := HoleSpeicher(cRecht); // Ausführung: Result := DLL_UserHatRecht(pO, pR); // Aufräumen: FreeMem(pO); // <-- hier kommt die Zugriffsverletzung FreeMem(pR); end; Ich rufe an anderen Stellen bereits andere Funktionen aus der DLL auf, bei der zum Teil sogar Strings zurückgegeben und deutlich mehr Parameter an die DLL übergeben werden, ohne dass etwas passiert. Nur bei dieser Funktion knallt es. :roll: Irgendjemand eine Idee, was hier schiefläuft? Gruß hsg |
AW: Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Hallo,
was mir komisch vorkommt. Du sprichst überall von Ansi, hier aber benutzt du den normalen Delphi-String. cObj := String(cPObj); Und benutzt z.B. nicht deine StrLCopies, um den String zu füllen. cObj liegt ja auf dem Stack und wird "hinter" dem end freigegeben. Was dann mit cPObj ist, mag ich mir gerade nicht ausdenken ... |
AW: Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Zitat:
Mache es doch nicht schwieriger als es ist, sondern einfach so:
Delphi-Quellcode:
Es wäre anders, wenn du die Strings als Rückgabe-Puffer benötigen solltest, aber das ist ja hier nicht der Fall.
function UserHatRecht(cObj, cRecht: string): Boolean;
begin Result := DLL_UserHatRecht(PAnsiChar(AnsiString(cObj)), PAnsiChar(AnsiString(cRecht))); end; Übrigens, du solltest mal meinen Artikel (nur in English, sorry) über DLLs lesen: ![]() |
AW: Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Moin,
erst einmal Danke für die bisherigen Antworten. Zitat:
Die Funktion habe ich als Verzweiflungstat eingeführt, weil meine vorherigen Versuche alles mögliche ergaben, nur nicht das erwünschte. Unter anderem stand in einer Version in allen Parametern das gleiche. Inzwischen ist mir klar, was ich dort falsch gemacht habe, doch da hatte ich bereits die Funktion eingebaut und die meisten Stellen geändert. Danke für den Link, ich werde ihn mir gleich in aller Ruhe zu Gemüte führen. Zitat:
StrLCopy wäre doch hier dann eh falsch, da es mit (P)AnsiChar arbeitet? Gruß hsg |
AW: Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Hallo,
Zitat:
Das ist ja gerade das gefährliche von TypeCasts. Du umgehst die komplette Typ-Prüfung und oft auch die "Compiler-Magie" bei Strings. Im Zweifel kannst du ja den Assembler-Code mit einblenden. Zumindestens Copy-Befehle oder deren nicht Vorhandensein müssten zu sehen sein. |
AW: Delphi-DLL mit PAnsiChar: Zugriffsverletzung bei Freigabe
Ich habe mir jetzt mal angesehen, was an der Stelle passiert. Leider reichen meine Assembler-Kenntnisse nicht mehr aus, um wirklich alles zu verstehen.
Allerdings wird der PAnsiChar über einen festen Zwischenbuffer (Buffer: array[0..2047] of WideChar) und der Microsoft-Funktion MultiByteToWideChar in einen Unicode-String gewandelt, der dann über neu allozierten Speicher zurückgegeben (System.GetMem wird explizit aufgerufen) wird. Von daher ist die Konvertierung wohl an der Stelle gefahrlos. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 01:30 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