⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 proc.c

📁 linux subdivision ying gai ke yi le ba
💻 C
📖 第 1 页 / 共 2 页
字号:
        }    }     else #endif    {        /* Win32 is _different_ than unix.  While unix will find the given         * program since it's already chdir'ed, Win32 cannot since the parent         * attempts to open the program with it's own path.         * ###: This solution isn't much better - it may defeat path searching         * when the path search was desired.  Open to further discussion.         */        i = strlen(progname);        if (i >= 4 && (strcasecmp(progname + i - 4, ".bat") == 0                    || strcasecmp(progname + i - 4, ".cmd") == 0))        {            char *shellcmd = getenv("COMSPEC");            if (!shellcmd) {                if (attr->errfn) {                    attr->errfn(pool, APR_EINVAL, "COMSPEC envar is not set");                }                return APR_EINVAL;            }            if (shellcmd[0] == '"') {                progname = apr_pstrndup(pool, shellcmd + 1, strlen(shellcmd) - 2);            }            else {                progname = shellcmd;                if (has_space(shellcmd)) {                    shellcmd = apr_pstrcat(pool, "\"", shellcmd, "\"", NULL);                }            }            i = strlen(progname);            if (i >= 11 && strcasecmp(progname + i - 11, "command.com") == 0) {                /* XXX: Still insecure - need doubled-quotes on each individual                 * arg of cmdline.  Suspect we need to postpone cmdline parsing                 * until this moment in all four code paths, with some flags                 * to toggle 'which flavor' is needed.                 */                cmdline = apr_pstrcat(pool, shellcmd, " /C ", argv0, cmdline, NULL);            }            else {                /* We must protect the cmdline args from any interpolation - this                 * is not a shellcmd, and the source of argv[] is untrusted.                 * Notice we escape ALL the cmdline args, including the quotes                 * around the individual args themselves.  No sense in allowing                 * the shift-state to be toggled, and the application will                  * not see the caret escapes.                 */                cmdline = apr_caret_escape_args(pool, cmdline);                /*                 * Our app name must always be quoted so the quotes surrounding                 * the entire /c "command args" are unambigious.                 */                if (*argv0 != '"') {                    cmdline = apr_pstrcat(pool, shellcmd, " /C \"\"", argv0, "\"", cmdline, "\"", NULL);                }                else {                    cmdline = apr_pstrcat(pool, shellcmd, " /C \"", argv0, cmdline, "\"", NULL);                }            }        }        else {            /* A simple command we are directly invoking.  Do not pass             * the first arg to CreateProc() for APR_PROGRAM_PATH             * invocation, since it would need to be a literal and             * complete file path.  That is; "c:\bin\aprtest.exe"             * would succeed, but "c:\bin\aprtest" or "aprtest.exe"             * can fail.             */            cmdline = apr_pstrcat(pool, argv0, cmdline, NULL);            if (attr->cmdtype == APR_PROGRAM_PATH) {                progname = NULL;            }        }    }    if (!env || attr->cmdtype == APR_PROGRAM_ENV ||        attr->cmdtype == APR_SHELLCMD_ENV) {        pEnvBlock = NULL;    }    else {        apr_size_t iEnvBlockLen;        /*         * Win32's CreateProcess call requires that the environment         * be passed in an environment block, a null terminated block of         * null terminated strings.         */          i = 0;        iEnvBlockLen = 1;        while (env[i]) {            iEnvBlockLen += strlen(env[i]) + 1;            i++;        }        if (!i)             ++iEnvBlockLen;#if APR_HAS_UNICODE_FS        IF_WIN_OS_IS_UNICODE        {            apr_wchar_t *pNext;            pEnvBlock = (char *)apr_palloc(pool, iEnvBlockLen * 2);            dwCreationFlags |= CREATE_UNICODE_ENVIRONMENT;            i = 0;            pNext = (apr_wchar_t*)pEnvBlock;            while (env[i]) {                apr_size_t in = strlen(env[i]) + 1;                if ((rv = apr_conv_utf8_to_ucs2(env[i], &in,                                                 pNext, &iEnvBlockLen))                         != APR_SUCCESS) {                    if (attr->errfn) {                        attr->errfn(pool, rv,                                     apr_pstrcat(pool,                                                 "utf8 to ucs2 conversion failed"                                                 " on this string: ", env[i], NULL));                    }                    return rv;                }                pNext = wcschr(pNext, L'\0') + 1;                i++;            }	    if (!i)                *(pNext++) = L'\0';	    *pNext = L'\0';        }#endif /* APR_HAS_UNICODE_FS */#if APR_HAS_ANSI_FS        ELSE_WIN_OS_IS_ANSI        {            char *pNext;            pEnvBlock = (char *)apr_palloc(pool, iEnvBlockLen);                i = 0;            pNext = pEnvBlock;            while (env[i]) {                strcpy(pNext, env[i]);                pNext = strchr(pNext, '\0') + 1;                i++;            }	    if (!i)                *(pNext++) = '\0';	    *pNext = '\0';        }#endif /* APR_HAS_ANSI_FS */    }     new->invoked = cmdline;#if APR_HAS_UNICODE_FS    IF_WIN_OS_IS_UNICODE    {        STARTUPINFOW si;        apr_wchar_t *wprg = NULL;        apr_wchar_t *wcmd = NULL;        apr_wchar_t *wcwd = NULL;        if (progname) {            apr_size_t nprg = strlen(progname) + 1;            apr_size_t nwprg = nprg + 6;            wprg = apr_palloc(pool, nwprg * sizeof(wprg[0]));            if ((rv = apr_conv_utf8_to_ucs2(progname, &nprg, wprg, &nwprg))                   != APR_SUCCESS) {                if (attr->errfn) {                    attr->errfn(pool, rv,                                 apr_pstrcat(pool,                                             "utf8 to ucs2 conversion failed"                                             " on progname: ", progname, NULL));                }                return rv;            }        }        if (cmdline) {            apr_size_t ncmd = strlen(cmdline) + 1;            apr_size_t nwcmd = ncmd;            wcmd = apr_palloc(pool, nwcmd * sizeof(wcmd[0]));            if ((rv = apr_conv_utf8_to_ucs2(cmdline, &ncmd, wcmd, &nwcmd))                    != APR_SUCCESS) {                if (attr->errfn) {                    attr->errfn(pool, rv,                                 apr_pstrcat(pool,                                             "utf8 to ucs2 conversion failed"                                             " on cmdline: ", cmdline, NULL));                }                return rv;            }        }        if (attr->currdir)        {            apr_size_t ncwd = strlen(attr->currdir) + 1;            apr_size_t nwcwd = ncwd;            wcwd = apr_palloc(pool, ncwd * sizeof(wcwd[0]));            if ((rv = apr_conv_utf8_to_ucs2(attr->currdir, &ncwd,                                             wcwd, &nwcwd))                    != APR_SUCCESS) {                if (attr->errfn) {                    attr->errfn(pool, rv,                                 apr_pstrcat(pool,                                             "utf8 to ucs2 conversion failed"                                             " on currdir: ", attr->currdir, NULL));                }                return rv;            }        }        memset(&si, 0, sizeof(si));        si.cb = sizeof(si);        if (attr->detached) {            si.dwFlags |= STARTF_USESHOWWINDOW;            si.wShowWindow = SW_HIDE;        }#ifndef _WIN32_WCE        if ((attr->child_in && attr->child_in->filehand)            || (attr->child_out && attr->child_out->filehand)            || (attr->child_err && attr->child_err->filehand))        {            si.dwFlags |= STARTF_USESTDHANDLES;            si.hStdInput = (attr->child_in)                               ? attr->child_in->filehand                              : INVALID_HANDLE_VALUE;            si.hStdOutput = (attr->child_out)                              ? attr->child_out->filehand                              : INVALID_HANDLE_VALUE;            si.hStdError = (attr->child_err)                              ? attr->child_err->filehand                              : INVALID_HANDLE_VALUE;        }        rv = CreateProcessW(wprg, wcmd,        /* Executable & Command line */                            NULL, NULL,        /* Proc & thread security attributes */                            TRUE,              /* Inherit handles */                            dwCreationFlags,   /* Creation flags */                            pEnvBlock,         /* Environment block */                            wcwd,              /* Current directory name */                            &si, &pi);#else        rv = CreateProcessW(wprg, wcmd,        /* Executable & Command line */                            NULL, NULL,        /* Proc & thread security attributes */                            FALSE,             /* must be 0 */                            dwCreationFlags,   /* Creation flags */                            NULL,              /* Environment block must be NULL */                            NULL,              /* Current directory name must be NULL*/                            NULL,              /* STARTUPINFO not supported */                            &pi);#endif    }#endif /* APR_HAS_UNICODE_FS */#if APR_HAS_ANSI_FS    ELSE_WIN_OS_IS_ANSI    {        STARTUPINFOA si;        memset(&si, 0, sizeof(si));        si.cb = sizeof(si);        if (attr->detached) {            si.dwFlags |= STARTF_USESHOWWINDOW;            si.wShowWindow = SW_HIDE;        }        if ((attr->child_in && attr->child_in->filehand)            || (attr->child_out && attr->child_out->filehand)            || (attr->child_err && attr->child_err->filehand))        {            si.dwFlags |= STARTF_USESTDHANDLES;            si.hStdInput = (attr->child_in)                               ? attr->child_in->filehand                              : INVALID_HANDLE_VALUE;            si.hStdOutput = (attr->child_out)                              ? attr->child_out->filehand                              : INVALID_HANDLE_VALUE;            si.hStdError = (attr->child_err)                              ? attr->child_err->filehand                              : INVALID_HANDLE_VALUE;        }        rv = CreateProcessA(progname, cmdline, /* Command line */                            NULL, NULL,        /* Proc & thread security attributes */                            TRUE,              /* Inherit handles */                            dwCreationFlags,   /* Creation flags */                            pEnvBlock,         /* Environment block */                            attr->currdir,     /* Current directory name */                            &si, &pi);    }#endif /* APR_HAS_ANSI_FS */    /* Check CreateProcess result      */    if (!rv)        return apr_get_os_error();    /* XXX Orphaned handle warning - no fix due to broken apr_proc_t api.     */    new->hproc = pi.hProcess;    new->pid = pi.dwProcessId;    if (attr->child_in) {        apr_file_close(attr->child_in);    }    if (attr->child_out) {        apr_file_close(attr->child_out);    }    if (attr->child_err) {        apr_file_close(attr->child_err);    }    CloseHandle(pi.hThread);    return APR_SUCCESS;}APR_DECLARE(apr_status_t) apr_proc_wait_all_procs(apr_proc_t *proc,                                                  int *exitcode,                                                  apr_exit_why_e *exitwhy,                                                  apr_wait_how_e waithow,                                                  apr_pool_t *p){    /* Unix does apr_proc_wait(proc(-1), exitcode, exitwhy, waithow)     * but Win32's apr_proc_wait won't work that way.  We can either     * register all APR created processes in some sort of AsyncWait     * thread, or simply walk from the global process pool for all      * apr_pool_note_subprocess()es registered with APR.     */    return APR_ENOTIMPL;}static apr_exit_why_e why_from_exit_code(DWORD exit) {    /* See WinNT.h STATUS_ACCESS_VIOLATION and family for how     * this class of failures was determined     */    if (((exit & 0xC0000000) == 0xC0000000)                     && !(exit & 0x3FFF0000))        return APR_PROC_SIGNAL;    else        return APR_PROC_EXIT;    /* ### No way to tell if Dr Watson grabbed a core, AFAICT. */}APR_DECLARE(apr_status_t) apr_proc_wait(apr_proc_t *proc,                                        int *exitcode, apr_exit_why_e *exitwhy,                                        apr_wait_how_e waithow){    DWORD stat;    DWORD time;    if (waithow == APR_WAIT)        time = INFINITE;    else        time = 0;    if ((stat = WaitForSingleObject(proc->hproc, time)) == WAIT_OBJECT_0) {        if (GetExitCodeProcess(proc->hproc, &stat)) {            if (exitcode)                *exitcode = stat;            if (exitwhy)                *exitwhy = why_from_exit_code(stat);            CloseHandle(proc->hproc);            proc->hproc = NULL;            return APR_CHILD_DONE;        }    }    else if (stat == WAIT_TIMEOUT) {        return APR_CHILD_NOTDONE;    }    return apr_get_os_error();}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -