mirror of
https://github.com/python/cpython.git
synced 2024-12-19 06:44:47 +08:00
697842f58c
parameter for the return string (as unix pathnames are not limited by the 255 char pstring limit). Implemented the function for MachO-Python, where it returns unix pathnames.
267 lines
6.7 KiB
C
267 lines
6.7 KiB
C
/***********************************************************
|
|
Copyright 1991-1997 by Stichting Mathematisch Centrum, Amsterdam,
|
|
The Netherlands.
|
|
|
|
All Rights Reserved
|
|
|
|
Permission to use, copy, modify, and distribute this software and its
|
|
documentation for any purpose and without fee is hereby granted,
|
|
provided that the above copyright notice appear in all copies and that
|
|
both that copyright notice and this permission notice appear in
|
|
supporting documentation, and that the names of Stichting Mathematisch
|
|
Centrum or CWI not be used in advertising or publicity pertaining to
|
|
distribution of the software without specific, written prior permission.
|
|
|
|
STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO
|
|
THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE
|
|
FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
|
|
OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
******************************************************************/
|
|
|
|
/* Construct argc and argv for main() by using Apple Events */
|
|
/* From Jack's implementation for STDWIN */
|
|
|
|
#include <stdlib.h>
|
|
|
|
#ifdef WITHOUT_FRAMEWORKS
|
|
#include <Types.h>
|
|
#include <Files.h>
|
|
#include <Events.h>
|
|
#include <Memory.h>
|
|
#include <Processes.h>
|
|
#include <Errors.h>
|
|
#include <AppleEvents.h>
|
|
#include <AEObjects.h>
|
|
#include <Fonts.h>
|
|
#include <TextEdit.h>
|
|
#include <Menus.h>
|
|
#include <Dialogs.h>
|
|
#include <Windows.h>
|
|
#else
|
|
#include <Carbon/Carbon.h>
|
|
#endif /* WITHOUT_FRAMEWORKS */
|
|
|
|
#if UNIVERSAL_INTERFACES_VERSION >= 0x0340
|
|
typedef long refcontype;
|
|
#else
|
|
typedef unsigned long refcontype;
|
|
#endif
|
|
|
|
#include "Python.h"
|
|
#include "macglue.h"
|
|
|
|
#ifdef TARGET_API_MAC_OSX
|
|
#define PATHNAMELEN 1024
|
|
#else
|
|
#define PATHNAMELEN 256
|
|
#endif
|
|
|
|
static int arg_count;
|
|
static char *arg_vector[256];
|
|
FSSpec PyMac_ApplicationFSSpec;
|
|
char PyMac_ApplicationPath[PATHNAMELEN];
|
|
|
|
/* Duplicate a string to the heap. We also export this since it isn't standard
|
|
** and others use it
|
|
*/
|
|
#ifndef HAVE_STRDUP
|
|
char *
|
|
strdup(const char *src)
|
|
{
|
|
char *dst = malloc(strlen(src) + 1);
|
|
if (dst)
|
|
strcpy(dst, src);
|
|
return dst;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if !TARGET_API_MAC_OSX
|
|
/* Initialize FSSpec and full name of current application */
|
|
|
|
OSErr
|
|
PyMac_init_process_location(void)
|
|
{
|
|
ProcessSerialNumber currentPSN;
|
|
ProcessInfoRec info;
|
|
OSErr err;
|
|
static int applocation_inited;
|
|
|
|
if ( applocation_inited ) return 0;
|
|
currentPSN.highLongOfPSN = 0;
|
|
currentPSN.lowLongOfPSN = kCurrentProcess;
|
|
info.processInfoLength = sizeof(ProcessInfoRec);
|
|
info.processName = NULL;
|
|
info.processAppSpec = &PyMac_ApplicationFSSpec;
|
|
if ( err=GetProcessInformation(¤tPSN, &info))
|
|
return err;
|
|
if ( err=PyMac_GetFullPathname(&PyMac_ApplicationFSSpec, PyMac_ApplicationPath, PATHNAMELEN) )
|
|
return err;
|
|
applocation_inited = 1;
|
|
return 0;
|
|
}
|
|
#endif /* !TARGET_API_MAC_OSX */
|
|
|
|
/* Check that there aren't any args remaining in the event */
|
|
|
|
static OSErr
|
|
get_missing_params(const AppleEvent *theAppleEvent)
|
|
{
|
|
DescType theType;
|
|
Size actualSize;
|
|
OSErr err;
|
|
|
|
err = AEGetAttributePtr(theAppleEvent, keyMissedKeywordAttr, typeWildCard,
|
|
&theType, nil, 0, &actualSize);
|
|
if (err == errAEDescNotFound)
|
|
return noErr;
|
|
else
|
|
return errAEEventNotHandled;
|
|
}
|
|
|
|
static int got_one; /* Flag that we can stop getting events */
|
|
|
|
/* Handle the Print or Quit events (by failing) */
|
|
|
|
static pascal OSErr
|
|
handle_not(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon)
|
|
{
|
|
#pragma unused (reply, refCon)
|
|
got_one = 1;
|
|
return errAEEventNotHandled;
|
|
}
|
|
|
|
/* Handle the Open Application event (by ignoring it) */
|
|
|
|
static pascal OSErr
|
|
handle_open_app(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon)
|
|
{
|
|
#pragma unused (reply, refCon)
|
|
#if 0
|
|
/* Test by Jack: would removing this facilitate debugging? */
|
|
got_one = 1;
|
|
#endif
|
|
return get_missing_params(theAppleEvent);
|
|
}
|
|
|
|
/* Handle the Open Document event, by adding an argument */
|
|
|
|
static pascal OSErr
|
|
handle_open_doc(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon)
|
|
{
|
|
#pragma unused (reply, refCon)
|
|
OSErr err;
|
|
AEDescList doclist;
|
|
AEKeyword keywd;
|
|
DescType rttype;
|
|
long i, ndocs, size;
|
|
FSSpec fss;
|
|
char path[PATHNAMELEN];
|
|
|
|
got_one = 1;
|
|
if ((err = AEGetParamDesc(theAppleEvent,
|
|
keyDirectObject, typeAEList, &doclist)))
|
|
return err;
|
|
if ((err = get_missing_params(theAppleEvent)))
|
|
return err;
|
|
if ((err = AECountItems(&doclist, &ndocs)))
|
|
return err;
|
|
for(i = 1; i <= ndocs; i++) {
|
|
err = AEGetNthPtr(&doclist, i, typeFSS,
|
|
&keywd, &rttype, &fss, sizeof(fss), &size);
|
|
if (err)
|
|
break;
|
|
PyMac_GetFullPathname(&fss, path, PATHNAMELEN);
|
|
arg_vector[arg_count++] = strdup(path);
|
|
}
|
|
return err;
|
|
}
|
|
|
|
/* Install standard core event handlers */
|
|
AEEventHandlerUPP open_doc_upp;
|
|
AEEventHandlerUPP open_app_upp;
|
|
AEEventHandlerUPP not_upp;
|
|
|
|
static void
|
|
set_ae_handlers(void)
|
|
{
|
|
open_doc_upp = NewAEEventHandlerUPP(&handle_open_doc);
|
|
open_app_upp = NewAEEventHandlerUPP(&handle_open_app);
|
|
not_upp = NewAEEventHandlerUPP(&handle_not);
|
|
|
|
AEInstallEventHandler(kCoreEventClass, kAEOpenApplication,
|
|
open_app_upp, 0L, false);
|
|
AEInstallEventHandler(kCoreEventClass, kAEOpenDocuments,
|
|
open_doc_upp, 0L, false);
|
|
AEInstallEventHandler(kCoreEventClass, kAEPrintDocuments,
|
|
not_upp, 0L, false);
|
|
AEInstallEventHandler(kCoreEventClass, kAEQuitApplication,
|
|
not_upp, 0L, false);
|
|
}
|
|
|
|
/* Uninstall standard core event handlers */
|
|
|
|
static void
|
|
reset_ae_handlers(void)
|
|
{
|
|
AERemoveEventHandler(kCoreEventClass, kAEOpenApplication,
|
|
open_app_upp, false);
|
|
AERemoveEventHandler(kCoreEventClass, kAEOpenDocuments,
|
|
open_doc_upp, false);
|
|
AERemoveEventHandler(kCoreEventClass, kAEPrintDocuments,
|
|
not_upp, false);
|
|
AERemoveEventHandler(kCoreEventClass, kAEQuitApplication,
|
|
not_upp, false);
|
|
}
|
|
|
|
/* Wait for events until a core event has been handled */
|
|
|
|
static void
|
|
event_loop(void)
|
|
{
|
|
EventRecord event;
|
|
int n;
|
|
int ok;
|
|
|
|
got_one = 0;
|
|
for (n = 0; n < 100 && !got_one; n++) {
|
|
#if !TARGET_API_MAC_CARBON
|
|
SystemTask();
|
|
#endif
|
|
ok = GetNextEvent(everyEvent, &event);
|
|
if (ok && event.what == kHighLevelEvent) {
|
|
AEProcessAppleEvent(&event);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Get the argv vector, return argc */
|
|
|
|
int
|
|
PyMac_GetArgv(char ***pargv, int noevents)
|
|
{
|
|
arg_count = 0;
|
|
#if TARGET_API_MAC_OSX
|
|
/* In an OSX bundle argv[0] is okay */
|
|
arg_count++;
|
|
#else
|
|
(void)PyMac_init_process_location();
|
|
arg_vector[arg_count++] = strdup(PyMac_ApplicationPath);
|
|
#endif /* TARGET_API_MAC_OSX */
|
|
|
|
if( !noevents ) {
|
|
set_ae_handlers();
|
|
event_loop();
|
|
reset_ae_handlers();
|
|
}
|
|
|
|
arg_vector[arg_count] = NULL;
|
|
|
|
*pargv = arg_vector;
|
|
return arg_count;
|
|
}
|