2012-01-09 62 views
1

我想開始從Windows服務(本地系統)使用createProcessasUser桂托盤應用程序 - 像這樣:CreateProcessasuser - AccessViolationError

public static System.Diagnostics.Process StartProcessInSession(int sessionID, String commandLine) 
    { 
     IntPtr userToken; 
     if (WTSQueryUserToken(sessionID, out userToken)) 
     { 
      //note that WTSQueryUserToken only works when in context of local system account with SE_TCB_NAME 
      IntPtr lpEnvironment; 
      if (CreateEnvironmentBlock(out lpEnvironment, userToken, false)) 
      { 
       StartupInfo si = new StartupInfo(); 
       si.cb = Marshal.SizeOf(si); 
       si.lpDesktop = "winsta0\\default"; 
       si.dwFlags = STARTF.STARTF_USESHOWWINDOW; 
       si.wShowWindow = ShowWindow.SW_SHOW; 
       ProcessInformation pi; 
       if (CreateProcessAsUser(userToken, null, new StringBuilder(commandLine), IntPtr.Zero, IntPtr.Zero, false, CreationFlags.CREATE_NEW_CONSOLE | CreationFlags.CREATE_UNICODE_ENVIRONMENT, lpEnvironment, null, ref si, out pi)) 
       { 
        CloseHandle(pi.hThread); 
        CloseHandle(pi.hProcess); 
        //context.Undo(); 
        try 
        { 
         return System.Diagnostics.Process.GetProcessById(pi.dwProcessId); 
        } 
        catch (ArgumentException e) 
        { 
         //The process ID couldn't be found - which is what always happens because it has closed 
         return null; 
        } 
       } 
       else 
       { 
        int err = Marshal.GetLastWin32Error(); 
        throw new System.ComponentModel.Win32Exception(err, "Could not create process.\nWin32 error: " + err.ToString()); 
       } 
      } 
      else 
      { 
       int err = Marshal.GetLastWin32Error(); 
       throw new System.ComponentModel.Win32Exception(err, "Could not create environment block.\nWin32 error: " + err.ToString()); 
      } 
     } 
     else 
     { 
      int err = System.Runtime.InteropServices.Marshal.GetLastWin32Error(); 
      if (err == 1008) return null; //There is no token 
      throw new System.ComponentModel.Win32Exception(err, "Could not get the user token from session " + sessionID.ToString() + " - Error: " + err.ToString()); 
     } 
    } 

我使用的功能,以便:

protected override void OnStart(string[] args) 
    { 
     _agentProcess = StartProcessInSession(WTSGetActiveConsoleSessionId(), "Some_correct_path"); 
    } 

這實際上工作了一小會兒,但在我運行的一個突然停止工作......給執行CreateProccessAsUser命令時出現以下錯誤(不能去任何更深)

{"Attempted to read or write protected memory. This is often an indication that other memory is corrupt."} 

我不知道爲什麼這種情況正在發生,甚至如何繼續調試,總之有什麼想法?因爲這對我沒有任何意義。

CreateProccessasuser定義:

[DllImport("advapi32.dll", SetLastError = true, CharSet = CharSet.Auto)] 
     static extern bool CreateProcessAsUser(IntPtr hToken, String lpApplicationName, [In] StringBuilder lpCommandLine, IntPtr /*to a SecurityAttributes struct or null*/ lpProcessAttributes, IntPtr /*to a SecurityAttributes struct or null*/ lpThreadAttributes, bool bInheritHandles, CreationFlags creationFlags, IntPtr lpEnvironment, String lpCurrentDirectory, ref StartupInfo lpStartupInfo, out ProcessInformation lpProcessInformation); 

感謝

+0

'lpEnvironment'被設置爲一個有效的指針嗎? 'userToken'是一個有效的'HANDLE'? – 2012-01-09 18:46:03

+0

你是怎麼定義'CreateProcessAsUser'的? – 2012-01-09 18:54:11

+0

已添加定義。 – Menyh 2012-01-10 06:51:38

回答

1

是您ProcessInformation類型的值類型(結構)或引用類型(類)?

顯示它的定義和P/Invoke的聲明CreateProcessAsUser

BTW,所有GetLastWin32Error檢查是你用p做/如果你使用正確的屬性調用。

+0

[StructLayout(LayoutKind.Sequential)] 內部結構PROCESSINFORMATION { \t \t#地區數據成員(4) 公衆詮釋dwProcessId; public int dwThreadId; public IntPtr hProcess; public IntPtr hThread; \t \t #endregion數據成員 } – Menyh 2012-01-10 06:49:21

相關問題