office-gobmx/sal/osl/w32/process.cxx
Stephan Bergmann 10ce581cfc osl_getCommandArgCount can be called w/o prior osl_setCommandArgs.
...e.g., with "import uno" in a stand-alone pyhton process (where it
is hopefully harmless that UNO does not get access to any -env
command line args).
2011-10-20 23:04:11 +02:00

662 lines
20 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*************************************************************************
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright 2000, 2010 Oracle and/or its affiliates.
*
* OpenOffice.org - a multi-platform office productivity suite
*
* This file is part of OpenOffice.org.
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
*
* OpenOffice.org is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
* <http://www.openoffice.org/license.html>
* for a copy of the LGPLv3 License.
*
************************************************************************/
#define UNICODE
#include "system.h"
#include <string.h>
#ifdef _MSC_VER
#pragma warning(push,1) /* disable warnings within system headers */
#endif
#include <shellapi.h>
#ifdef _MSC_VER
#pragma warning(pop)
#endif
#include <osl/diagnose.h>
#include <osl/security.h>
#include <osl/nlsupport.h>
#include <osl/mutex.h>
#include <osl/thread.h>
#include "procimpl.h"
#include "sockimpl.h"
#include "file_url.h"
#include "path_helper.hxx"
#include <rtl/ustrbuf.h>
#include <rtl/alloc.h>
/***************************************************************************
* Process.
***************************************************************************/
oslProcessError SAL_CALL osl_terminateProcess(oslProcess Process)
{
if (Process == NULL)
return osl_Process_E_Unknown;
if (TerminateProcess(((oslProcessImpl*)Process)->m_hProcess, 0))
return osl_Process_E_None;
return osl_Process_E_Unknown;
}
/***************************************************************************/
oslProcess SAL_CALL osl_getProcess(oslProcessIdentifier Ident)
{
oslProcessImpl* pProcImpl;
HANDLE hProcess = OpenProcess(
STANDARD_RIGHTS_REQUIRED | PROCESS_QUERY_INFORMATION | SYNCHRONIZE, FALSE, (DWORD)Ident);
if (hProcess)
{
pProcImpl = reinterpret_cast< oslProcessImpl*>( rtl_allocateMemory(sizeof(oslProcessImpl)) );
pProcImpl->m_hProcess = hProcess;
pProcImpl->m_IdProcess = Ident;
}
else
pProcImpl = NULL;
return (pProcImpl);
}
/***************************************************************************/
void SAL_CALL osl_freeProcessHandle(oslProcess Process)
{
if (Process != NULL)
{
CloseHandle(((oslProcessImpl*)Process)->m_hProcess);
rtl_freeMemory((oslProcessImpl*)Process);
}
}
/***************************************************************************/
oslProcessError SAL_CALL osl_getProcessInfo(oslProcess Process, oslProcessData Fields,
oslProcessInfo* pInfo)
{
HANDLE hProcess;
DWORD IdProcess;
if (Process == NULL)
{
hProcess = GetCurrentProcess();
IdProcess = GetCurrentProcessId();
}
else
{
hProcess = ((oslProcessImpl*)Process)->m_hProcess;
IdProcess = ((oslProcessImpl*)Process)->m_IdProcess;
}
if (! pInfo || (pInfo->Size != sizeof(oslProcessInfo)))
return osl_Process_E_Unknown;
pInfo->Fields = 0;
if (Fields & osl_Process_IDENTIFIER)
{
pInfo->Ident = IdProcess;
pInfo->Fields |= osl_Process_IDENTIFIER;
}
if (Fields & osl_Process_EXITCODE)
{
if (GetExitCodeProcess(hProcess, &(pInfo->Code)) && (pInfo->Code != STILL_ACTIVE))
pInfo->Fields |= osl_Process_EXITCODE;
}
if (Fields & osl_Process_HEAPUSAGE)
{
void* lpAddress=0;
MEMORY_BASIC_INFORMATION Info;
pInfo->HeapUsage = 0;
do
{
if (VirtualQueryEx(hProcess, lpAddress, &Info, sizeof(Info)) == 0)
break;
if ((Info.State == MEM_COMMIT) && (Info.Type == MEM_PRIVATE))
pInfo->HeapUsage += Info.RegionSize;
lpAddress = (LPBYTE)lpAddress + Info.RegionSize;
}
while (lpAddress < (void *)0x80000000); // 2GB address space
pInfo->Fields |= osl_Process_HEAPUSAGE;
}
if (Fields & osl_Process_CPUTIMES)
{
FILETIME CreationTime, ExitTime, KernelTime, UserTime;
if (GetProcessTimes(hProcess, &CreationTime, &ExitTime,
&KernelTime, &UserTime))
{
__int64 Value;
Value = *((__int64 *)&UserTime);
pInfo->UserTime.Seconds = (unsigned long) (Value / 10000000L);
pInfo->UserTime.Nanosec = (unsigned long)((Value % 10000000L) * 100);
Value = *((__int64 *)&KernelTime);
pInfo->SystemTime.Seconds = (unsigned long) (Value / 10000000L);
pInfo->SystemTime.Nanosec = (unsigned long)((Value % 10000000L) * 100);
pInfo->Fields |= osl_Process_CPUTIMES;
}
}
return (pInfo->Fields == Fields) ? osl_Process_E_None : osl_Process_E_Unknown;
}
/***************************************************************************/
oslProcessError SAL_CALL osl_joinProcess(oslProcess Process)
{
return osl_joinProcessWithTimeout(Process, NULL);
}
/***************************************************************************/
oslProcessError SAL_CALL osl_joinProcessWithTimeout(oslProcess Process, const TimeValue* pTimeout)
{
DWORD timeout = INFINITE;
oslProcessError osl_error = osl_Process_E_None;
DWORD ret;
if (NULL == Process)
return osl_Process_E_Unknown;
if (pTimeout)
timeout = pTimeout->Seconds * 1000 + pTimeout->Nanosec / 1000000L;
ret = WaitForSingleObject(((oslProcessImpl*)Process)->m_hProcess, timeout);
if (WAIT_FAILED == ret)
osl_error = osl_Process_E_Unknown;
else if (WAIT_TIMEOUT == ret)
osl_error = osl_Process_E_TimedOut;
return osl_error;
}
/***************************************************************************
* osl_bootstrap_getExecutableFile_Impl().
*
* @internal
* @see rtl_bootstrap
* @see #i37371#
*
***************************************************************************/
extern "C" oslProcessError SAL_CALL osl_bootstrap_getExecutableFile_Impl (
rtl_uString ** ppFileURL
) SAL_THROW_EXTERN_C()
{
oslProcessError result = osl_Process_E_NotFound;
::osl::LongPathBuffer< sal_Unicode > aBuffer( MAX_LONG_PATH );
DWORD buflen = 0;
if ((buflen = GetModuleFileNameW (0, ::osl::mingw_reinterpret_cast<LPWSTR>(aBuffer), aBuffer.getBufSizeInSymbols())) > 0)
{
rtl_uString * pAbsPath = 0;
rtl_uString_newFromStr_WithLength (&(pAbsPath), aBuffer, buflen);
if (pAbsPath)
{
/* Convert from path to url. */
if (osl_getFileURLFromSystemPath (pAbsPath, ppFileURL) == osl_File_E_None)
{
/* Success. */
result = osl_Process_E_None;
}
rtl_uString_release (pAbsPath);
}
}
return (result);
}
/***************************************************************************
* Command Line Arguments.
***************************************************************************/
struct CommandArgs_Impl
{
sal_uInt32 m_nCount;
rtl_uString ** m_ppArgs;
};
static struct CommandArgs_Impl g_command_args =
{
0,
0
};
#ifdef _MSC_VER
#pragma warning( push )
#pragma warning( disable: 4100 )
#endif
static rtl_uString ** osl_createCommandArgs_Impl (int argc, char ** argv)
{
rtl_uString ** ppArgs =
(rtl_uString**)rtl_allocateZeroMemory (argc * sizeof(rtl_uString*));
if (ppArgs != 0)
{
int i;
int nArgs;
LPWSTR *wargv = CommandLineToArgvW( GetCommandLineW(), &nArgs );
OSL_ASSERT( nArgs == argc );
for (i = 0; i < nArgs; i++)
{
/* Convert to unicode */
rtl_uString_newFromStr( &(ppArgs[i]), reinterpret_cast<const sal_Unicode*>(wargv[i]) );
}
if (ppArgs[0] != 0)
{
/* Ensure absolute path */
::osl::LongPathBuffer< sal_Unicode > aBuffer( MAX_LONG_PATH );
DWORD dwResult = 0;
dwResult = SearchPath (
0, reinterpret_cast<LPCWSTR>(ppArgs[0]->buffer), L".exe", aBuffer.getBufSizeInSymbols(), ::osl::mingw_reinterpret_cast<LPWSTR>(aBuffer), 0);
if ((0 < dwResult) && (dwResult < aBuffer.getBufSizeInSymbols()))
{
/* Replace argv[0] with it's absolute path */
rtl_uString_newFromStr_WithLength(
&(ppArgs[0]), aBuffer, dwResult);
}
}
if (ppArgs[0] != 0)
{
/* Convert to FileURL, see @ osl_getExecutableFile() */
rtl_uString * pResult = 0;
osl_getFileURLFromSystemPath (ppArgs[0], &pResult);
if (pResult != 0)
{
rtl_uString_assign (&(ppArgs[0]), pResult);
rtl_uString_release (pResult);
}
}
}
return (ppArgs);
}
#ifdef _MSC_VER
#pragma warning( pop )
#endif
/***************************************************************************/
oslProcessError SAL_CALL osl_getExecutableFile( rtl_uString **ppustrFile )
{
oslProcessError result = osl_Process_E_NotFound;
osl_acquireMutex (*osl_getGlobalMutex());
OSL_ASSERT(g_command_args.m_nCount > 0);
if (g_command_args.m_nCount > 0)
{
/* CommandArgs set. Obtain arv[0]. */
rtl_uString_assign (ppustrFile, g_command_args.m_ppArgs[0]);
result = osl_Process_E_None;
}
osl_releaseMutex (*osl_getGlobalMutex());
return (result);
}
/***************************************************************************/
sal_uInt32 SAL_CALL osl_getCommandArgCount(void)
{
sal_uInt32 result = 0;
osl_acquireMutex (*osl_getGlobalMutex());
if (g_command_args.m_nCount == 0) {
OSL_TRACE(
OSL_LOG_PREFIX
"osl_getCommandArgCount w/o prior call to osl_setCommandArgs");
}
if (g_command_args.m_nCount > 0)
{
/* We're not counting argv[0] here. */
result = g_command_args.m_nCount - 1;
}
osl_releaseMutex (*osl_getGlobalMutex());
return (result);
}
/***************************************************************************/
oslProcessError SAL_CALL osl_getCommandArg( sal_uInt32 nArg, rtl_uString **strCommandArg)
{
oslProcessError result = osl_Process_E_NotFound;
osl_acquireMutex (*osl_getGlobalMutex());
OSL_ASSERT(g_command_args.m_nCount > 0);
if (g_command_args.m_nCount > (nArg + 1))
{
/* We're not counting argv[0] here. */
rtl_uString_assign (strCommandArg, g_command_args.m_ppArgs[nArg + 1]);
result = osl_Process_E_None;
}
osl_releaseMutex (*osl_getGlobalMutex());
return (result);
}
/***************************************************************************/
void SAL_CALL osl_setCommandArgs (int argc, char ** argv)
{
OSL_ASSERT(argc > 0);
osl_acquireMutex (*osl_getGlobalMutex());
if (g_command_args.m_nCount == 0)
{
rtl_uString** ppArgs = osl_createCommandArgs_Impl (argc, argv);
if (ppArgs != 0)
{
g_command_args.m_nCount = argc;
g_command_args.m_ppArgs = ppArgs;
}
}
osl_releaseMutex (*osl_getGlobalMutex());
}
/***************************************************************************
* Environment
***************************************************************************/
/*
#109941# because of a bug in the M$ unicows library we have to
allocate a buffer large enough to hold the requested environment
variable instead of testing for the required size. This wastes
some stack space, maybe we should revoke this work around if
unicows library is fixed.
*/
#define ENV_BUFFER_SIZE (32*1024-1)
oslProcessError SAL_CALL osl_getEnvironment(rtl_uString *ustrVar, rtl_uString **ustrValue)
{
WCHAR buff[ENV_BUFFER_SIZE];
if (GetEnvironmentVariableW(reinterpret_cast<LPCWSTR>(ustrVar->buffer), buff, ENV_BUFFER_SIZE) > 0)
{
rtl_uString_newFromStr(ustrValue, reinterpret_cast<const sal_Unicode*>(buff));
return osl_Process_E_None;
}
return osl_Process_E_Unknown;
}
oslProcessError SAL_CALL osl_setEnvironment(rtl_uString *ustrVar, rtl_uString *ustrValue)
{
LPCWSTR lpName = reinterpret_cast<LPCWSTR>(ustrVar->buffer);
LPCWSTR lpValue = reinterpret_cast<LPCWSTR>(ustrValue->buffer);
if (SetEnvironmentVariableW(lpName, lpValue))
{
wchar_t *buffer = new wchar_t[wcslen(lpName) + 1 + wcslen(lpValue) + 1];
wcscpy(buffer, lpName);
wcscat(buffer, L"=");
wcscat(buffer, lpValue);
_wputenv(buffer);
delete[] buffer;
return osl_Process_E_None;
}
return osl_Process_E_Unknown;
}
oslProcessError SAL_CALL osl_clearEnvironment(rtl_uString *ustrVar)
{
//If the second parameter is NULL, the variable is deleted from the current
//process's environment.
LPCWSTR lpName = reinterpret_cast<LPCWSTR>(ustrVar->buffer);
if (SetEnvironmentVariableW(lpName, NULL))
{
wchar_t *buffer = new wchar_t[wcslen(lpName) + 1 + 1];
wcscpy(buffer, lpName);
wcscat(buffer, L"=");
_wputenv(buffer);
delete[] buffer;
return osl_Process_E_None;
}
return osl_Process_E_Unknown;
}
/***************************************************************************
* Current Working Directory.
***************************************************************************/
extern "C" oslMutex g_CurrentDirectoryMutex;
oslProcessError SAL_CALL osl_getProcessWorkingDir( rtl_uString **pustrWorkingDir )
{
::osl::LongPathBuffer< sal_Unicode > aBuffer( MAX_LONG_PATH );
DWORD dwLen = 0;
osl_acquireMutex( g_CurrentDirectoryMutex );
dwLen = GetCurrentDirectory( aBuffer.getBufSizeInSymbols(), ::osl::mingw_reinterpret_cast<LPWSTR>(aBuffer) );
osl_releaseMutex( g_CurrentDirectoryMutex );
if ( dwLen && dwLen < aBuffer.getBufSizeInSymbols() )
{
oslFileError eError;
rtl_uString *ustrTemp = NULL;;
rtl_uString_newFromStr_WithLength( &ustrTemp, aBuffer, dwLen );
eError = osl_getFileURLFromSystemPath( ustrTemp, pustrWorkingDir );
rtl_uString_release( ustrTemp );
if ( osl_File_E_None != eError )
return osl_Process_E_Unknown;
else
return osl_Process_E_None;
}
else
return osl_Process_E_Unknown;
}
/***************************************************************************
* Process Locale.
***************************************************************************/
extern "C" void _imp_getProcessLocale( rtl_Locale ** ppLocale );
static rtl_Locale * g_theProcessLocale = NULL;
/***************************************************************************/
oslProcessError SAL_CALL osl_getProcessLocale( rtl_Locale ** ppLocale )
{
osl_acquireMutex( *osl_getGlobalMutex() );
/* determine the users default locale */
if( NULL == g_theProcessLocale )
_imp_getProcessLocale( &g_theProcessLocale );
/* or return the cached value */
*ppLocale = g_theProcessLocale;
osl_releaseMutex( *osl_getGlobalMutex() );
return osl_Process_E_None;
}
/***************************************************************************/
oslProcessError SAL_CALL osl_setProcessLocale( rtl_Locale * pLocale )
{
osl_acquireMutex( *osl_getGlobalMutex() );
/* check if locale is supported */
if( RTL_TEXTENCODING_DONTKNOW == osl_getTextEncodingFromLocale( pLocale ) )
return osl_Process_E_Unknown;
/* just remember the locale here */
g_theProcessLocale = pLocale;
osl_releaseMutex( *osl_getGlobalMutex() );
return osl_Process_E_None;
}
/************************************************
* Portal send/receive interface implementation
************************************************/
static sal_Bool ReadPipe(oslPipe hPipe,
void* pBuffer,
sal_Int32 BytesToRead,
sal_Int32* nBytes)
{
*nBytes = osl_receivePipe(hPipe, pBuffer, BytesToRead);
OSL_TRACE("tried to recieve %d, recieved %d.\n",
BytesToRead, *nBytes);
return (sal_Bool)((*nBytes >= 0) && (osl_getLastPipeError(hPipe) == osl_Pipe_E_None));
}
static sal_Bool WritePipe(oslPipe hPipe,
void* pBuffer,
sal_Int32 BytesToSend,
sal_Int32* nBytes)
{
*nBytes = osl_sendPipe(hPipe, pBuffer, BytesToSend);
OSL_TRACE("tried to send %d, sent %d\n",
BytesToSend, *nBytes);
return (sal_Bool)((*nBytes == BytesToSend) && (osl_getLastPipeError(hPipe) == osl_Pipe_E_None));
}
sal_Bool SAL_CALL osl_sendResourcePipe(oslPipe hPipe, oslSocket pSocket)
{
sal_Bool bRet = sal_False;
sal_Int32 bytes = 0;
/* duplicate handle on this other side ->
receive remote process
duplicate handle and send it */
DWORD remoteProcessID = 0;
HANDLE fd = (HANDLE)pSocket->m_Socket;
oslDescriptorType code = osl_Process_TypeSocket;
OSL_TRACE("osl_sendResourcePipe: enter...");
if (ReadPipe(hPipe, &remoteProcessID, sizeof(remoteProcessID), &bytes))
{
HANDLE hRemoteProc = OpenProcess(PROCESS_DUP_HANDLE,
FALSE,
remoteProcessID);
if (hRemoteProc != (HANDLE)NULL)
{
HANDLE newFd;
if (DuplicateHandle(GetCurrentProcess(),
fd,
hRemoteProc,
&newFd,
0, FALSE, DUPLICATE_SAME_ACCESS))
{
if (
WritePipe(hPipe, &code, sizeof(code), &bytes) &&
WritePipe(hPipe, &newFd, sizeof(fd), &bytes)
)
bRet = sal_True;
}
CloseHandle(hRemoteProc);
}
}
if (bRet)
{
sal_Int32 commitCode;
OSL_TRACE("osl_sendResourcePipe: handle sent successfully, verify...");
if (
!ReadPipe(hPipe, &commitCode, sizeof(commitCode), &bytes) ||
(commitCode <= 0)
)
bRet = sal_False;
}
OSL_TRACE("osl_sendResourcePipe: exit... %d", bRet);
return(bRet);
}
oslSocket SAL_CALL osl_receiveResourcePipe(oslPipe hPipe)
{
sal_Bool bRet = sal_False;
sal_Int32 bytes = 0;
sal_Int32 commitCode;
oslSocket pSocket = NULL;
/* duplicate handle on the other side ->
send my process id receive duplicated handle */
HANDLE fd = INVALID_HANDLE_VALUE;
DWORD myProcessID = GetCurrentProcessId();
oslDescriptorType code = osl_Process_TypeNone;
OSL_TRACE("osl_receiveResourcePipe: enter...");
if (
WritePipe(hPipe, &myProcessID, sizeof(myProcessID), &bytes) &&
ReadPipe(hPipe, &code, sizeof(code), &bytes) &&
ReadPipe(hPipe, &fd, sizeof(fd), &bytes)
)
{
if (code == osl_Process_TypeSocket)
{
pSocket = __osl_createSocketImpl((SOCKET)fd);
bRet = sal_True;
}
else
{
OSL_TRACE("osl_receiveResourcePipe: UKNOWN");
bRet = sal_False;
}
}
if (bRet)
commitCode = 1;
else
commitCode = 0;
WritePipe(hPipe, &commitCode, sizeof(commitCode), &bytes);
OSL_TRACE("osl_receiveResourcePipe: exit... %d, %p", bRet, pSocket);
return pSocket;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */