blob: 9e8fbc57b8287505f3542fb07198a2bbe47837cb [file] [log] [blame]
Guido van Rossumb0f3c821994-08-23 13:34:25 +00001/***********************************************************
Jack Jansen42218ce1997-01-31 16:15:11 +00002Copyright 1991-1997 by Stichting Mathematisch Centrum, Amsterdam,
Guido van Rossum99546991995-01-08 14:33:34 +00003The Netherlands.
Guido van Rossumb0f3c821994-08-23 13:34:25 +00004
5 All Rights Reserved
6
7Permission to use, copy, modify, and distribute this software and its
8documentation for any purpose and without fee is hereby granted,
9provided that the above copyright notice appear in all copies and that
10both that copyright notice and this permission notice appear in
11supporting documentation, and that the names of Stichting Mathematisch
12Centrum or CWI not be used in advertising or publicity pertaining to
13distribution of the software without specific, written prior permission.
14
15STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO
16THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
17FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE
18FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
19WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
20ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
21OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
22
23******************************************************************/
24
Jack Jansen696c9581995-08-14 12:33:20 +000025/* Python interpreter main program */
Guido van Rossumb0f3c821994-08-23 13:34:25 +000026
Jack Jansen696c9581995-08-14 12:33:20 +000027#include "Python.h"
28#include "pythonresources.h"
29#include "import.h"
30#include "marshal.h"
Jack Jansenf6865f71996-09-04 15:24:59 +000031#include "macglue.h"
Guido van Rossumb0f3c821994-08-23 13:34:25 +000032
Jack Jansen696c9581995-08-14 12:33:20 +000033#include <Memory.h>
34#include <Resources.h>
Guido van Rossumb0f3c821994-08-23 13:34:25 +000035#include <stdio.h>
Jack Jansen696c9581995-08-14 12:33:20 +000036#include <Events.h>
37#include <Windows.h>
Jack Jansen2429c721996-03-07 15:17:11 +000038#include <Fonts.h>
Guido van Rossumb0f3c821994-08-23 13:34:25 +000039
Jack Jansenc76fd391995-02-02 14:27:31 +000040#ifdef __MWERKS__
41#include <SIOUX.h>
Jack Jansen1e8557a1995-11-10 14:51:26 +000042#define USE_SIOUX
Jack Jansen8c693211997-01-07 16:19:42 +000043#if __profile__ == 1
44#include <profiler.h>
45#endif
Jack Jansenc76fd391995-02-02 14:27:31 +000046#endif
47
Jack Jansen0168f271995-10-27 13:32:30 +000048#ifdef THINK_C
49#include <console.h>
50#endif
51
Jack Jansen696c9581995-08-14 12:33:20 +000052#define STARTUP "PythonStartup"
Jack Jansenbac428d1994-12-14 13:47:30 +000053
Jack Jansen696c9581995-08-14 12:33:20 +000054extern int Py_DebugFlag; /* For parser.c, declared in pythonrun.c */
55extern int Py_VerboseFlag; /* For import.c, declared in pythonrun.c */
56extern int Py_SuppressPrintingFlag; /* For ceval.c, declared in pythonrun.c */
Jack Jansen3f7d2b41996-09-06 22:21:07 +000057short PyMac_AppRefNum; /* RefNum of application resource fork */
Jack Jansen696c9581995-08-14 12:33:20 +000058
59
60/* Subroutines that live in their own file */
Jack Jansena547dca1996-07-10 15:48:25 +000061extern char *Py_GetVersion Py_PROTO((void));
62extern char *Py_GetCopyright Py_PROTO((void));
Jack Jansen696c9581995-08-14 12:33:20 +000063
64
Jack Jansen1d2f8631996-08-02 15:16:16 +000065/* For Py_GetArgcArgv(); set by main() */
Jack Jansen696c9581995-08-14 12:33:20 +000066static char **orig_argv;
67static int orig_argc;
68
Jack Jansen7d5f9e81996-09-07 17:09:31 +000069PyMac_PrefRecord options;
Jack Jansen0168f271995-10-27 13:32:30 +000070
Jack Jansen76ceece1996-08-19 11:18:24 +000071static void Py_Main Py_PROTO((int, char **)); /* Forward */
72void PyMac_Exit Py_PROTO((int)); /* Forward */
73
Jack Jansen01fbc681996-02-28 15:42:47 +000074/* Initialize the Mac toolbox world */
75
76static void
77init_mac_world()
78{
79#ifdef THINK_C
80 printf("\n");
81#else
82 MaxApplZone();
83 InitGraf(&qd.thePort);
84 InitFonts();
85 InitWindows();
86 TEInit();
87 InitDialogs((long)0);
88 InitMenus();
89 InitCursor();
90#endif
91}
92
Jack Jansen7d5f9e81996-09-07 17:09:31 +000093/*
94** PyMac_InteractiveOptions - Allow user to set options if option key is pressed
95*/
Jack Jansen01fbc681996-02-28 15:42:47 +000096static void
Jack Jansen7d5f9e81996-09-07 17:09:31 +000097PyMac_InteractiveOptions(PyMac_PrefRecord *p, int *argcp, char ***argvp)
98{
99 KeyMap rmap;
100 unsigned char *map;
101 short item, type;
102 ControlHandle handle;
103 DialogPtr dialog;
104 Rect rect;
105 int old_argc = *argcp;
106 int i;
107
108 /*
109 ** If the preferences disallows interactive options we return,
110 ** similarly of <option> isn't pressed.
111 */
112 if (p->nointopt) return;
113
114 GetKeys(rmap);
115 map = (unsigned char *)rmap;
116 if ( ( map[0x3a>>3] & (1<<(0x3a&7)) ) == 0 ) /* option key is 3a */
117 return;
118
119 dialog = GetNewDialog(OPT_DIALOG, NULL, (WindowPtr)-1);
120 if ( dialog == NULL ) {
121 printf("Option dialog not found - cannot set options\n");
122 return;
123 }
124 SetDialogDefaultItem(dialog, OPT_OK);
125 SetDialogCancelItem(dialog, OPT_CANCEL);
126
127 /* Set default values */
128#define SET_OPT_ITEM(num, var) \
129 GetDialogItem(dialog, (num), &type, (Handle *)&handle, &rect); \
Jack Jansen08c3be31997-04-08 15:27:00 +0000130 SetControlValue(handle, (short)p->var);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000131
132 SET_OPT_ITEM(OPT_INSPECT, inspect);
133 SET_OPT_ITEM(OPT_VERBOSE, verbose);
134 SET_OPT_ITEM(OPT_SUPPRESS, suppress_print);
135 SET_OPT_ITEM(OPT_UNBUFFERED, unbuffered);
136 SET_OPT_ITEM(OPT_DEBUGGING, debugging);
137 SET_OPT_ITEM(OPT_KEEPNORMAL, keep_normal);
138 SET_OPT_ITEM(OPT_KEEPERROR, keep_error);
139 /* The rest are not settable interactively */
140
141#undef SET_OPT_ITEM
142
143 while (1) {
144 handle = NULL;
145 ModalDialog(NULL, &item);
146 if ( item == OPT_OK )
147 break;
148 if ( item == OPT_CANCEL ) {
Jack Jansen08c3be31997-04-08 15:27:00 +0000149 DisposeDialog(dialog);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000150 exit(0);
151 }
152 if ( item == OPT_CMDLINE ) {
153 int new_argc, newer_argc;
154 char **new_argv, **newer_argv;
155
156 new_argc = ccommand(&new_argv);
157 newer_argc = (new_argc-1) + old_argc;
158 newer_argv = malloc((newer_argc+1)*sizeof(char *));
159 if( !newer_argv )
160 Py_FatalError("Cannot malloc argv\n");
161 for(i=0; i<old_argc; i++)
162 newer_argv[i] = (*argvp)[i];
163 for(i=old_argc; i<=newer_argc; i++) /* Copy the NULL too */
164 newer_argv[i] = new_argv[i-old_argc+1];
165 *argvp = newer_argv;
166 *argcp = newer_argc;
167
168 /* XXXX Is it not safe to use free() here, apparently */
169 }
170#define OPT_ITEM(num, var) \
171 if ( item == (num) ) { \
172 p->var = !p->var; \
173 GetDialogItem(dialog, (num), &type, (Handle *)&handle, &rect); \
Jack Jansen08c3be31997-04-08 15:27:00 +0000174 SetControlValue(handle, (short)p->var); \
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000175 }
176
177 OPT_ITEM(OPT_INSPECT, inspect);
178 OPT_ITEM(OPT_VERBOSE, verbose);
179 OPT_ITEM(OPT_SUPPRESS, suppress_print);
180 OPT_ITEM(OPT_UNBUFFERED, unbuffered);
181 OPT_ITEM(OPT_DEBUGGING, debugging);
182 OPT_ITEM(OPT_KEEPNORMAL, keep_normal);
183 OPT_ITEM(OPT_KEEPERROR, keep_error);
184
185#undef OPT_ITEM
186 }
Jack Jansen08c3be31997-04-08 15:27:00 +0000187 DisposeDialog(dialog);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000188}
189
190/*
191** Initialization code, shared by interpreter and applets
192*/
193static void
Jack Jansen52ac0371997-01-15 15:49:08 +0000194init_common(int *argcp, char ***argvp, int embedded)
Jack Jansen01fbc681996-02-28 15:42:47 +0000195{
Jack Jansen3f7d2b41996-09-06 22:21:07 +0000196 /* Remember resource fork refnum, for later */
197 PyMac_AppRefNum = CurResFile();
198
Jack Jansen01fbc681996-02-28 15:42:47 +0000199 /* Initialize toolboxes */
200 init_mac_world();
201
202#ifdef USE_MAC_SHARED_LIBRARY
203 /* Add the shared library to the stack of resource files */
204 PyMac_AddLibResources();
205#endif
206
207#if defined(USE_GUSI)
208 /* Setup GUSI */
209 GUSIDefaultSetup();
Jack Jansenf6865f71996-09-04 15:24:59 +0000210 PyMac_SetGUSISpin();
Jack Jansen3f7d2b41996-09-06 22:21:07 +0000211 PyMac_SetGUSIOptions();
Jack Jansen01fbc681996-02-28 15:42:47 +0000212#endif
213
214#ifdef USE_SIOUX
215 /* Set various SIOUX flags. Some are changed later based on options */
216 SIOUXSettings.asktosaveonclose = 0;
217 SIOUXSettings.showstatusline = 0;
218 SIOUXSettings.tabspaces = 4;
219#endif
220
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000221 /* Get options from preference file (or from applet resource fork) */
222 options.keep_error = 1; /* default-default */
223 PyMac_PreferenceOptions(&options);
224
Jack Jansen52ac0371997-01-15 15:49:08 +0000225 if ( embedded ) {
226 static char *emb_argv[] = {"embedded-python", 0};
227
228 *argcp = 1;
229 *argvp = emb_argv;
230 } else {
231 /* Create argc/argv. Do it before we go into the options event loop. */
232 *argcp = PyMac_GetArgv(argvp, options.noargs);
233
234 /* Do interactive option setting, if allowed and <option> depressed */
235 PyMac_InteractiveOptions(&options, argcp, argvp);
236 }
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000237
238 /* Copy selected options to where the machine-independent stuff wants it */
239 Py_VerboseFlag = options.verbose;
240 Py_SuppressPrintingFlag = options.suppress_print;
241 Py_DebugFlag = options.debugging;
242 if ( options.noargs )
243 PyMac_DoYieldEnabled = 0;
244
245 /* Set buffering */
246 if (options.unbuffered) {
247#ifndef MPW
248 setbuf(stdout, (char *)NULL);
249 setbuf(stderr, (char *)NULL);
250#else
251 /* On MPW (3.2) unbuffered seems to hang */
252 setvbuf(stdout, (char *)NULL, _IOLBF, BUFSIZ);
253 setvbuf(stderr, (char *)NULL, _IOLBF, BUFSIZ);
254#endif
255 }
Jack Jansen8c693211997-01-07 16:19:42 +0000256#if __profile__ == 1
257 /* collectSummary or collectDetailed, timebase, #routines, max stack depth */
258 ProfilerInit(collectSummary, bestTimeBase, 2000, 150);
259#endif
Jack Jansen01fbc681996-02-28 15:42:47 +0000260}
261
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000262/*
263** Inspection mode after script/applet termination
264*/
265static int
266run_inspect()
267{
268 int sts = 0;
269
270 if (options.inspect && isatty((int)fileno(stdin)))
271 sts = PyRun_AnyFile(stdin, "<stdin>") != 0;
272 return sts;
273}
Jack Jansen01fbc681996-02-28 15:42:47 +0000274
Jack Jansen696c9581995-08-14 12:33:20 +0000275#ifdef USE_MAC_APPLET_SUPPORT
276/* Applet support */
277
278/* Run a compiled Python Python script from 'PYC ' resource __main__ */
279static int
280run_main_resource()
281{
282 Handle h;
283 long size;
284 PyObject *code;
285 PyObject *result;
286
287 h = GetNamedResource('PYC ', "\p__main__");
288 if (h == NULL) {
289 Alert(NOPYC_ALERT, NULL);
290 return 1;
291 }
292 size = GetResourceSizeOnDisk(h);
293 HLock(h);
294 code = PyMarshal_ReadObjectFromString(*h + 8, (int)(size - 8));
295 HUnlock(h);
296 ReleaseResource(h);
297 if (code == NULL) {
298 PyErr_Print();
299 return 1;
300 }
301 result = PyImport_ExecCodeModule("__main__", code);
302 Py_DECREF(code);
303 if (result == NULL) {
304 PyErr_Print();
305 return 1;
306 }
307 Py_DECREF(result);
308 return 0;
309}
310
311/* Initialization sequence for applets */
312void
313PyMac_InitApplet()
314{
Guido van Rossumb0f3c821994-08-23 13:34:25 +0000315 int argc;
316 char **argv;
Jack Jansen696c9581995-08-14 12:33:20 +0000317 int err;
318
Jack Jansen52ac0371997-01-15 15:49:08 +0000319 init_common(&argc, &argv, 0);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000320
Jack Jansen696c9581995-08-14 12:33:20 +0000321 Py_Initialize();
322 PySys_SetArgv(argc, argv);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000323
Jack Jansen696c9581995-08-14 12:33:20 +0000324 err = run_main_resource();
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000325
326 err = (run_inspect() || err);
327
Jack Jansen696c9581995-08-14 12:33:20 +0000328 fflush(stderr);
329 fflush(stdout);
Jack Jansen0168f271995-10-27 13:32:30 +0000330 PyMac_Exit(err);
Jack Jansen696c9581995-08-14 12:33:20 +0000331 /* XXX Should we bother to Py_Exit(sts)? */
332}
333
Jack Jansen52ac0371997-01-15 15:49:08 +0000334/*
335** Hook for embedding python.
336*/
337void
338PyMac_Initialize()
339{
340 int argc;
341 char **argv;
342
343 init_common(&argc, &argv, 1);
344 Py_Initialize();
345 PySys_SetArgv(argc, argv);
346}
347
Jack Jansen696c9581995-08-14 12:33:20 +0000348#endif /* USE_MAC_APPLET_SUPPORT */
349
350/* For normal application */
351void
352PyMac_InitApplication()
353{
354 int argc;
355 char **argv;
356
Jack Jansen52ac0371997-01-15 15:49:08 +0000357 init_common(&argc, &argv, 0);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000358
Jack Jansen696c9581995-08-14 12:33:20 +0000359 if ( argc > 1 ) {
360 /* We're running a script. Attempt to change current directory */
361 char curwd[256], *endp;
362
363 strcpy(curwd, argv[1]);
364 endp = strrchr(curwd, ':');
365 if ( endp && endp > curwd ) {
366 *endp = '\0';
367
368 chdir(curwd);
Jack Jansen378815c1996-03-06 16:21:34 +0000369#ifdef USE_GUSI
370 /* Change MacOS's idea of wd too */
371 PyMac_FixGUSIcd();
372#endif
Jack Jansen696c9581995-08-14 12:33:20 +0000373 }
374 }
375 Py_Main(argc, argv);
376}
377
Jack Jansen696c9581995-08-14 12:33:20 +0000378/* Main program */
379
Jack Jansen76ceece1996-08-19 11:18:24 +0000380static void
Jack Jansen696c9581995-08-14 12:33:20 +0000381Py_Main(argc, argv)
382 int argc;
383 char **argv;
384{
Jack Jansen696c9581995-08-14 12:33:20 +0000385 int sts;
386 char *command = NULL;
387 char *filename = NULL;
388 FILE *fp = stdin;
Jack Jansen696c9581995-08-14 12:33:20 +0000389
Jack Jansen1d2f8631996-08-02 15:16:16 +0000390 orig_argc = argc; /* For Py_GetArgcArgv() */
Jack Jansen696c9581995-08-14 12:33:20 +0000391 orig_argv = argv;
Jack Jansen696c9581995-08-14 12:33:20 +0000392 filename = argv[1];
393
394 if (Py_VerboseFlag ||
395 command == NULL && filename == NULL && isatty((int)fileno(fp)))
396 fprintf(stderr, "Python %s\n%s\n",
Jack Jansena547dca1996-07-10 15:48:25 +0000397 Py_GetVersion(), Py_GetCopyright());
Jack Jansen696c9581995-08-14 12:33:20 +0000398
399 if (filename != NULL) {
400 if ((fp = fopen(filename, "r")) == NULL) {
401 fprintf(stderr, "%s: can't open file '%s'\n",
402 argv[0], filename);
Jack Jansen0168f271995-10-27 13:32:30 +0000403 PyMac_Exit(2);
Jack Jansen696c9581995-08-14 12:33:20 +0000404 }
405 }
406
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000407 /* We initialize the menubar here, hoping SIOUX is initialized by now */
Jack Jansen3469e991996-09-06 00:30:45 +0000408 PyMac_InitMenuBar();
409
Jack Jansen696c9581995-08-14 12:33:20 +0000410 Py_Initialize();
411
412 PySys_SetArgv(argc-1, argv+1);
413
414 if (filename == NULL && isatty((int)fileno(fp))) {
415 FILE *fp = fopen(STARTUP, "r");
416 if (fp != NULL) {
417 (void) PyRun_SimpleFile(fp, STARTUP);
418 PyErr_Clear();
419 fclose(fp);
420 }
421 }
422 sts = PyRun_AnyFile(
423 fp, filename == NULL ? "<stdin>" : filename) != 0;
424 if (filename != NULL)
425 fclose(fp);
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000426
427 if ( filename != NULL || command != NULL )
428 sts = (run_inspect() || sts);
Jack Jansen696c9581995-08-14 12:33:20 +0000429
430 Py_Exit(sts);
431 /*NOTREACHED*/
432}
433
Jack Jansen0168f271995-10-27 13:32:30 +0000434/*
435** Terminate application
436*/
Jack Jansen76ceece1996-08-19 11:18:24 +0000437void
Jack Jansen0168f271995-10-27 13:32:30 +0000438PyMac_Exit(status)
439 int status;
440{
441 int keep;
Jack Jansen8c693211997-01-07 16:19:42 +0000442
443#if __profile__ == 1
444 ProfilerDump("\pPython Profiler Results");
445 ProfilerTerm();
446#endif
Jack Jansen0168f271995-10-27 13:32:30 +0000447 if ( status )
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000448 keep = options.keep_error;
Jack Jansen0168f271995-10-27 13:32:30 +0000449 else
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000450 keep = options.keep_normal;
Jack Jansen0168f271995-10-27 13:32:30 +0000451
Jack Jansen1e8557a1995-11-10 14:51:26 +0000452#ifdef USE_SIOUX
453 if (keep) {
454 SIOUXSettings.standalone = 1;
455 SIOUXSettings.autocloseonquit = 0;
Jack Jansen415571c1996-03-25 15:46:03 +0000456 SIOUXSetTitle("\p\307terminated\310");
Jack Jansene44545f1997-05-07 15:48:54 +0000457#ifdef USE_MSL
458 /*
459 ** Temporary workaround: autocloseonquit clearing does not
460 ** currently work for the MSL/GUSI combo.
461 */
462 while(getchar() > 0);
463#endif
Jack Jansen1e8557a1995-11-10 14:51:26 +0000464 }
Jack Jansen0168f271995-10-27 13:32:30 +0000465 else
466 SIOUXSettings.autocloseonquit = 1;
Jack Jansenf6865f71996-09-04 15:24:59 +0000467#ifdef USE_GUSI
468 /*
469 ** Workaround for Sioux/GUSI combo: we should not call
470 ** SiouxHandleOneEvent after the window is closed
471 */
472 PyMac_ConsoleIsDead = 1;
473#endif /* USE_GUSI */
474#endif /* USE_SIOUX */
Jack Jansen0168f271995-10-27 13:32:30 +0000475#ifdef THINK_C
476 console_options.pause_atexit = keep;
477#endif
478
479 exit(status);
480}
Jack Jansen696c9581995-08-14 12:33:20 +0000481
482/* Return the program name -- some code out there needs this. */
483
484char *
Jack Jansen1d2f8631996-08-02 15:16:16 +0000485Py_GetProgramName()
Jack Jansen696c9581995-08-14 12:33:20 +0000486{
Jack Jansen7d5f9e81996-09-07 17:09:31 +0000487 return orig_argv[0];
Jack Jansen696c9581995-08-14 12:33:20 +0000488}
489
490
491/* Make the *original* argc/argv available to other modules.
492 This is rare, but it is needed by the secureware extension. */
493
494void
Jack Jansen1d2f8631996-08-02 15:16:16 +0000495Py_GetArgcArgv(argc,argv)
Jack Jansen696c9581995-08-14 12:33:20 +0000496 int *argc;
497 char ***argv;
498{
499 *argc = orig_argc;
500 *argv = orig_argv;
Guido van Rossumb0f3c821994-08-23 13:34:25 +0000501}
Jack Jansen1d2f8631996-08-02 15:16:16 +0000502
503/* More cruft that shouldn't really be here, used in sysmodule.c */
504
505char *
506Py_GetPrefix()
507{
508 return "";
509}
510
511char *
512Py_GetExecPrefix()
513{
514 return "";
515}