/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ #include #include #include #define WIN32_LEAN_AND_MEAN #if defined _MSC_VER #pragma warning(push, 1) #endif #include #if defined _MSC_VER #pragma warning(pop) #endif #include "tools/pathutils.hxx" #include "pyversion.hxx" #define MY_LENGTH(s) (sizeof (s) / sizeof *(s) - 1) #define MY_STRING(s) (s), MY_LENGTH(s) wchar_t * encode(wchar_t * buffer, wchar_t const * text) { *buffer++ = L'"'; std::size_t n = 0; for (;;) { wchar_t c = *text++; if (c == L'\0') { break; } else if (c == L'"') { // Double any preceding backslashes as required by Windows: for (std::size_t i = 0; i < n; ++i) { *buffer++ = L'\\'; } *buffer++ = L'\\'; *buffer++ = L'"'; n = 0; } else if (c == L'\\') { *buffer++ = L'\\'; ++n; } else { *buffer++ = c; n = 0; } } // The command line will continue with a double quote, so double any // preceding backslashes as required by Windows: for (std::size_t i = 0; i < n; ++i) { *buffer++ = L'\\'; } *buffer++ = L'"'; return buffer; } #ifdef __MINGW32__ int main(int argc, char ** argv, char **) { #else int wmain(int argc, wchar_t ** argv, wchar_t **) { #endif wchar_t path[MAX_PATH]; DWORD n = GetModuleFileNameW(NULL, path, MAX_PATH); if (n == 0 || n >= MAX_PATH) { exit(EXIT_FAILURE); } wchar_t * pathEnd = tools::filename(path); *pathEnd = L'\0'; n = GetEnvironmentVariableW(L"UNO_PATH", NULL, 0); if (n == 0) { if (GetLastError() != ERROR_ENVVAR_NOT_FOUND || !SetEnvironmentVariableW(L"UNO_PATH", path)) { exit(EXIT_FAILURE); } } wchar_t bootstrap[MY_LENGTH(L"vnd.sun.star.pathname:") + MAX_PATH] = L"vnd.sun.star.pathname:"; //TODO: overflow wchar_t * bootstrapEnd = tools::buildPath( bootstrap + MY_LENGTH(L"vnd.sun.star.pathname:"), path, pathEnd, MY_STRING(L"fundamental.ini")); if (bootstrapEnd == NULL || (tools::buildPath(path, path, pathEnd, MY_STRING(L"..\\basis-link")) == NULL)) { exit(EXIT_FAILURE); } pathEnd = tools::resolveLink(path); wchar_t path1[MAX_PATH]; wchar_t * path1End = tools::buildPath( path1, path, pathEnd, MY_STRING(L"\\program")); if (path1End == NULL) { exit(EXIT_FAILURE); } wchar_t pythonpath2[MAX_PATH]; wchar_t * pythonpath2End = tools::buildPath( pythonpath2, path, pathEnd, MY_STRING(L"\\program\\python-core-" MY_PYVERSION L"\\lib")); if (pythonpath2End == NULL) { exit(EXIT_FAILURE); } wchar_t pythonpath3[MAX_PATH]; wchar_t * pythonpath3End = tools::buildPath( pythonpath3, path, pathEnd, MY_STRING( L"\\program\\python-core-" MY_PYVERSION L"\\lib\\site-packages")); if (pythonpath3End == NULL) { exit(EXIT_FAILURE); } #ifdef __MINGW32__ wchar_t pythonpath4[MAX_PATH]; wchar_t * pythonpath4End = tools::buildPath( pythonpath4, path, pathEnd, MY_STRING(L"\\program\\python-core-" MY_PYVERSION L"\\lib\\lib-dynload")); if (pythonpath4End == NULL) { exit(EXIT_FAILURE); } wchar_t pythonpath5[MAX_PATH]; wchar_t * pythonpath5End = tools::buildPath( pythonpath5, path, pathEnd, MY_STRING(L"\\program\\python-core-" MY_PYVERSION L"\\lib\\lib-dynload")); if (pythonpath5End == NULL) { exit(EXIT_FAILURE); } #endif wchar_t pythonhome[MAX_PATH]; wchar_t * pythonhomeEnd = tools::buildPath( pythonhome, path, pathEnd, MY_STRING(L"\\program\\python-core-" MY_PYVERSION)); if (pythonhomeEnd == NULL) { exit(EXIT_FAILURE); } wchar_t pythonexe[MAX_PATH]; wchar_t * pythonexeEnd = tools::buildPath( pythonexe, path, pathEnd, #ifdef __MINGW32__ MY_STRING( L"\\program\\python-core-" MY_PYVERSION L"\\bin\\python.bin")); #else MY_STRING( L"\\program\\python-core-" MY_PYVERSION L"\\bin\\python.exe")); #endif if (pythonexeEnd == NULL) { exit(EXIT_FAILURE); } if (tools::buildPath(path, path, pathEnd, MY_STRING(L"\\ure-link")) == NULL) { exit(EXIT_FAILURE); } pathEnd = tools::resolveLink(path); if (pathEnd == NULL) { exit(EXIT_FAILURE); } pathEnd = tools::buildPath(path, path, pathEnd, MY_STRING(L"\\bin")); if (pathEnd == NULL) { exit(EXIT_FAILURE); } std::size_t clSize = MY_LENGTH(L"\"") + 4 * (pythonexeEnd - pythonexe) + MY_LENGTH(L"\"\0"); //TODO: overflow // 4 * len: each char preceded by backslash, each trailing backslash // doubled for (int i = 1; i < argc; ++i) { #ifdef __MINGW32__ clSize += MY_LENGTH(L" \"") + 4 * strlen(argv[i]) + #else clSize += MY_LENGTH(L" \"") + 4 * wcslen(argv[i]) + #endif MY_LENGTH(L"\""); //TODO: overflow } wchar_t * cl = new wchar_t[clSize]; if (cl == NULL) { exit(EXIT_FAILURE); } wchar_t * cp = encode(cl, pythonhome); for (int i = 1; i < argc; ++i) { *cp++ = L' '; #ifdef __MINGW32__ int nNeededWStrBuffSize = MultiByteToWideChar(CP_ACP, 0, argv[i], -1, NULL, 0); WCHAR *buff = new WCHAR[nNeededWStrBuffSize+1]; MultiByteToWideChar(CP_ACP, 0, argv[i], -1, buff, nNeededWStrBuffSize); buff[nNeededWStrBuffSize] = 0; cp = encode(cp, buff); delete [] buff; #else cp = encode(cp, argv[i]); #endif } *cp = L'\0'; n = GetEnvironmentVariableW(L"PATH", NULL, 0); wchar_t * orig; if (n == 0) { if (GetLastError() != ERROR_ENVVAR_NOT_FOUND) { exit(EXIT_FAILURE); } orig = L""; } else { orig = new wchar_t[n]; if (orig == NULL || GetEnvironmentVariableW(L"PATH", orig, n) != n - 1) { exit(EXIT_FAILURE); } } wchar_t * value = new wchar_t[ (pathEnd - path) + MY_LENGTH(L";") + (path1End - path1) + (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow wsprintfW(value, L"%s;%s%s%s", path, path1, n == 0 ? L"" : L";", orig); if (!SetEnvironmentVariableW(L"PATH", value)) { exit(EXIT_FAILURE); } if (n != 0) { delete [] orig; } delete [] value; n = GetEnvironmentVariableW(L"PYTHONPATH", NULL, 0); if (n == 0) { if (GetLastError() != ERROR_ENVVAR_NOT_FOUND) { exit(EXIT_FAILURE); } orig = L""; } else { orig = new wchar_t[n]; if (orig == NULL || GetEnvironmentVariableW(L"PYTHONPATH", orig, n) != n - 1) { exit(EXIT_FAILURE); } } #ifdef __MINGW32__ value = new wchar_t[ (path1End - path1) + MY_LENGTH(L";") + (pythonpath2End - pythonpath2) + MY_LENGTH(L";") + (pythonpath4End - pythonpath4) + MY_LENGTH(L";") + (pythonpath5End - pythonpath5) + MY_LENGTH(L";") + (pythonpath3End - pythonpath3) + (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow wsprintfW( value, L"%s;%s;%s;%s;%s%s%s", path1, pythonpath2, pythonpath4, pythonpath5, pythonpath3, n == 0 ? L"" : L";", orig); #else value = new wchar_t[ (path1End - path1) + MY_LENGTH(L";") + (pythonpath2End - pythonpath2) + MY_LENGTH(L";") + (pythonpath3End - pythonpath3) + (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow wsprintfW( value, L"%s;%s;%s%s%s", path1, pythonpath2, pythonpath3, n == 0 ? L"" : L";", orig); #endif if (!SetEnvironmentVariableW(L"PYTHONPATH", value)) { exit(EXIT_FAILURE); } if (n != 0) { delete [] orig; } delete [] value; if (!SetEnvironmentVariableW(L"PYTHONHOME", pythonhome)) { exit(EXIT_FAILURE); } n = GetEnvironmentVariableW(L"URE_BOOTSTRAP", NULL, 0); if (n == 0) { if (GetLastError() != ERROR_ENVVAR_NOT_FOUND || !SetEnvironmentVariableW(L"URE_BOOTSTRAP", bootstrap)) { exit(EXIT_FAILURE); } } STARTUPINFOW startinfo; ZeroMemory(&startinfo, sizeof (STARTUPINFOW)); startinfo.cb = sizeof (STARTUPINFOW); PROCESS_INFORMATION procinfo; if (!CreateProcessW( pythonexe, cl, NULL, NULL, FALSE, CREATE_UNICODE_ENVIRONMENT, NULL, NULL, &startinfo, &procinfo)) { exit(EXIT_FAILURE); } WaitForSingleObject(procinfo.hProcess,INFINITE); DWORD exitStatus; GetExitCodeProcess(procinfo.hProcess,&exitStatus); exit(exitStatus); }