blob: d605bab5dd46fb77726117f9b979613420864229 [file] [log] [blame]
Guido van Rossum667d7041995-08-04 04:20:48 +00001/* Python interpreter main program */
2
3#include "Python.h"
Guido van Rossuma075ce11997-12-05 21:56:45 +00004#include "osdefs.h"
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +00005#include "import.h"
Guido van Rossum667d7041995-08-04 04:20:48 +00006
Antoine Pitrou5651eaa2008-09-06 20:46:58 +00007#include <locale.h>
8
Martin v. Löwis79acb9e2002-12-06 12:48:53 +00009#ifdef __VMS
Martin v. Löwis7a924e62003-03-05 14:15:21 +000010#include <unixlib.h>
Martin v. Löwis79acb9e2002-12-06 12:48:53 +000011#endif
12
Sjoerd Mullender9cf424b2002-08-09 13:35:18 +000013#if defined(MS_WINDOWS) || defined(__CYGWIN__)
Martin v. Löwis945362c2007-08-30 14:57:25 +000014#include <windows.h>
Thomas Wouters477c8d52006-05-27 19:21:47 +000015#ifdef HAVE_FCNTL_H
Guido van Rossum3e7ae7a1997-01-17 22:05:38 +000016#include <fcntl.h>
Martin v. Löwis790465f2008-04-05 20:41:37 +000017#define PATH_MAX MAXPATHLEN
Guido van Rossum3e7ae7a1997-01-17 22:05:38 +000018#endif
Thomas Wouters477c8d52006-05-27 19:21:47 +000019#endif
Guido van Rossum3e7ae7a1997-01-17 22:05:38 +000020
Martin v. Löwis945362c2007-08-30 14:57:25 +000021#ifdef _MSC_VER
22#include <crtdbg.h>
23#endif
24
Andrew MacIntyre7bf68332002-03-03 02:59:16 +000025#if (defined(PYOS_OS2) && !defined(PYCC_GCC)) || defined(MS_WINDOWS)
Guido van Rossuma075ce11997-12-05 21:56:45 +000026#define PYTHONHOMEHELP "<prefix>\\lib"
27#else
Andrew MacIntyre7bf68332002-03-03 02:59:16 +000028#if defined(PYOS_OS2) && defined(PYCC_GCC)
29#define PYTHONHOMEHELP "<prefix>/Lib"
30#else
Marc-André Lemburgda4dbc32001-06-12 16:13:51 +000031#define PYTHONHOMEHELP "<prefix>/pythonX.X"
Guido van Rossuma075ce11997-12-05 21:56:45 +000032#endif
Andrew MacIntyre7bf68332002-03-03 02:59:16 +000033#endif
Guido van Rossuma075ce11997-12-05 21:56:45 +000034
Thomas Wouters2cffc7d2000-11-03 08:18:37 +000035#include "pygetopt.h"
36
Guido van Rossuma22865e2000-09-05 04:41:18 +000037#define COPYRIGHT \
Guido van Rossum36002d72001-07-18 16:59:46 +000038 "Type \"help\", \"copyright\", \"credits\" or \"license\" " \
39 "for more information."
Guido van Rossuma22865e2000-09-05 04:41:18 +000040
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000041#ifdef __cplusplus
42extern "C" {
43#endif
44
Guido van Rossumac56b031996-07-21 02:33:38 +000045/* For Py_GetArgcArgv(); set by main() */
Martin v. Löwis790465f2008-04-05 20:41:37 +000046static wchar_t **orig_argv;
Guido van Rossum667d7041995-08-04 04:20:48 +000047static int orig_argc;
48
Guido van Rossumbceccf52001-04-10 22:07:43 +000049/* command line options */
Christian Heimes8dc226f2008-05-06 23:45:46 +000050#define BASE_OPTS L"bBc:dEhiJm:OsStuvVW:xX?"
Guido van Rossumbceccf52001-04-10 22:07:43 +000051
Guido van Rossumbceccf52001-04-10 22:07:43 +000052#define PROGRAM_OPTS BASE_OPTS
Guido van Rossum3ed4c152001-03-02 06:18:03 +000053
Guido van Rossum667d7041995-08-04 04:20:48 +000054/* Short usage message (with %s for argv0) */
55static char *usage_line =
Martin v. Löwis790465f2008-04-05 20:41:37 +000056"usage: %ls [option] ... [-c cmd | -m mod | file | -] [arg] ...\n";
Guido van Rossum667d7041995-08-04 04:20:48 +000057
58/* Long usage message, split into parts < 512 bytes */
Guido van Rossum393661d2001-08-31 17:40:15 +000059static char *usage_1 = "\
Guido van Rossum667d7041995-08-04 04:20:48 +000060Options and arguments (and corresponding environment variables):\n\
Christian Heimes2ab34442008-09-03 20:31:07 +000061-b : issue warnings about str(bytes_instance), str(bytearray_instance)\n\
62 and comparing bytes/bytearray with str. (-bb: issue errors)\n\
Christian Heimes790c8232008-01-07 21:14:23 +000063-B : don't write .py[co] files on import; also PYTHONDONTWRITEBYTECODE=x\n\
Guido van Rossum393661d2001-08-31 17:40:15 +000064-c cmd : program passed in as string (terminates option list)\n\
Thomas Wouters89f507f2006-12-13 04:49:30 +000065-d : debug output from parser; also PYTHONDEBUG=x\n\
Christian Heimes790c8232008-01-07 21:14:23 +000066-E : ignore PYTHON* environment variables (such as PYTHONPATH)\n\
Thomas Wouters0e3f5912006-08-11 14:57:12 +000067-h : print this help message and exit (also --help)\n\
Guido van Rossum61c345f2001-09-04 03:26:15 +000068";
69static char *usage_2 = "\
Guido van Rossum98297ee2007-11-06 21:34:58 +000070-i : inspect interactively after running script; forces a prompt even\n\
71 if stdin does not appear to be a terminal; also PYTHONINSPECT=x\n\
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +000072-m mod : run library module as a script (terminates option list)\n\
Thomas Wouters89f507f2006-12-13 04:49:30 +000073-O : optimize generated bytecode slightly; also PYTHONOPTIMIZE=x\n\
Guido van Rossum6b86a421999-01-28 15:07:47 +000074-OO : remove doc-strings in addition to the -O optimizations\n\
Christian Heimes8dc226f2008-05-06 23:45:46 +000075-s : don't add user site directory to sys.path; also PYTHONNOUSERSITE\n\
Guido van Rossum7922bd71997-08-29 22:34:47 +000076-S : don't imply 'import site' on initialization\n\
Guido van Rossum393661d2001-08-31 17:40:15 +000077";
78static char *usage_3 = "\
Guido van Rossum98297ee2007-11-06 21:34:58 +000079-u : unbuffered binary stdout and stderr; also PYTHONUNBUFFERED=x\n\
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +000080 see man page for details on internal buffering relating to '-u'\n\
Thomas Wouters89f507f2006-12-13 04:49:30 +000081-v : verbose (trace import statements); also PYTHONVERBOSE=x\n\
82 can be supplied multiple times to increase verbosity\n\
Thomas Wouters0e3f5912006-08-11 14:57:12 +000083-V : print the Python version number and exit (also --version)\n\
Thomas Wouters89f507f2006-12-13 04:49:30 +000084-W arg : warning control; arg is action:message:category:module:lineno\n\
Philip Jenvey0805ca32010-04-07 04:04:10 +000085 also PYTHONWARNINGS=arg\n\
Guido van Rossum393661d2001-08-31 17:40:15 +000086-x : skip first line of source, allowing use of non-Unix forms of #!cmd\n\
Guido van Rossum7922bd71997-08-29 22:34:47 +000087";
Guido van Rossum393661d2001-08-31 17:40:15 +000088static char *usage_4 = "\
Guido van Rossum98297ee2007-11-06 21:34:58 +000089file : program read from script file\n\
90- : program read from stdin (default; interactive mode if a tty)\n\
Thomas Wouters89f507f2006-12-13 04:49:30 +000091arg ...: arguments passed to program in sys.argv[1:]\n\n\
Guido van Rossum667d7041995-08-04 04:20:48 +000092Other environment variables:\n\
93PYTHONSTARTUP: file executed on interactive startup (no default)\n\
Guido van Rossuma075ce11997-12-05 21:56:45 +000094PYTHONPATH : '%c'-separated list of directories prefixed to the\n\
Guido van Rossum667d7041995-08-04 04:20:48 +000095 default module search path. The result is sys.path.\n\
Christian Heimes790c8232008-01-07 21:14:23 +000096";
97static char *usage_5 = "\
Guido van Rossuma075ce11997-12-05 21:56:45 +000098PYTHONHOME : alternate <prefix> directory (or <prefix>%c<exec_prefix>).\n\
99 The default module search path uses %s.\n\
Tim Peters793de092001-02-22 00:39:47 +0000100PYTHONCASEOK : ignore case in 'import' statements (Windows).\n\
Martin v. Löwis0f599892008-06-02 11:13:03 +0000101PYTHONIOENCODING: Encoding[:errors] used for stdin/stdout/stderr.\n\
Guido van Rossum667d7041995-08-04 04:20:48 +0000102";
103
Antoine Pitroue9b428f2010-08-13 22:25:01 +0000104FILE *
105_Py_wfopen(const wchar_t *path, const wchar_t *mode)
Martin v. Löwis790465f2008-04-05 20:41:37 +0000106{
Antoine Pitroue9b428f2010-08-13 22:25:01 +0000107#ifndef MS_WINDOWS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000108 char cpath[PATH_MAX];
109 char cmode[10];
110 size_t r;
111 r = wcstombs(cpath, path, PATH_MAX);
112 if (r == (size_t)-1 || r >= PATH_MAX) {
113 errno = EINVAL;
114 return NULL;
115 }
116 r = wcstombs(cmode, mode, 10);
117 if (r == (size_t)-1 || r >= 10) {
118 errno = EINVAL;
119 return NULL;
120 }
121 return fopen(cpath, cmode);
Antoine Pitroue9b428f2010-08-13 22:25:01 +0000122#else
123 return _wfopen(path, mode);
Martin v. Löwis790465f2008-04-05 20:41:37 +0000124#endif
Antoine Pitroue9b428f2010-08-13 22:25:01 +0000125}
Martin v. Löwis790465f2008-04-05 20:41:37 +0000126
Guido van Rossum667d7041995-08-04 04:20:48 +0000127
Martin v. Löwis852ba7e2003-03-30 17:09:58 +0000128static int
Martin v. Löwis790465f2008-04-05 20:41:37 +0000129usage(int exitcode, wchar_t* program)
Barry Warsaw3b2aedb2000-09-15 18:40:42 +0000130{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000131 FILE *f = exitcode ? stderr : stdout;
Guido van Rossum393661d2001-08-31 17:40:15 +0000132
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000133 fprintf(f, usage_line, program);
134 if (exitcode)
135 fprintf(f, "Try `python -h' for more information.\n");
136 else {
137 fputs(usage_1, f);
138 fputs(usage_2, f);
139 fputs(usage_3, f);
140 fprintf(f, usage_4, DELIM);
141 fprintf(f, usage_5, DELIM, PYTHONHOMEHELP);
142 }
Martin v. Löwis79acb9e2002-12-06 12:48:53 +0000143#if defined(__VMS)
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000144 if (exitcode == 0) {
145 /* suppress 'error' message */
146 return 1;
147 }
148 else {
149 /* STS$M_INHIB_MSG + SS$_ABORT */
150 return 0x1000002c;
151 }
Martin v. Löwis79acb9e2002-12-06 12:48:53 +0000152#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000153 return exitcode;
Martin v. Löwis79acb9e2002-12-06 12:48:53 +0000154#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000155 /*NOTREACHED*/
Barry Warsaw3b2aedb2000-09-15 18:40:42 +0000156}
157
Martin v. Löwis6caea372003-11-18 19:46:25 +0000158static void RunStartupFile(PyCompilerFlags *cf)
159{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000160 char *startup = Py_GETENV("PYTHONSTARTUP");
161 if (startup != NULL && startup[0] != '\0') {
162 FILE *fp = fopen(startup, "r");
163 if (fp != NULL) {
164 (void) PyRun_SimpleFileExFlags(fp, startup, 0, cf);
165 PyErr_Clear();
166 fclose(fp);
167 } else {
168 int save_errno;
169
170 save_errno = errno;
171 PySys_WriteStderr("Could not open PYTHONSTARTUP\n");
172 errno = save_errno;
173 PyErr_SetFromErrnoWithFilename(PyExc_IOError,
174 startup);
175 PyErr_Print();
176 PyErr_Clear();
177 }
178 }
Martin v. Löwis6caea372003-11-18 19:46:25 +0000179}
180
Thomas Woutersa9773292006-04-21 09:43:23 +0000181
Antoine Pitrou5651eaa2008-09-06 20:46:58 +0000182static int RunModule(wchar_t *modname, int set_argv0)
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +0000183{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000184 PyObject *module, *runpy, *runmodule, *runargs, *result;
185 runpy = PyImport_ImportModule("runpy");
186 if (runpy == NULL) {
187 fprintf(stderr, "Could not import runpy module\n");
188 return -1;
189 }
190 runmodule = PyObject_GetAttrString(runpy, "_run_module_as_main");
191 if (runmodule == NULL) {
192 fprintf(stderr, "Could not access runpy._run_module_as_main\n");
193 Py_DECREF(runpy);
194 return -1;
195 }
196 module = PyUnicode_FromWideChar(modname, wcslen(modname));
197 if (module == NULL) {
198 fprintf(stderr, "Could not convert module name to unicode\n");
199 Py_DECREF(runpy);
200 Py_DECREF(runmodule);
201 return -1;
202 }
203 runargs = Py_BuildValue("(Oi)", module, set_argv0);
204 if (runargs == NULL) {
205 fprintf(stderr,
206 "Could not create arguments for runpy._run_module_as_main\n");
207 Py_DECREF(runpy);
208 Py_DECREF(runmodule);
209 Py_DECREF(module);
210 return -1;
211 }
212 result = PyObject_Call(runmodule, runargs, NULL);
213 if (result == NULL) {
214 PyErr_Print();
215 }
216 Py_DECREF(runpy);
217 Py_DECREF(runmodule);
218 Py_DECREF(module);
219 Py_DECREF(runargs);
220 if (result == NULL) {
221 return -1;
222 }
223 Py_DECREF(result);
224 return 0;
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +0000225}
Barry Warsaw3b2aedb2000-09-15 18:40:42 +0000226
Martin v. Löwis790465f2008-04-05 20:41:37 +0000227static int RunMainFromImporter(wchar_t *filename)
Christian Heimes9cd17752007-11-18 19:35:23 +0000228{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000229 PyObject *argv0 = NULL, *importer = NULL;
Christian Heimes9cd17752007-11-18 19:35:23 +0000230
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000231 if ((argv0 = PyUnicode_FromWideChar(filename,wcslen(filename))) &&
232 (importer = PyImport_GetImporter(argv0)) &&
233 (importer->ob_type != &PyNullImporter_Type))
234 {
235 /* argv0 is usable as an import source, so
236 put it in sys.path[0] and import __main__ */
237 PyObject *sys_path = NULL;
238 if ((sys_path = PySys_GetObject("path")) &&
239 !PyList_SetItem(sys_path, 0, argv0))
240 {
241 Py_INCREF(argv0);
242 Py_DECREF(importer);
243 sys_path = NULL;
244 return RunModule(L"__main__", 0) != 0;
Guido van Rossum74c29c72007-11-19 18:36:41 +0000245 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000246 }
247 Py_XDECREF(argv0);
248 Py_XDECREF(importer);
249 if (PyErr_Occurred()) {
250 PyErr_Print();
251 return 1;
252 }
253 else {
254 return -1;
255 }
Christian Heimes9cd17752007-11-18 19:35:23 +0000256}
257
Victor Stinnera62207c2010-08-07 10:57:17 +0000258static int
259run_command(wchar_t *command, PyCompilerFlags *cf)
260{
261 PyObject *unicode, *bytes;
262 int ret;
263
264 unicode = PyUnicode_FromWideChar(command, -1);
265 if (unicode == NULL)
266 goto error;
267 bytes = PyUnicode_AsUTF8String(unicode);
268 Py_DECREF(unicode);
269 if (bytes == NULL)
270 goto error;
271 ret = PyRun_SimpleStringFlags(PyBytes_AsString(bytes), cf);
272 Py_DECREF(bytes);
273 return ret != 0;
274
275error:
276 PyErr_Print();
277 return 1;
278}
279
Victor Stinner0a3ddad2010-08-07 16:34:25 +0000280static int
281run_file(FILE *fp, const wchar_t *filename, PyCompilerFlags *p_cf)
282{
283 PyObject *unicode, *bytes = NULL;
284 char *filename_str;
285 int run;
286
287 /* call pending calls like signal handlers (SIGINT) */
288 if (Py_MakePendingCalls() == -1) {
289 PyErr_Print();
290 return 1;
291 }
292
293 if (filename) {
294 unicode = PyUnicode_FromWideChar(filename, wcslen(filename));
295 if (unicode != NULL) {
296 bytes = PyUnicode_AsUTF8String(unicode);
297 Py_DECREF(unicode);
298 }
299 if (bytes != NULL)
300 filename_str = PyBytes_AsString(bytes);
301 else {
302 PyErr_Clear();
303 filename_str = "<decoding error>";
304 }
305 }
306 else
307 filename_str = "<stdin>";
308
309 run = PyRun_AnyFileExFlags(fp, filename_str, filename != NULL, p_cf);
310 Py_XDECREF(bytes);
311 return run != 0;
312}
313
Christian Heimes9cd17752007-11-18 19:35:23 +0000314
Guido van Rossum667d7041995-08-04 04:20:48 +0000315/* Main program */
316
Mark Hammondfe51c6d2002-08-02 02:27:13 +0000317int
Martin v. Löwis790465f2008-04-05 20:41:37 +0000318Py_Main(int argc, wchar_t **argv)
Guido van Rossum667d7041995-08-04 04:20:48 +0000319{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000320 int c;
321 int sts;
322 wchar_t *command = NULL;
323 wchar_t *filename = NULL;
324 wchar_t *module = NULL;
325 FILE *fp = stdin;
326 char *p;
Philip Jenveye53de3d2010-04-14 03:01:39 +0000327#ifdef MS_WINDOWS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000328 wchar_t *wp;
Philip Jenveye53de3d2010-04-14 03:01:39 +0000329#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000330 int skipfirstline = 0;
331 int stdin_is_interactive = 0;
332 int help = 0;
333 int version = 0;
334 int saw_unbuffered_flag = 0;
335 PyCompilerFlags cf;
Guido van Rossum667d7041995-08-04 04:20:48 +0000336
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000337 cf.cf_flags = 0;
Guido van Rossum393661d2001-08-31 17:40:15 +0000338
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000339 orig_argc = argc; /* For Py_GetArgcArgv() */
340 orig_argv = argv;
Guido van Rossum667d7041995-08-04 04:20:48 +0000341
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000342 PySys_ResetWarnOptions();
Guido van Rossum47f5fdc2000-12-15 22:00:54 +0000343
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000344 while ((c = _PyOS_GetOpt(argc, argv, PROGRAM_OPTS)) != EOF) {
345 if (c == 'c') {
346 size_t len;
347 /* -c is the last option; following arguments
348 that look like options are left for the
349 command to interpret. */
Amaury Forgeot d'Arc9a5499b2008-11-11 23:04:59 +0000350
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000351 len = wcslen(_PyOS_optarg) + 1 + 1;
352 command = (wchar_t *)malloc(sizeof(wchar_t) * len);
353 if (command == NULL)
354 Py_FatalError(
355 "not enough memory to copy -c argument");
356 wcscpy(command, _PyOS_optarg);
357 command[len - 2] = '\n';
358 command[len - 1] = 0;
359 break;
360 }
Guido van Rossum393661d2001-08-31 17:40:15 +0000361
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000362 if (c == 'm') {
363 /* -m is the last option; following arguments
364 that look like options are left for the
365 module to interpret. */
366 module = _PyOS_optarg;
367 break;
368 }
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +0000369
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000370 switch (c) {
371 case 'b':
372 Py_BytesWarningFlag++;
373 break;
Guido van Rossum667d7041995-08-04 04:20:48 +0000374
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000375 case 'd':
376 Py_DebugFlag++;
377 break;
Guido van Rossum667d7041995-08-04 04:20:48 +0000378
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000379 case 'i':
380 Py_InspectFlag++;
381 Py_InteractiveFlag++;
382 break;
Guido van Rossum667d7041995-08-04 04:20:48 +0000383
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000384 /* case 'J': reserved for Jython */
Christian Heimes33fe8092008-04-13 13:53:33 +0000385
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000386 case 'O':
387 Py_OptimizeFlag++;
388 break;
Guido van Rossum7614da61997-03-03 19:14:45 +0000389
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000390 case 'B':
391 Py_DontWriteBytecodeFlag++;
392 break;
Christian Heimes790c8232008-01-07 21:14:23 +0000393
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000394 case 's':
395 Py_NoUserSiteDirectory++;
396 break;
Christian Heimes8dc226f2008-05-06 23:45:46 +0000397
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000398 case 'S':
399 Py_NoSiteFlag++;
400 break;
Guido van Rossum7922bd71997-08-29 22:34:47 +0000401
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000402 case 'E':
403 Py_IgnoreEnvironmentFlag++;
404 break;
Neil Schemenauer7d4bb9f2001-07-23 16:30:27 +0000405
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000406 case 't':
407 /* ignored for backwards compatibility */
408 break;
Guido van Rossumbba92ca1998-04-10 19:39:15 +0000409
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000410 case 'u':
411 Py_UnbufferedStdioFlag = 1;
412 saw_unbuffered_flag = 1;
413 break;
Guido van Rossum667d7041995-08-04 04:20:48 +0000414
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000415 case 'v':
416 Py_VerboseFlag++;
417 break;
Guido van Rossum667d7041995-08-04 04:20:48 +0000418
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000419 case 'x':
420 skipfirstline = 1;
421 break;
Guido van Rossuma075ce11997-12-05 21:56:45 +0000422
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000423 /* case 'X': reserved for implementation-specific arguments */
Christian Heimes33fe8092008-04-13 13:53:33 +0000424
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000425 case 'h':
426 case '?':
427 help++;
428 break;
Guido van Rossum45aecf42006-03-15 04:58:47 +0000429
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000430 case 'V':
431 version++;
432 break;
Guido van Rossumc15a9a12000-05-01 17:54:33 +0000433
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000434 case 'W':
435 PySys_AddWarnOption(_PyOS_optarg);
436 break;
Guido van Rossum47f5fdc2000-12-15 22:00:54 +0000437
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000438 /* This space reserved for other options */
Guido van Rossum667d7041995-08-04 04:20:48 +0000439
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000440 default:
441 return usage(2, argv[0]);
442 /*NOTREACHED*/
Guido van Rossum667d7041995-08-04 04:20:48 +0000443
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000444 }
445 }
Guido van Rossum667d7041995-08-04 04:20:48 +0000446
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000447 if (help)
448 return usage(0, argv[0]);
Barry Warsaw3b2aedb2000-09-15 18:40:42 +0000449
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000450 if (version) {
451 fprintf(stderr, "Python %s\n", PY_VERSION);
452 return 0;
453 }
Barry Warsaw3b2aedb2000-09-15 18:40:42 +0000454
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000455 if (!Py_InspectFlag &&
456 (p = Py_GETENV("PYTHONINSPECT")) && *p != '\0')
457 Py_InspectFlag = 1;
458 if (!saw_unbuffered_flag &&
459 (p = Py_GETENV("PYTHONUNBUFFERED")) && *p != '\0')
460 Py_UnbufferedStdioFlag = 1;
Neil Schemenauer7d4bb9f2001-07-23 16:30:27 +0000461
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000462 if (!Py_NoUserSiteDirectory &&
463 (p = Py_GETENV("PYTHONNOUSERSITE")) && *p != '\0')
464 Py_NoUserSiteDirectory = 1;
Christian Heimes8dc226f2008-05-06 23:45:46 +0000465
Philip Jenveye53de3d2010-04-14 03:01:39 +0000466#ifdef MS_WINDOWS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000467 if (!Py_IgnoreEnvironmentFlag && (wp = _wgetenv(L"PYTHONWARNINGS")) &&
468 *wp != L'\0') {
469 wchar_t *buf, *warning;
Philip Jenvey0805ca32010-04-07 04:04:10 +0000470
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000471 buf = (wchar_t *)malloc((wcslen(wp) + 1) * sizeof(wchar_t));
472 if (buf == NULL)
473 Py_FatalError(
474 "not enough memory to copy PYTHONWARNINGS");
475 wcscpy(buf, wp);
476 for (warning = wcstok(buf, L",");
477 warning != NULL;
478 warning = wcstok(NULL, L",")) {
479 PySys_AddWarnOption(warning);
480 }
481 free(buf);
482 }
Philip Jenveye53de3d2010-04-14 03:01:39 +0000483#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000484 if ((p = Py_GETENV("PYTHONWARNINGS")) && *p != '\0') {
485 char *buf, *oldloc;
Victor Stinner9ca9c252010-05-19 16:53:30 +0000486 PyObject *warning;
Philip Jenveye53de3d2010-04-14 03:01:39 +0000487
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000488 /* settle for strtok here as there's no one standard
489 C89 wcstok */
490 buf = (char *)malloc(strlen(p) + 1);
491 if (buf == NULL)
492 Py_FatalError(
493 "not enough memory to copy PYTHONWARNINGS");
494 strcpy(buf, p);
495 oldloc = strdup(setlocale(LC_ALL, NULL));
496 setlocale(LC_ALL, "");
497 for (p = strtok(buf, ","); p != NULL; p = strtok(NULL, ",")) {
Victor Stinner9ca9c252010-05-19 16:53:30 +0000498 warning = PyUnicode_DecodeFSDefault(p);
499 if (warning != NULL) {
500 PySys_AddWarnOptionUnicode(warning);
501 Py_DECREF(warning);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000502 }
503 }
504 setlocale(LC_ALL, oldloc);
505 free(oldloc);
506 free(buf);
507 }
Philip Jenveye53de3d2010-04-14 03:01:39 +0000508#endif
Philip Jenvey0805ca32010-04-07 04:04:10 +0000509
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000510 if (command == NULL && module == NULL && _PyOS_optind < argc &&
511 wcscmp(argv[_PyOS_optind], L"-") != 0)
512 {
Martin v. Löwis7a924e62003-03-05 14:15:21 +0000513#ifdef __VMS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000514 filename = decc$translate_vms(argv[_PyOS_optind]);
515 if (filename == (char *)0 || filename == (char *)-1)
516 filename = argv[_PyOS_optind];
Martin v. Löwis7a924e62003-03-05 14:15:21 +0000517
518#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000519 filename = argv[_PyOS_optind];
Martin v. Löwis7a924e62003-03-05 14:15:21 +0000520#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000521 }
Guido van Rossum775af911997-02-14 19:50:32 +0000522
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000523 stdin_is_interactive = Py_FdIsInteractive(stdin, (char *)0);
Guido van Rossum775af911997-02-14 19:50:32 +0000524
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000525 if (Py_UnbufferedStdioFlag) {
Sjoerd Mullender9cf424b2002-08-09 13:35:18 +0000526#if defined(MS_WINDOWS) || defined(__CYGWIN__)
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000527 _setmode(fileno(stdin), O_BINARY);
528 _setmode(fileno(stdout), O_BINARY);
Guido van Rossumf22d7e21997-01-11 19:28:55 +0000529#endif
Guido van Rossum22ffac11998-03-06 15:30:39 +0000530#ifdef HAVE_SETVBUF
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000531 setvbuf(stdin, (char *)NULL, _IONBF, BUFSIZ);
532 setvbuf(stdout, (char *)NULL, _IONBF, BUFSIZ);
533 setvbuf(stderr, (char *)NULL, _IONBF, BUFSIZ);
Guido van Rossum22ffac11998-03-06 15:30:39 +0000534#else /* !HAVE_SETVBUF */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000535 setbuf(stdin, (char *)NULL);
536 setbuf(stdout, (char *)NULL);
537 setbuf(stderr, (char *)NULL);
Guido van Rossum22ffac11998-03-06 15:30:39 +0000538#endif /* !HAVE_SETVBUF */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000539 }
540 else if (Py_InteractiveFlag) {
Guido van Rossumb31c7dc1997-04-11 22:19:12 +0000541#ifdef MS_WINDOWS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000542 /* Doesn't have to have line-buffered -- use unbuffered */
543 /* Any set[v]buf(stdin, ...) screws up Tkinter :-( */
544 setvbuf(stdout, (char *)NULL, _IONBF, BUFSIZ);
Guido van Rossum22ffac11998-03-06 15:30:39 +0000545#else /* !MS_WINDOWS */
546#ifdef HAVE_SETVBUF
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000547 setvbuf(stdin, (char *)NULL, _IOLBF, BUFSIZ);
548 setvbuf(stdout, (char *)NULL, _IOLBF, BUFSIZ);
Guido van Rossum22ffac11998-03-06 15:30:39 +0000549#endif /* HAVE_SETVBUF */
550#endif /* !MS_WINDOWS */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000551 /* Leave stderr alone - it should be unbuffered anyway. */
552 }
Martin v. Löwis79acb9e2002-12-06 12:48:53 +0000553#ifdef __VMS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000554 else {
555 setvbuf (stdout, (char *)NULL, _IOLBF, BUFSIZ);
556 }
Martin v. Löwis79acb9e2002-12-06 12:48:53 +0000557#endif /* __VMS */
Guido van Rossum667d7041995-08-04 04:20:48 +0000558
Just van Rossum2ac79ef2003-03-05 15:46:54 +0000559#ifdef __APPLE__
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000560 /* On MacOS X, when the Python interpreter is embedded in an
561 application bundle, it gets executed by a bootstrapping script
562 that does os.execve() with an argv[0] that's different from the
563 actual Python executable. This is needed to keep the Finder happy,
564 or rather, to work around Apple's overly strict requirements of
565 the process name. However, we still need a usable sys.executable,
566 so the actual executable path is passed in an environment variable.
567 See Lib/plat-mac/bundlebuiler.py for details about the bootstrap
568 script. */
569 if ((p = Py_GETENV("PYTHONEXECUTABLE")) && *p != '\0') {
570 wchar_t* buffer;
571 size_t len = strlen(p);
572 size_t r;
Ronald Oussoren3e264e12009-02-12 15:55:38 +0000573
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000574 buffer = malloc(len * sizeof(wchar_t));
575 if (buffer == NULL) {
576 Py_FatalError(
577 "not enough memory to copy PYTHONEXECUTABLE");
578 }
Ronald Oussoren3e264e12009-02-12 15:55:38 +0000579
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000580 r = mbstowcs(buffer, p, len);
581 Py_SetProgramName(buffer);
582 /* buffer is now handed off - do not free */
583 } else {
584 Py_SetProgramName(argv[0]);
585 }
Just van Rossum2ac79ef2003-03-05 15:46:54 +0000586#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000587 Py_SetProgramName(argv[0]);
Jack Jansenfbd861b2003-03-05 16:00:15 +0000588#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000589 Py_Initialize();
Guido van Rossumed52aac1997-07-19 19:20:32 +0000590
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000591 if (Py_VerboseFlag ||
592 (command == NULL && filename == NULL && module == NULL && stdin_is_interactive)) {
593 fprintf(stderr, "Python %s on %s\n",
594 Py_GetVersion(), Py_GetPlatform());
595 if (!Py_NoSiteFlag)
596 fprintf(stderr, "%s\n", COPYRIGHT);
597 }
Guido van Rossum393661d2001-08-31 17:40:15 +0000598
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000599 if (command != NULL) {
600 /* Backup _PyOS_optind and force sys.argv[0] = '-c' */
601 _PyOS_optind--;
602 argv[_PyOS_optind] = L"-c";
603 }
Guido van Rossum667d7041995-08-04 04:20:48 +0000604
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000605 if (module != NULL) {
606 /* Backup _PyOS_optind and force sys.argv[0] = '-c'
607 so that PySys_SetArgv correctly sets sys.path[0] to ''*/
608 _PyOS_optind--;
609 argv[_PyOS_optind] = L"-c";
610 }
Raymond Hettingerdb29e0f2004-10-07 06:46:25 +0000611
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000612 PySys_SetArgv(argc-_PyOS_optind, argv+_PyOS_optind);
Guido van Rossum667d7041995-08-04 04:20:48 +0000613
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000614 if ((Py_InspectFlag || (command == NULL && filename == NULL && module == NULL)) &&
615 isatty(fileno(stdin))) {
616 PyObject *v;
617 v = PyImport_ImportModule("readline");
618 if (v == NULL)
619 PyErr_Clear();
620 else
621 Py_DECREF(v);
622 }
Guido van Rossum3d26cc91997-09-16 16:11:28 +0000623
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000624 if (command) {
Victor Stinnera62207c2010-08-07 10:57:17 +0000625 sts = run_command(command, &cf);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000626 free(command);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000627 } else if (module) {
628 sts = RunModule(module, 1);
629 }
630 else {
Christian Heimes9cd17752007-11-18 19:35:23 +0000631
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000632 if (filename == NULL && stdin_is_interactive) {
633 Py_InspectFlag = 0; /* do exit on SystemExit */
634 RunStartupFile(&cf);
635 }
636 /* XXX */
Christian Heimes9cd17752007-11-18 19:35:23 +0000637
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000638 sts = -1; /* keep track of whether we've already run __main__ */
Christian Heimes9cd17752007-11-18 19:35:23 +0000639
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000640 if (filename != NULL) {
641 sts = RunMainFromImporter(filename);
642 }
Christian Heimes9cd17752007-11-18 19:35:23 +0000643
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000644 if (sts==-1 && filename!=NULL) {
Antoine Pitroue9b428f2010-08-13 22:25:01 +0000645 if ((fp = _Py_wfopen(filename, L"r")) == NULL) {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000646 char cfilename[PATH_MAX];
647 size_t r = wcstombs(cfilename, filename, PATH_MAX);
648 if (r == PATH_MAX)
649 /* cfilename is not null-terminated;
650 * forcefully null-terminating it
651 * might break the shift state */
652 strcpy(cfilename, "<file name too long>");
653 if (r == ((size_t)-1))
654 strcpy(cfilename, "<unprintable file name>");
655 fprintf(stderr, "%ls: can't open file '%s': [Errno %d] %s\n",
656 argv[0], cfilename, errno, strerror(errno));
Christian Heimesada8c3b2008-03-18 18:26:33 +0000657
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000658 return 2;
659 }
660 else if (skipfirstline) {
661 int ch;
662 /* Push back first newline so line numbers
663 remain the same */
664 while ((ch = getc(fp)) != EOF) {
665 if (ch == '\n') {
666 (void)ungetc(ch, fp);
667 break;
668 }
669 }
670 }
671 {
672 /* XXX: does this work on Win/Win64? (see posix_fstat) */
673 struct stat sb;
674 if (fstat(fileno(fp), &sb) == 0 &&
675 S_ISDIR(sb.st_mode)) {
676 fprintf(stderr, "%ls: '%ls' is a directory, cannot continue\n", argv[0], filename);
677 fclose(fp);
678 return 1;
679 }
680 }
681 }
Christian Heimes9cd17752007-11-18 19:35:23 +0000682
Victor Stinner0a3ddad2010-08-07 16:34:25 +0000683 if (sts == -1)
684 sts = run_file(fp, filename, &cf);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000685 }
Barry Warsawd86dcd32003-06-29 17:07:06 +0000686
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000687 /* Check this environment variable at the end, to give programs the
688 * opportunity to set it from Python.
689 */
690 if (!Py_InspectFlag &&
691 (p = Py_GETENV("PYTHONINSPECT")) && *p != '\0')
692 {
693 Py_InspectFlag = 1;
694 }
Guido van Rossum667d7041995-08-04 04:20:48 +0000695
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000696 if (Py_InspectFlag && stdin_is_interactive &&
697 (filename != NULL || command != NULL || module != NULL)) {
698 Py_InspectFlag = 0;
699 /* XXX */
700 sts = PyRun_AnyFileFlags(stdin, "<stdin>", &cf) != 0;
701 }
702
703 Py_Finalize();
Barry Warsaw3e13b1e2001-02-23 16:46:39 +0000704
705#ifdef __INSURE__
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000706 /* Insure++ is a memory analysis tool that aids in discovering
707 * memory leaks and other memory problems. On Python exit, the
708 * interned string dictionaries are flagged as being in use at exit
709 * (which it is). Under normal circumstances, this is fine because
710 * the memory will be automatically reclaimed by the system. Under
711 * memory debugging, it's a huge source of useless noise, so we
712 * trade off slower shutdown for less distraction in the memory
713 * reports. -baw
714 */
715 _Py_ReleaseInternedStrings();
716 _Py_ReleaseInternedUnicodeStrings();
Barry Warsaw3e13b1e2001-02-23 16:46:39 +0000717#endif /* __INSURE__ */
718
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000719 return sts;
Guido van Rossum667d7041995-08-04 04:20:48 +0000720}
721
Skip Montanaro786ea6b2004-03-01 15:44:05 +0000722/* this is gonna seem *real weird*, but if you put some other code between
723 Py_Main() and Py_GetArgcArgv() you will need to adjust the test in the
724 while statement in Misc/gdbinit:ppystack */
Guido van Rossum667d7041995-08-04 04:20:48 +0000725
Guido van Rossum667d7041995-08-04 04:20:48 +0000726/* Make the *original* argc/argv available to other modules.
727 This is rare, but it is needed by the secureware extension. */
728
729void
Martin v. Löwis790465f2008-04-05 20:41:37 +0000730Py_GetArgcArgv(int *argc, wchar_t ***argv)
Guido van Rossum667d7041995-08-04 04:20:48 +0000731{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000732 *argc = orig_argc;
733 *argv = orig_argv;
Guido van Rossum667d7041995-08-04 04:20:48 +0000734}
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000735
Ronald Oussorend61deca2010-04-18 14:46:12 +0000736
737wchar_t*
738_Py_char2wchar(char* arg)
739{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000740 wchar_t *res;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000741#ifdef HAVE_BROKEN_MBSTOWCS
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000742 /* Some platforms have a broken implementation of
743 * mbstowcs which does not count the characters that
744 * would result from conversion. Use an upper bound.
745 */
746 size_t argsize = strlen(arg);
Ronald Oussorend61deca2010-04-18 14:46:12 +0000747#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000748 size_t argsize = mbstowcs(NULL, arg, 0);
Ronald Oussorend61deca2010-04-18 14:46:12 +0000749#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000750 size_t count;
751 unsigned char *in;
752 wchar_t *out;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000753#ifdef HAVE_MBRTOWC
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000754 mbstate_t mbs;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000755#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000756 if (argsize != (size_t)-1) {
757 res = (wchar_t *)PyMem_Malloc((argsize+1)*sizeof(wchar_t));
758 if (!res)
759 goto oom;
760 count = mbstowcs(res, arg, argsize+1);
761 if (count != (size_t)-1) {
762 wchar_t *tmp;
763 /* Only use the result if it contains no
764 surrogate characters. */
765 for (tmp = res; *tmp != 0 &&
766 (*tmp < 0xd800 || *tmp > 0xdfff); tmp++)
767 ;
768 if (*tmp == 0)
769 return res;
770 }
771 PyMem_Free(res);
772 }
773 /* Conversion failed. Fall back to escaping with surrogateescape. */
Ronald Oussorend61deca2010-04-18 14:46:12 +0000774#ifdef HAVE_MBRTOWC
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000775 /* Try conversion with mbrtwoc (C99), and escape non-decodable bytes. */
776
777 /* Overallocate; as multi-byte characters are in the argument, the
778 actual output could use less memory. */
779 argsize = strlen(arg) + 1;
780 res = (wchar_t*)PyMem_Malloc(argsize*sizeof(wchar_t));
781 if (!res) goto oom;
782 in = (unsigned char*)arg;
783 out = res;
784 memset(&mbs, 0, sizeof mbs);
785 while (argsize) {
786 size_t converted = mbrtowc(out, (char*)in, argsize, &mbs);
787 if (converted == 0)
788 /* Reached end of string; null char stored. */
789 break;
790 if (converted == (size_t)-2) {
791 /* Incomplete character. This should never happen,
792 since we provide everything that we have -
793 unless there is a bug in the C library, or I
794 misunderstood how mbrtowc works. */
795 fprintf(stderr, "unexpected mbrtowc result -2\n");
796 return NULL;
797 }
798 if (converted == (size_t)-1) {
799 /* Conversion error. Escape as UTF-8b, and start over
800 in the initial shift state. */
801 *out++ = 0xdc00 + *in++;
802 argsize--;
803 memset(&mbs, 0, sizeof mbs);
804 continue;
805 }
806 if (*out >= 0xd800 && *out <= 0xdfff) {
807 /* Surrogate character. Escape the original
808 byte sequence with surrogateescape. */
809 argsize -= converted;
810 while (converted--)
811 *out++ = 0xdc00 + *in++;
812 continue;
813 }
814 /* successfully converted some bytes */
815 in += converted;
816 argsize -= converted;
817 out++;
818 }
Ronald Oussorend61deca2010-04-18 14:46:12 +0000819#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000820 /* Cannot use C locale for escaping; manually escape as if charset
821 is ASCII (i.e. escape all bytes > 128. This will still roundtrip
822 correctly in the locale's charset, which must be an ASCII superset. */
823 res = PyMem_Malloc((strlen(arg)+1)*sizeof(wchar_t));
824 if (!res) goto oom;
825 in = (unsigned char*)arg;
826 out = res;
827 while(*in)
828 if(*in < 128)
829 *out++ = *in++;
830 else
831 *out++ = 0xdc00 + *in++;
832 *out = 0;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000833#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000834 return res;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000835oom:
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000836 fprintf(stderr, "out of memory\n");
837 return NULL;
Ronald Oussorend61deca2010-04-18 14:46:12 +0000838}
839
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000840#ifdef __cplusplus
841}
842#endif