AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein Win32/Win64 API (native code) Zugriffsverletzung bei Benutzung von Pipes
Thema durchsuchen
Ansicht
Themen-Optionen

Zugriffsverletzung bei Benutzung von Pipes

Ein Thema von Neutral General · begonnen am 3. Feb 2011 · letzter Beitrag vom 3. Feb 2011
Antwort Antwort
Benutzerbild von Neutral General
Neutral General

Registriert seit: 16. Jan 2004
Ort: Bendorf
5.219 Beiträge
 
Delphi 10.2 Tokyo Professional
 
#1

Zugriffsverletzung bei Benutzung von Pipes

  Alt 3. Feb 2011, 09:14
Hallo,

Ich kann grad nicht verstehen, warum mir mein Code Probleme macht...
Ich öffne .bat Dateien und lesen deren Ausgabe aus und Leite Eingaben wieder an die Batch weiter.

Mein Code ist im Prinzip relativ übersichtlich:

Delphi-Quellcode:
procedure TForm1.btStartClick(Sender: TObject);
var si: TStartupInfo;
    pi: TProcessInformation;
    cnt, br: Cardinal;
    Buff: Pointer;
    SA: TSecurityAttributes;
    Dest: AnsiString;
begin
  if od.Execute then
  begin
    edBatch.Text := od.FileName;

    SA.nLength := SizeOf(TSecurityAttributes);
    SA.lpSecurityDescriptor := nil;
    SA.bInheritHandle := true;

    CreatePipe(FPipeOutRead,FPipeOutWrite,@SA,0);
    CreatePipe(FPipeInRead,FPipeInWrite,@SA,0);

    FillChar(si,SizeOf(TStartupInfo),0);
    si.cb := SizeOf(TStartupInfo);
    si.dwFlags := STARTF_USESHOWWINDOW or STARTF_USESTDHANDLES;
    si.wShowWindow := SW_HIDE;
    si.hStdOutput := FPipeOutWrite;
    si.hStdError := FPipeOutWrite;
    si.hStdInput := FPipeInRead;

    CreateProcess(nil,PChar(od.FileName),nil,nil,true,CREATE_NEW_CONSOLE,nil,nil,si,pi);
    try
      repeat
        cnt := 0;
        PeekNamedPipe(FPipeOutRead,nil,0,nil,@cnt,nil);
        if cnt > 0 then
        begin
          GetMem(Buff,cnt);
          try
            FillChar(Buff^,cnt,0);
            ReadFile(FPipeOutRead,Buff^,cnt,br,nil);
            if br > 0 then
            begin
              SetLength(Dest,br);
              try
                OemToCharA(Buff,@Dest[1]);
                Memo1.Lines.Text := Memo1.Lines.Text + String(Dest); // <----
              finally
                SetLength(Dest,0);
              end;
            end;
          finally
            FreeMem(Buff);
          end;
        end;
        Application.ProcessMessages;
      until FClosing;
    finally
      CloseHandle(pi.hThread);
      CloseHandle(pi.hProcess);

      CloseHandle(FPipeOutRead);
      CloseHandle(FPipeOutWrite);
      CloseHandle(FPipeInRead);
      CloseHandle(FPipeInWrite);
    end;
  end;
end;

procedure TForm1.btEingabeClick(Sender: TObject);
var br: Cardinal;
    InStr: AnsiString;
begin
  InStr := AnsiString(edEingabe.Text) + #13#10;
  WriteFile(FPipeInWrite,InStr[1],Length(InStr),br,nil);
  edEingabe.Clear;
end;

procedure TForm1.FormCloseQuery(Sender: TObject; var CanClose: Boolean);
begin
  FClosing := true;
end;
Jetzt kommt es manchmal (Ich schreibe hektisch Daten in die Pipe (btEingabeClick)) dazu, dass an der im Code markierten Stelle eine AV auftritt. Kann aber nicht nachvollziehen warum. Der Debugger will mir auch nicht den Inhalt von "Dest" anzeigen.
Dann kommt es in 99% aller Fälle dazu, dass ich eine Zugriffsverletzung bekomme wenn ich das Programm beende. Das kann ich leider auch nicht verstehen

Kann mir da vllt. jemand helfen?

Edit: Sehe grade meine Pipe-Handles sind nicht optimal per try-finally geschützt, aber das sollte für mein Problem keine Bedeutung haben.

