VirtualBox

Changeset 27451 in vbox for trunk/src/VBox/Runtime/r3


Ignore:
Timestamp:
Mar 17, 2010 3:06:44 PM (15 years ago)
Author:
vboxsync
Message:

IPRT/process-win: Update on RTProcCreateEx() + testcase.

File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Runtime/r3/win/process-win.cpp

    r27416 r27451  
    169169RTR3DECL(int)   RTProcCreateEx(const char *pszExec, const char * const *papszArgs, RTENV hEnv, uint32_t fFlags,
    170170                               PCRTHANDLE phStdIn, PCRTHANDLE phStdOut, PCRTHANDLE phStdErr, const char *pszAsUser,
    171                                PRTPROCESS phProcess)
     171                               const char *pszPassword, PRTPROCESS phProcess)
    172172{
    173173#if 1 /* needs more work... dinner time. */
     
    183183    AssertPtrReturn(papszArgs, VERR_INVALID_PARAMETER);
    184184    /** @todo search the PATH (add flag for this). */
    185     AssertPtrNullReturn(pszAsUser, VERR_INVALID_POINTER);
    186185
    187186    /*
     
    253252            if (RT_SUCCESS(rc))
    254253            {
    255                 HANDLE hToken = INVALID_HANDLE_VALUE;
    256                 if (pszAsUser)
     254                /*
     255                 * Get going...
     256                 */
     257                PROCESS_INFORMATION ProcInfo;
     258                RT_ZERO(ProcInfo);
     259                BOOL fRc;
     260                if (pszAsUser == NULL)
     261                    fRc = CreateProcessW(pwszExec,
     262                                         pwszCmdLine,
     263                                         NULL,         /* pProcessAttributes */
     264                                         NULL,         /* pThreadAttributes */
     265                                         TRUE,         /* fInheritHandles */
     266                                         CREATE_UNICODE_ENVIRONMENT, /* dwCreationFlags */
     267                                         pwszzBlock,
     268                                         NULL,          /* pCurrentDirectory */
     269                                         &StartupInfo,
     270                                         &ProcInfo);
     271                else
    257272                {
    258                     /** @todo - Maybe use CreateProcessWithLoginW? That'll require a password, but
    259                      *        we may need that anyway because it looks like LogonUserW is the only
    260                      *        way to get a hToken.  FIXME */
    261                     rc = VERR_NOT_IMPLEMENTED;
     273                    HANDLE hToken = INVALID_HANDLE_VALUE;
     274                    PRTUTF16 pwszUser;
     275
     276                    rc = RTStrToUtf16(pszAsUser, &pwszUser);                   
     277                    if (RT_SUCCESS(rc))
     278                    {
     279                        PRTUTF16 pwszPassword;
     280                        rc = RTStrToUtf16(pszPassword, &pwszPassword);   
     281                        if (RT_SUCCESS(rc))
     282                        {
     283                            /*
     284                             * User needs to be specified in UPN format because we
     285                             * don't fill a domain name.
     286                             */
     287
     288                            /*
     289                             * The following rights are needed in order to use
     290                             * LogonUserW and CreateProcessAsUserW:
     291                             *
     292                             * - SE_ASSIGNPRIMARYTOKEN_NAME
     293                             * - SE_INCREASE_QUOTA_NAME
     294                             * - SE_TCB_NAME
     295                             */
     296                            fRc = LogonUserW(pwszUser,
     297                                             NULL,      /* lpDomain */
     298                                             pszPassword ? pwszPassword : L"",
     299                                             LOGON32_LOGON_INTERACTIVE,
     300                                             LOGON32_PROVIDER_DEFAULT,
     301                                             &hToken);
     302                            if (fRc)
     303                            {
     304                                fRc = CreateProcessAsUserW(hToken,
     305                                                           pwszExec,
     306                                                           pwszCmdLine,
     307                                                           NULL,         /* pProcessAttributes */
     308                                                           NULL,         /* pThreadAttributes */
     309                                                           TRUE,         /* fInheritHandles */
     310                                                           CREATE_UNICODE_ENVIRONMENT, /* dwCreationFlags */
     311                                                           pwszzBlock,
     312                                                           NULL,         /* pCurrentDirectory */
     313                                                           &StartupInfo,
     314                                                           &ProcInfo);
     315                                if (!fRc)
     316                                {
     317/* CreateProcessWithLogonW is not available on NT4 ... so enabling the following code
     318* would blow up compatibility but is a legitim fallback if the above method isn't working. */
     319#if 0
     320                                    DWORD dwErr = GetLastError();
     321       
     322                                    /*
     323                                     * If we don't hold enough priviledges to spawn a new
     324                                     * process with different credentials we have to use
     325                                     * CreateProcessWithLogonW here.
     326                                     *
     327                                     * @todo Use fFlags to either use this feature or just fail.
     328                                     */
     329                                    if (ERROR_PRIVILEGE_NOT_HELD == dwErr)
     330                                    {
     331                                        fRc = CreateProcessWithLogonW(pwszUser,
     332                                                                      NULL,                 /* lpDomain*/
     333                                                                      pwszPassword,
     334                                                                      LOGON_WITH_PROFILE,   /* dwLogonFlags */
     335                                                                      pwszExec,
     336                                                                      pwszCmdLine,
     337                                                                      CREATE_UNICODE_ENVIRONMENT, /* dwCreationFlags */
     338                                                                      pwszzBlock,
     339                                                                      NULL,                 /* pCurrentDirectory */
     340                                                                      &StartupInfo,
     341                                                                      &ProcInfo);
     342                                    }
     343#endif
     344                                }
     345                                CloseHandle(hToken);
     346                            }
     347                            RTUtf16Free(pwszPassword);
     348                        }
     349                        RTUtf16Free(pwszUser);
     350                    }
    262351                }
    263                 if (RT_SUCCESS(rc))
     352
     353                if (fRc)
    264354                {
    265                     /*
    266                      * Get going...
    267                      */
    268                     PROCESS_INFORMATION ProcInfo;
    269                     RT_ZERO(ProcInfo);
    270                     BOOL fRc;
    271                     if (hToken == INVALID_HANDLE_VALUE)
    272                         fRc = CreateProcessW(pwszExec,
    273                                              pwszCmdLine,
    274                                              NULL,         /* pProcessAttributes */
    275                                              NULL,         /* pThreadAttributes */
    276                                              TRUE,         /* fInheritHandles */
    277                                              CREATE_UNICODE_ENVIRONMENT, /* dwCreationFlags */
    278                                              pwszzBlock,
    279                                              NULL,          /* pCurrentDirectory */
    280                                              &StartupInfo,
    281                                              &ProcInfo);
    282                     else
    283                         fRc = CreateProcessAsUserW(hToken,
    284                                                    pwszExec,
    285                                                    pwszCmdLine,
    286                                                    NULL,         /* pProcessAttributes */
    287                                                    NULL,         /* pThreadAttributes */
    288                                                    TRUE,         /* fInheritHandles */
    289                                                    CREATE_UNICODE_ENVIRONMENT, /* dwCreationFlags */
    290                                                    pwszzBlock,
    291                                                    NULL,          /* pCurrentDirectory */
    292                                                    &StartupInfo,
    293                                                    &ProcInfo);
    294 
    295                     if (fRc)
     355                    CloseHandle(ProcInfo.hThread);
     356                    if (phProcess)
    296357                    {
    297                         CloseHandle(ProcInfo.hThread);
    298                         if (phProcess)
    299                         {
    300                             /** @todo Remember the process handle and pick it up in RTProcWait. */
    301                             *phProcess = ProcInfo.dwProcessId;
    302                         }
    303                         else
    304                             CloseHandle(ProcInfo.hProcess);
    305                         rc = VINF_SUCCESS;
     358                        /** @todo Remember the process handle and pick it up in RTProcWait. */
     359                        *phProcess = ProcInfo.dwProcessId;
    306360                    }
    307361                    else
    308                         rc = RTErrConvertFromWin32(GetLastError());
    309 
    310                     if (hToken != INVALID_HANDLE_VALUE)
    311                         CloseHandle(hToken);
     362                        CloseHandle(ProcInfo.hProcess);
     363                    rc = VINF_SUCCESS;
    312364                }
     365                else
     366                    rc = RTErrConvertFromWin32(GetLastError());
    313367                RTUtf16Free(pwszExec);
    314368            }
Note: See TracChangeset for help on using the changeset viewer.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette