xref: /aoo4110/main/pyuno/zipcore/python.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski #include <cstddef>
25*b1cdbd2cSJim Jagielski #include <stdlib.h>
26*b1cdbd2cSJim Jagielski #include <wchar.h>
27*b1cdbd2cSJim Jagielski 
28*b1cdbd2cSJim Jagielski #define WIN32_LEAN_AND_MEAN
29*b1cdbd2cSJim Jagielski #if defined _MSC_VER
30*b1cdbd2cSJim Jagielski #pragma warning(push, 1)
31*b1cdbd2cSJim Jagielski #endif
32*b1cdbd2cSJim Jagielski #include <windows.h>
33*b1cdbd2cSJim Jagielski #if defined _MSC_VER
34*b1cdbd2cSJim Jagielski #pragma warning(pop)
35*b1cdbd2cSJim Jagielski #endif
36*b1cdbd2cSJim Jagielski 
37*b1cdbd2cSJim Jagielski #include "tools/pathutils.hxx"
38*b1cdbd2cSJim Jagielski 
39*b1cdbd2cSJim Jagielski #include "pyversion.hxx"
40*b1cdbd2cSJim Jagielski 
41*b1cdbd2cSJim Jagielski #define MY_LENGTH(s) (sizeof (s) / sizeof *(s) - 1)
42*b1cdbd2cSJim Jagielski #define MY_STRING(s) (s), MY_LENGTH(s)
43*b1cdbd2cSJim Jagielski 
encode(wchar_t * buffer,wchar_t const * text)44*b1cdbd2cSJim Jagielski wchar_t * encode(wchar_t * buffer, wchar_t const * text) {
45*b1cdbd2cSJim Jagielski     *buffer++ = L'"';
46*b1cdbd2cSJim Jagielski     std::size_t n = 0;
47*b1cdbd2cSJim Jagielski     for (;;) {
48*b1cdbd2cSJim Jagielski         wchar_t c = *text++;
49*b1cdbd2cSJim Jagielski         if (c == L'\0') {
50*b1cdbd2cSJim Jagielski             break;
51*b1cdbd2cSJim Jagielski         } else if (c == L'"') {
52*b1cdbd2cSJim Jagielski             // Double any preceding backslashes as required by Windows:
53*b1cdbd2cSJim Jagielski             for (std::size_t i = 0; i < n; ++i) {
54*b1cdbd2cSJim Jagielski                 *buffer++ = L'\\';
55*b1cdbd2cSJim Jagielski             }
56*b1cdbd2cSJim Jagielski             *buffer++ = L'\\';
57*b1cdbd2cSJim Jagielski             *buffer++ = L'"';
58*b1cdbd2cSJim Jagielski             n = 0;
59*b1cdbd2cSJim Jagielski         } else if (c == L'\\') {
60*b1cdbd2cSJim Jagielski             *buffer++ = L'\\';
61*b1cdbd2cSJim Jagielski             ++n;
62*b1cdbd2cSJim Jagielski         } else {
63*b1cdbd2cSJim Jagielski             *buffer++ = c;
64*b1cdbd2cSJim Jagielski             n = 0;
65*b1cdbd2cSJim Jagielski         }
66*b1cdbd2cSJim Jagielski     }
67*b1cdbd2cSJim Jagielski     // The command line will continue with a double quote, so double any
68*b1cdbd2cSJim Jagielski     // preceding backslashes as required by Windows:
69*b1cdbd2cSJim Jagielski     for (std::size_t i = 0; i < n; ++i) {
70*b1cdbd2cSJim Jagielski         *buffer++ = L'\\';
71*b1cdbd2cSJim Jagielski     }
72*b1cdbd2cSJim Jagielski     *buffer++ = L'"';
73*b1cdbd2cSJim Jagielski     return buffer;
74*b1cdbd2cSJim Jagielski }
75*b1cdbd2cSJim Jagielski 
76*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
main(int argc,char ** argv,char **)77*b1cdbd2cSJim Jagielski int main(int argc, char ** argv, char **) {
78*b1cdbd2cSJim Jagielski #else
79*b1cdbd2cSJim Jagielski int wmain(int argc, wchar_t ** argv, wchar_t **) {
80*b1cdbd2cSJim Jagielski #endif
81*b1cdbd2cSJim Jagielski     wchar_t path[MAX_PATH];
82*b1cdbd2cSJim Jagielski     DWORD n = GetModuleFileNameW(NULL, path, MAX_PATH);
83*b1cdbd2cSJim Jagielski     if (n == 0 || n >= MAX_PATH) {
84*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
85*b1cdbd2cSJim Jagielski     }
86*b1cdbd2cSJim Jagielski     wchar_t * pathEnd = tools::filename(path);
87*b1cdbd2cSJim Jagielski     *pathEnd = L'\0';
88*b1cdbd2cSJim Jagielski     n = GetEnvironmentVariableW(L"UNO_PATH", NULL, 0);
89*b1cdbd2cSJim Jagielski     if (n == 0) {
90*b1cdbd2cSJim Jagielski         if (GetLastError() != ERROR_ENVVAR_NOT_FOUND ||
91*b1cdbd2cSJim Jagielski             !SetEnvironmentVariableW(L"UNO_PATH", path))
92*b1cdbd2cSJim Jagielski         {
93*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
94*b1cdbd2cSJim Jagielski         }
95*b1cdbd2cSJim Jagielski     }
96*b1cdbd2cSJim Jagielski 
97*b1cdbd2cSJim Jagielski     wchar_t bootstrap[MY_LENGTH(L"vnd.sun.star.pathname:") + MAX_PATH] =
98*b1cdbd2cSJim Jagielski         L"vnd.sun.star.pathname:"; //TODO: overflow
99*b1cdbd2cSJim Jagielski     wchar_t * bootstrapEnd = tools::buildPath(
100*b1cdbd2cSJim Jagielski         bootstrap + MY_LENGTH(L"vnd.sun.star.pathname:"), path, pathEnd,
101*b1cdbd2cSJim Jagielski         MY_STRING(L"fundamental.ini"));
102*b1cdbd2cSJim Jagielski     if (bootstrapEnd == NULL)
103*b1cdbd2cSJim Jagielski     {
104*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
105*b1cdbd2cSJim Jagielski     }
106*b1cdbd2cSJim Jagielski 
107*b1cdbd2cSJim Jagielski     wchar_t pythonpath2[MAX_PATH];
108*b1cdbd2cSJim Jagielski     wchar_t * pythonpath2End = tools::buildPath(
109*b1cdbd2cSJim Jagielski         pythonpath2, path, pathEnd,
110*b1cdbd2cSJim Jagielski         MY_STRING(L"python-core-" MY_PYVERSION L"\\lib"));
111*b1cdbd2cSJim Jagielski     if (pythonpath2End == NULL) {
112*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
113*b1cdbd2cSJim Jagielski     }
114*b1cdbd2cSJim Jagielski 
115*b1cdbd2cSJim Jagielski     wchar_t pythonpath3[MAX_PATH];
116*b1cdbd2cSJim Jagielski     wchar_t * pythonpath3End = tools::buildPath(
117*b1cdbd2cSJim Jagielski         pythonpath3, path, pathEnd,
118*b1cdbd2cSJim Jagielski         MY_STRING(
119*b1cdbd2cSJim Jagielski             L"python-core-" MY_PYVERSION L"\\lib\\site-packages"));
120*b1cdbd2cSJim Jagielski     if (pythonpath3End == NULL) {
121*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
122*b1cdbd2cSJim Jagielski     }
123*b1cdbd2cSJim Jagielski 
124*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
125*b1cdbd2cSJim Jagielski     wchar_t pythonpath4[MAX_PATH];
126*b1cdbd2cSJim Jagielski     wchar_t * pythonpath4End = tools::buildPath(
127*b1cdbd2cSJim Jagielski         pythonpath4, path, pathEnd,
128*b1cdbd2cSJim Jagielski         MY_STRING(L"python-core-" MY_PYVERSION L"\\lib\\lib-dynload"));
129*b1cdbd2cSJim Jagielski     if (pythonpath4End == NULL) {
130*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
131*b1cdbd2cSJim Jagielski     }
132*b1cdbd2cSJim Jagielski     wchar_t pythonpath5[MAX_PATH];
133*b1cdbd2cSJim Jagielski     wchar_t * pythonpath5End = tools::buildPath(
134*b1cdbd2cSJim Jagielski         pythonpath5, path, pathEnd,
135*b1cdbd2cSJim Jagielski         MY_STRING(L"python-core-" MY_PYVERSION L"\\lib\\lib-dynload"));
136*b1cdbd2cSJim Jagielski     if (pythonpath5End == NULL) {
137*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
138*b1cdbd2cSJim Jagielski     }
139*b1cdbd2cSJim Jagielski #endif
140*b1cdbd2cSJim Jagielski     wchar_t pythonhome[MAX_PATH];
141*b1cdbd2cSJim Jagielski     wchar_t * pythonhomeEnd = tools::buildPath(
142*b1cdbd2cSJim Jagielski         pythonhome, path, pathEnd,
143*b1cdbd2cSJim Jagielski         MY_STRING(L"python-core-" MY_PYVERSION));
144*b1cdbd2cSJim Jagielski     if (pythonhomeEnd == NULL) {
145*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
146*b1cdbd2cSJim Jagielski     }
147*b1cdbd2cSJim Jagielski     wchar_t pythonexe[MAX_PATH];
148*b1cdbd2cSJim Jagielski     wchar_t * pythonexeEnd = tools::buildPath(
149*b1cdbd2cSJim Jagielski         pythonexe, path, pathEnd,
150*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
151*b1cdbd2cSJim Jagielski         MY_STRING(
152*b1cdbd2cSJim Jagielski             L"python-core-" MY_PYVERSION L"\\bin\\python.bin"));
153*b1cdbd2cSJim Jagielski #else
154*b1cdbd2cSJim Jagielski         MY_STRING(
155*b1cdbd2cSJim Jagielski             L"python-core-" MY_PYVERSION L"\\bin\\python.exe"));
156*b1cdbd2cSJim Jagielski #endif
157*b1cdbd2cSJim Jagielski     if (pythonexeEnd == NULL) {
158*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
159*b1cdbd2cSJim Jagielski     }
160*b1cdbd2cSJim Jagielski 
161*b1cdbd2cSJim Jagielski     std::size_t clSize = MY_LENGTH(L"\"") + 4 * (pythonexeEnd - pythonexe) +
162*b1cdbd2cSJim Jagielski         MY_LENGTH(L"\"\0"); //TODO: overflow
163*b1cdbd2cSJim Jagielski         // 4 * len: each char preceded by backslash, each trailing backslash
164*b1cdbd2cSJim Jagielski         // doubled
165*b1cdbd2cSJim Jagielski     for (int i = 1; i < argc; ++i) {
166*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
167*b1cdbd2cSJim Jagielski         clSize += MY_LENGTH(L" \"") + 4 * strlen(argv[i]) +
168*b1cdbd2cSJim Jagielski #else
169*b1cdbd2cSJim Jagielski         clSize += MY_LENGTH(L" \"") + 4 * wcslen(argv[i]) +
170*b1cdbd2cSJim Jagielski #endif
171*b1cdbd2cSJim Jagielski             MY_LENGTH(L"\""); //TODO: overflow
172*b1cdbd2cSJim Jagielski     }
173*b1cdbd2cSJim Jagielski     wchar_t * cl = new wchar_t[clSize];
174*b1cdbd2cSJim Jagielski     if (cl == NULL) {
175*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
176*b1cdbd2cSJim Jagielski     }
177*b1cdbd2cSJim Jagielski     wchar_t * cp = encode(cl, pythonhome);
178*b1cdbd2cSJim Jagielski     for (int i = 1; i < argc; ++i) {
179*b1cdbd2cSJim Jagielski         *cp++ = L' ';
180*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
181*b1cdbd2cSJim Jagielski         int nNeededWStrBuffSize = MultiByteToWideChar(CP_ACP, 0, argv[i], -1, NULL, 0);
182*b1cdbd2cSJim Jagielski         WCHAR *buff = new WCHAR[nNeededWStrBuffSize+1];
183*b1cdbd2cSJim Jagielski         MultiByteToWideChar(CP_ACP, 0, argv[i], -1, buff, nNeededWStrBuffSize);
184*b1cdbd2cSJim Jagielski         buff[nNeededWStrBuffSize] = 0;
185*b1cdbd2cSJim Jagielski         cp = encode(cp, buff);
186*b1cdbd2cSJim Jagielski         delete [] buff;
187*b1cdbd2cSJim Jagielski #else
188*b1cdbd2cSJim Jagielski         cp = encode(cp, argv[i]);
189*b1cdbd2cSJim Jagielski #endif
190*b1cdbd2cSJim Jagielski     }
191*b1cdbd2cSJim Jagielski     *cp = L'\0';
192*b1cdbd2cSJim Jagielski     n = GetEnvironmentVariableW(L"PATH", NULL, 0);
193*b1cdbd2cSJim Jagielski     wchar_t * orig;
194*b1cdbd2cSJim Jagielski     if (n == 0) {
195*b1cdbd2cSJim Jagielski         if (GetLastError() != ERROR_ENVVAR_NOT_FOUND) {
196*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
197*b1cdbd2cSJim Jagielski         }
198*b1cdbd2cSJim Jagielski         orig = L"";
199*b1cdbd2cSJim Jagielski     } else {
200*b1cdbd2cSJim Jagielski         orig = new wchar_t[n];
201*b1cdbd2cSJim Jagielski         if (orig == NULL ||
202*b1cdbd2cSJim Jagielski             GetEnvironmentVariableW(L"PATH", orig, n) != n - 1)
203*b1cdbd2cSJim Jagielski         {
204*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
205*b1cdbd2cSJim Jagielski         }
206*b1cdbd2cSJim Jagielski     }
207*b1cdbd2cSJim Jagielski     wchar_t * value = new wchar_t[
208*b1cdbd2cSJim Jagielski        (pathEnd - path) + MY_LENGTH(L";") +
209*b1cdbd2cSJim Jagielski        (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow
210*b1cdbd2cSJim Jagielski 
211*b1cdbd2cSJim Jagielski     wsprintfW(value, L"%s%s%s", path, n == 0 ? L"" : L";", orig);
212*b1cdbd2cSJim Jagielski 
213*b1cdbd2cSJim Jagielski     if (!SetEnvironmentVariableW(L"PATH", value)) {
214*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
215*b1cdbd2cSJim Jagielski     }
216*b1cdbd2cSJim Jagielski     if (n != 0) {
217*b1cdbd2cSJim Jagielski         delete [] orig;
218*b1cdbd2cSJim Jagielski     }
219*b1cdbd2cSJim Jagielski     delete [] value;
220*b1cdbd2cSJim Jagielski     n = GetEnvironmentVariableW(L"PYTHONPATH", NULL, 0);
221*b1cdbd2cSJim Jagielski     if (n == 0) {
222*b1cdbd2cSJim Jagielski         if (GetLastError() != ERROR_ENVVAR_NOT_FOUND) {
223*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
224*b1cdbd2cSJim Jagielski         }
225*b1cdbd2cSJim Jagielski         orig = L"";
226*b1cdbd2cSJim Jagielski     } else {
227*b1cdbd2cSJim Jagielski         orig = new wchar_t[n];
228*b1cdbd2cSJim Jagielski         if (orig == NULL ||
229*b1cdbd2cSJim Jagielski             GetEnvironmentVariableW(L"PYTHONPATH", orig, n) != n - 1)
230*b1cdbd2cSJim Jagielski         {
231*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
232*b1cdbd2cSJim Jagielski         }
233*b1cdbd2cSJim Jagielski     }
234*b1cdbd2cSJim Jagielski #ifdef __MINGW32__
235*b1cdbd2cSJim Jagielski     value = new wchar_t[
236*b1cdbd2cSJim Jagielski         (path1End - path1) + MY_LENGTH(L";") + (pythonpath2End - pythonpath2) +
237*b1cdbd2cSJim Jagielski         MY_LENGTH(L";") + (pythonpath4End - pythonpath4) +
238*b1cdbd2cSJim Jagielski         MY_LENGTH(L";") + (pythonpath5End - pythonpath5) +
239*b1cdbd2cSJim Jagielski         MY_LENGTH(L";") + (pythonpath3End - pythonpath3) +
240*b1cdbd2cSJim Jagielski         (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow
241*b1cdbd2cSJim Jagielski     wsprintfW(
242*b1cdbd2cSJim Jagielski         value, L"%s;%s;%s;%s;%s%s%s", path1, pythonpath2, pythonpath4,
243*b1cdbd2cSJim Jagielski         pythonpath5, pythonpath3,
244*b1cdbd2cSJim Jagielski         n == 0 ? L"" : L";", orig);
245*b1cdbd2cSJim Jagielski #else
246*b1cdbd2cSJim Jagielski     value = new wchar_t[
247*b1cdbd2cSJim Jagielski         (pythonpath2End - pythonpath2) +
248*b1cdbd2cSJim Jagielski         MY_LENGTH(L";") + (pythonpath3End - pythonpath3) +
249*b1cdbd2cSJim Jagielski         (n == 0 ? 0 : MY_LENGTH(L";") + (n - 1)) + 1]; //TODO: overflow
250*b1cdbd2cSJim Jagielski     wsprintfW(
251*b1cdbd2cSJim Jagielski         value, L"%s;%s%s%s", pythonpath2, pythonpath3,
252*b1cdbd2cSJim Jagielski         n == 0 ? L"" : L";", orig);
253*b1cdbd2cSJim Jagielski #endif
254*b1cdbd2cSJim Jagielski     if (!SetEnvironmentVariableW(L"PYTHONPATH", value)) {
255*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
256*b1cdbd2cSJim Jagielski     }
257*b1cdbd2cSJim Jagielski     if (n != 0) {
258*b1cdbd2cSJim Jagielski         delete [] orig;
259*b1cdbd2cSJim Jagielski     }
260*b1cdbd2cSJim Jagielski     delete [] value;
261*b1cdbd2cSJim Jagielski     if (!SetEnvironmentVariableW(L"PYTHONHOME", pythonhome)) {
262*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
263*b1cdbd2cSJim Jagielski     }
264*b1cdbd2cSJim Jagielski     n = GetEnvironmentVariableW(L"URE_BOOTSTRAP", NULL, 0);
265*b1cdbd2cSJim Jagielski     if (n == 0) {
266*b1cdbd2cSJim Jagielski         if (GetLastError() != ERROR_ENVVAR_NOT_FOUND ||
267*b1cdbd2cSJim Jagielski             !SetEnvironmentVariableW(L"URE_BOOTSTRAP", bootstrap))
268*b1cdbd2cSJim Jagielski         {
269*b1cdbd2cSJim Jagielski             exit(EXIT_FAILURE);
270*b1cdbd2cSJim Jagielski         }
271*b1cdbd2cSJim Jagielski     }
272*b1cdbd2cSJim Jagielski     STARTUPINFOW startinfo;
273*b1cdbd2cSJim Jagielski     ZeroMemory(&startinfo, sizeof (STARTUPINFOW));
274*b1cdbd2cSJim Jagielski     startinfo.cb = sizeof (STARTUPINFOW);
275*b1cdbd2cSJim Jagielski     PROCESS_INFORMATION procinfo;
276*b1cdbd2cSJim Jagielski     if (!CreateProcessW(
277*b1cdbd2cSJim Jagielski             pythonexe, cl, NULL, NULL, FALSE, CREATE_UNICODE_ENVIRONMENT, NULL,
278*b1cdbd2cSJim Jagielski             NULL, &startinfo, &procinfo)) {
279*b1cdbd2cSJim Jagielski         exit(EXIT_FAILURE);
280*b1cdbd2cSJim Jagielski     }
281*b1cdbd2cSJim Jagielski     WaitForSingleObject(procinfo.hProcess,INFINITE);
282*b1cdbd2cSJim Jagielski     DWORD exitStatus;
283*b1cdbd2cSJim Jagielski     GetExitCodeProcess(procinfo.hProcess,&exitStatus);
284*b1cdbd2cSJim Jagielski     exit(exitStatus);
285*b1cdbd2cSJim Jagielski }
286