Gruß
Neutral General
Michael
"Programmers talk about software development on weekends, vacations, and over meals not because they lack imagination,
but because their imagination reveals worlds that others cannot see."

Geändert von Neutral General ( 3. Feb 2011 um 09:17 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von p80286
p80286

Registriert seit: 28. Apr 2008
Ort: Stolberg (Rhl)
6.659 Beiträge
 
FreePascal / Lazarus
 
#2

AW: Zugriffsverletzung bei Benutzung von Pipes

  Alt 3. Feb 2011, 09:36
Warum machst Du es nicht so:
Memo1.Lines.Add(Dest); und vertraust auf eine implizite Umwandlung?

Gruß
K-H
Programme gehorchen nicht Deinen Absichten sondern Deinen Anweisungen
R.E.D retired error detector
  Mit Zitat antworten Zitat
Dezipaitor

Registriert seit: 14. Apr 2003
Ort: Stuttgart
1.701 Beiträge
 
Delphi 7 Professional
 
#3

AW: Zugriffsverletzung bei Benutzung von Pipes

  Alt 3. Feb 2011, 09:54
Schau mal hier:
CreateProcess(nil,PChar(od.FileName),nil,nil,true,CREATE_NEW_CONSOLE,nil,nil,si,pi); Du verwendest sicher Delphi >= 2009? Dann ist CreateProcess = CreateProcessW.

Zitat von http://msdn.microsoft.com/en-us/library/ms682425%28VS.85%29.aspx:
The Unicode version of this function, CreateProcessW, can modify the contents of this string. Therefore, this parameter cannot be a pointer to read-only memory (such as a const variable or a literal string). If this parameter is a constant string, the function may cause an access violation.
Du brauchst also ne extra Stringvariable dafür.
Christian
Windows, Tokens, Access Control List, Dateisicherheit, Desktop, Vista Elevation?
Goto: JEDI API LIB & Windows Security Code Library (JWSCL)
  Mit Zitat antworten Zitat
Benutzerbild von Neutral General
Neutral General

Registriert seit: 16. Jan 2004
Ort: Bendorf
5.219 Beiträge
 
Delphi 10.2 Tokyo Professional
 
#4

AW: Zugriffsverletzung bei Benutzung von Pipes

  Alt 3. Feb 2011, 09:57
Warum machst Du es nicht so:
Memo1.Lines.Add(Dest); und vertraust auf eine implizite Umwandlung?

Gruß
K-H
Weil mir Lines.Add einen Zeilenumbruch einfügt, den ich nicht will.

Schau mal hier:
CreateProcess(nil,PChar(od.FileName),nil,nil,true,CREATE_NEW_CONSOLE,nil,nil,si,pi); Du verwendest sicher Delphi >= 2009? Dann ist CreateProcess = CreateProcessW.

Zitat von http://msdn.microsoft.com/en-us/library/ms682425%28VS.85%29.aspx:
The Unicode version of this function, CreateProcessW, can modify the contents of this string. Therefore, this parameter cannot be a pointer to read-only memory (such as a const variable or a literal string). If this parameter is a constant string, the function may cause an access violation.
Du brauchst also ne extra Stringvariable dafür.
Das kann ich machen. Allerdings kracht es ja nicht bei CreateProcess sondern irgendwann während der Prozess schon läuft und ich die Pipes auslese und reinschreibe.
Michael
"Programmers talk about software development on weekends, vacations, and over meals not because they lack imagination,
but because their imagination reveals worlds that others cannot see."
  Mit Zitat antworten Zitat
Benutzerbild von p80286
p80286

Registriert seit: 28. Apr 2008
Ort: Stolberg (Rhl)
6.659 Beiträge
 
FreePascal / Lazarus
 
#5

AW: Zugriffsverletzung bei Benutzung von Pipes

  Alt 3. Feb 2011, 10:42
Na dann nimm halt OemToCharW und verdoppele die Länge, bzw definiere Dest gleich als String .

Gruß
K-H

Das man denInhalt von Dest nicht sehen kann stört mich, hast Du mal versucht Dest vorher zu initialisieren?
Programme gehorchen nicht Deinen Absichten sondern Deinen Anweisungen
R.E.D retired error detector
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:36 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz