7

「コンソール アプリケーション」ではなく「Windows アプリケーション」としてコンパイルされた非常に単純な C# または C++ GUI プログラムから実行されているときに、PsExec のハングが発生しています。以下のセクション C) にコードを貼り付けて問題を再現し、セクション D) に C++ コードを貼り付けて同じ問題を再現しました。

psexec がハングアップすると、ローカルで psexec にアタッチされた後の windbg の出力がセクション B) の下に貼り付けられます。

セクション A) に貼り付けた出力をダンプした後、プログラムがハングします。

psexec コマンドをローカルのものに置き換えると、プログラムは正常に動作します。

誰かがそれを経験し、それに対する解決策を見つけたのではないかと思っていました. 助けていただければ幸いです。

ありがとう、シャラジェッシュ


A) psexec がハングしたときの C# プログラムの出力

PsExec v1.98 - プロセスをリモートで実行 Copyright (C) 2001-2010 Mark Russinovich Sysinternals - www.sysinternals.com

ドライブ C のボリュームにはラベルがありません。


B) ハング中の psexec の Windbg 出力

3 ID:1614.15E4サスペンド:1 TEB:7EFAC000 CHILDEBP RETADDR ARGS TO CHILD 02A3FE68
75A6D0C5 00000180 00000000000000 NTDLL! : [SEH]) 02a3ff14 00403bde 00000180 02a3ff44 00010000 kernel32!ReadFileImplementation+0xf0 (FPO: [SEH]) 警告: スタック アンワインド情報がありません。次のフレームは間違っている可能性があります。02a3ff2c 00000000 00291e48 00000000 02a5ff80 psexec+0x3bde


C) 問題を再現するための C# コード

using System;
using System.Windows.Forms;
using System.Diagnostics;

namespace WindowsFormsApplication1 {
  static class Program {
    static void DataReceiveHandler(object sender, DataReceivedEventArgs e) {
      Debug.WriteLine(e.Data);
    }

    public static void NotWorkingPsExec() {
      ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe",
        "\\\\raj-2k3-32 cmd.exe /c dir c:\\windows\\*.*");
      startInfo.UseShellExecute        = false;
      startInfo.CreateNoWindow         = true;
      startInfo.RedirectStandardOutput = true;
      startInfo.RedirectStandardError  = true;

      Process proc = new Process();
      proc.StartInfo           = startInfo;
      proc.ErrorDataReceived  += new DataReceivedEventHandler(DataReceiveHandler);
      proc.OutputDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
      proc.Start();
      proc.BeginErrorReadLine();
      proc.BeginOutputReadLine();
      proc.WaitForExit();
      Debug.WriteLine("Exit code = {0}", proc.ExitCode);
    }

    public static void WorkingPsExec() {
      ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe", 
        "\\\\raj-2k3-32 cmd.exe /c dir c:\\windows\\*.*");
      startInfo.UseShellExecute = false;

      Process proc = new Process();
      proc.StartInfo = startInfo;
      proc.Start();
      proc.WaitForExit();
      Debug.WriteLine("Exit code = {0}", proc.ExitCode);
    }

    static void Main() {
      NotWorkingPsExec();
      //WorkingPsExec(); //If uncommented will work 
    }
  }
}

D) 問題を再現するための C++ コード

#include <windows.h>
#include <tchar.h>
#include <stdio.h>
#include <strsafe.h>

HANDLE g_hStdoutRd = NULL;
HANDLE g_hStdoutWr = NULL;

void StartCommand(TCHAR *szCmdline);
void ReadOutput();
void ErrorExit(PTSTR);

int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) {
  SECURITY_ATTRIBUTES saAttr;
  saAttr.nLength              = sizeof(SECURITY_ATTRIBUTES);
  saAttr.bInheritHandle       = TRUE;
  saAttr.lpSecurityDescriptor = NULL;
  if (!CreatePipe(&g_hStdoutRd, &g_hStdoutWr, &saAttr, 0))
    ErrorExit(TEXT("Stdout SetHandleInformation"));
  if (!SetHandleInformation(g_hStdoutRd, HANDLE_FLAG_INHERIT, 0))
    ErrorExit(TEXT("Stdout SetHandleInformation"));
  TCHAR szCmdline[] = TEXT("psexec.exe \\\\raj-2k3-32 cmd.exe /c dir /s c:\\windows\\*.*"); // Not Working
  //TCHAR szCmdline[] = TEXT("cmd.exe /c dir /s c:\\windows\\*.*"); // Working
  StartCommand(szCmdline);
  ReadOutput();
  return 0;
}

void StartCommand(TCHAR *szCmdline) {
  PROCESS_INFORMATION piProcInfo  = {0};
  STARTUPINFO         siStartInfo = {0};
  siStartInfo.cb         = sizeof(STARTUPINFO);
  siStartInfo.hStdError  = g_hStdoutWr;
  siStartInfo.hStdOutput = g_hStdoutWr;
  siStartInfo.dwFlags   |= STARTF_USESTDHANDLES;
  BOOL bSuccess = CreateProcess(NULL, szCmdline, NULL, NULL, TRUE, 0, NULL, NULL, &siStartInfo, &piProcInfo);
  if (!bSuccess)
    ErrorExit(TEXT("CreateProcess"));
  else {
    CloseHandle(piProcInfo.hProcess);
    CloseHandle(piProcInfo.hThread);
  }
}

void ReadOutput() {
  if (!CloseHandle(g_hStdoutWr))
    ErrorExit(TEXT("StdOutWr CloseHandle"));
  for (;; ) {
    CHAR    chBuf[4096] = {0};
    DWORD   dwRead;
    BOOLEAN bSuccess    = ReadFile(g_hStdoutRd, chBuf, ARRAYSIZE(chBuf), &dwRead, NULL);
    if (!bSuccess || dwRead == 0)
      break;
    OutputDebugStringA(chBuf);
  }
}

void ErrorExit(PTSTR lpszFunction) {
  OutputDebugString(lpszFunction);
  ExitProcess(1);
}
4

6 に答える 6

7

PSExec がランダムにハングしていました。私はあなたの方法で問題を再現する努力はしていませんが、一見価値のある後継者である「PAExec」を使用することで問題を回避しました: http://www.poweradmin.com/PAExec/

于 2012-10-16T14:27:55.627 に答える
2

あなたのものかもしれないeulaが原因である同様の問題がありました:

考えられる理由:

1) psiexec.exe は、最初の実行時に EULA メッセージを表示します。

2) 権限

3) dll 関数は、ユーザー セッションを必要とする場合があります。

これらの問題を回避するには、次のシナリオを試してください。

1) 「-accepteula」引数付き

2) "-s" 引数付き

3) "-i" 引数付き

4) > 2 + 3 5) 2 + 3 + 1

参照:

0

Sysinternalsユーティリティはファイルの実行時アンパックを使用し、カーネルはアンパックされた(実際の)実行可能ファイルを見つけることができないため、startInfoでWorkingDirectoryプロパティも設定してください。

于 2011-04-13T19:19:28.533 に答える
0

これには簡単な解決策があります。

  1. C# は次のようにプロセスを実行します。

    Process.Start("start run.bat xx.txt");    //call it async
    
    //and then we make some code juse wait xx.txt appear and finish written .
    
  2. run.batは :

    psexec.bat > %1           //redirect to a text file
    exit
    
  3. psexec.batは:

    psexec.exe ..........................
     exit
    
于 2013-01-14T04:34:30.760 に答える
-1

ストリームでの同期読み取りは機能します。

        ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe", @"\\localhost cmd.exe /c dir c:\windows\*.*");
        startInfo.UseShellExecute = false;
        startInfo.CreateNoWindow = true;
        startInfo.RedirectStandardOutput = true;
        //startInfo.RedirectStandardError = true;
        //startInfo.RedirectStandardInput = true;

        Process proc = new Process();
        proc.StartInfo = startInfo;
        //proc.ErrorDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
        //proc.OutputDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
        proc.Start();
        //proc.BeginErrorReadLine();
        //proc.BeginOutputReadLine();
        string output = proc.StandardOutput.ReadToEnd();
        proc.WaitForExit();
        Console.WriteLine(output);
        Console.WriteLine("Exit code = {0}", proc.ExitCode);

ここでも、ReadToEnd()はWaitForExit()の前に実行する必要があることに注意してください。

PSExecにはいつもこのような問題があったと思います。Javaサービスで実行している場合、以前は出力をnulにリダイレクトし、実行中のプロセスの出力を取得できませんでしたが、PSExec自体の出力を取得できました。

以下の説明を参照してください。

http://forum.sysinternals.com/psexec-always-hangs-when-run-from-java_topic5013.html

http://forum.sysinternals.com/unusual-problem-with-psexecexe_topic6655.html

編集:

PSEXESVCクリーンアップに関する注意:ハングしたPSEXESVCプロセスを強制終了した後、C:\ Windows(またはC:\ Windows \ system32、あるいはその両方)にあるPSEXESVC.EXEファイルを削除してください。プロセス/ファイルが長引くと、さらに問題が発生します。

于 2011-04-13T22:27:35.703 に答える