mac-rom/ProcessMgr/Eppc.c
Elliot Nunn 0ba83392d4 Bring in CubeE sources
Resource forks are included only for .rsrc files. These are DeRezzed into their data fork. 'ckid' resources, from the Projector VCS, are not included.

The Tools directory, containing mostly junk, is also excluded.
2017-09-20 18:04:16 +08:00

2534 lines
79 KiB
C

/*
File: Eppc.c
Contains: Routines which implement Event Process-to-Process Communication (EPPC),
aka high-level events.
Written by: Jay Moreland
Copyright: © 1989-1992 by Apple Computer, Inc., all rights reserved.
Change History (most recent first):
<26> 5/29/92 DCL Included Script.h. GetEnvirons moved for the New Inside Mac.
<25> 11/25/91 DTY Add #include <MemoryMgrPriv.h> to get interface for MoveHLow in
its new home.
<24> 11/21/91 DTY Rolling in the 7¥Up version of findFreeSessionRecord that
allocates the session record low in the System Heap instead of
in the middle of the temporary heap. This is conditionalized for
CubeE only until a decision is made about whether this is really
the way we want to fix the problem.
<23> 3/29/91 JWM CCN,#85997: In GetSpecificHighLevelEvent, it was possible to
exit without restoring eppcBlk->mfMsgBlk to its original value.
This would cause event_common to call flushMsg for a HLE that
was still linked into the app's HLE Queue. As part of flushMsg
the memory would be returned to the free pool. This caused weird
memory roaching problems. The fix is a one line addition in
c_GetSpecificHighLevelEvent during the error path for the second
copyMsg.
<22> 3/26/91 DTY Having been thoroughly chastised by Jay for touching the source
code without him standing next to me, IÕm about to risk his
wrath again to make things the way they were in version <20>.
<21> 3/25/91 DTY JDR, #85618: The Gang of Five relegated this bug to Post 7.0, so
it needed to be rolled out. The other change, JWM-001, was left
in.
<20> 3/25/91 JWM jc,#JWM-001,#85618: ¥removing a DisposHandle from inside a loop
where interrupts are disabled. ¥removing the restriction on
starting a session when the application is being coerced.
<19> 3/18/91 JWM ewa,#85038, #85039: ¥The System crashes when
GetSpecificHighLevelEvent is called twice in a row with no
events pending and there is a partial HLEvent outstanding.
¥GetSpecificHighLevelEvent goes through the HighLevelEventQueue
twice if AcceptHighLevelEvent is called for the last event in
the HighLevelEventQueue from within the filter function.
<18> 2/26/91 JWM DFH,#83798: fix bug where ports are sometimes left open.
<17> 2/21/91 DFH dba,#82504, #82681, #83168, #83182, #83207: HiNewHandle rolled
into ProcessMgrHiNewHandle.
<16> 2/21/91 JWM ewa,#83492,#78885: ¥The System crashes when processing a return
receipt over a network for an application which has quit and is
no longer running. Fix is in scanReadBQ. ¥QuarkXPress 3.0 Ñ
Double clicking on the document opens the app, but not the
document. fix is in makePortname. ¥Also call PPC post processor
routines (scanReadBQ, scanWriteBQ and scanInformBQ) when
GetSpecificHighLevelEvent is called. This is for AppleEvent
manager.
<15> 2/9/91 JWM DFH,#79583,#81622,#82463: ¥Restart and Shutdown don't work in
low memory conditions. Routines getPPCRead, getPPCInformPB,
findFreeSessionRecord, GetMsgEventBuffer, fetchMsgData use to
call ProcessMgrHiNewHandle. ProcessMgrHiNewHandle just searched
the Process Managers heap. The routines now call GMBlk which
follows an unsucessful search of the Process Manager's heap with
a search of the System's heap. In addition,
AppleEventExtensions.c and Puppet.c were modified to use GMBlk
instead of just looking in the Process Manager's heap. GMBlk is
defined in Data.h and implemented in Memory.c. ¥Event PPC does
not register the correct port name with the correct name script
for foriegn systems. The PPCPortname has a script code which was
being stuffed with 0 in makePortName. A call is now made to
GetEnvirons to fetch the script code of the system script. This
value is place in the PPCPortName when open the PPCPort on
behalf of the app. ¥The maximum number of IDENTICAL Portnames
that can be browsed on the browser is 8. Any number over 8 will
not be recognize as EPPC aware application. A bug was introduced
in 7.0b1 in the routine GetNextPortName. StringByte is defined
as having a 0 based index instead of 1 based index. This caused
the code which runs in an attempt to generate a unique
PPCPortname to generate 8 uniques name instead of the 99 it
should have.
<14> 1/28/91 LN Changed connectionInValid to conectionInvalid to match change in
EPPC.h. Sorry Jay.
<13> 1/10/91 JWM (DFH) ¥1 BRC #79831 allow app to post using kCurrentProcess
constant. The change was in a parameter to the first
EPPCBlkFromPSN in GetConnectionType. Before <13>, EPPCBlkFromPSN
was called using the PSN Ptr in the PostHighLevelEvent. The fix
was to use the PSN in the process table. This means when the
caller of PostHighLevelEvent uses pCurrentProcess or
pFrontProcess, GetConnectionType will find the eppcBlk
associated with the target process. (EMT) ¥2 To keep the IO
completion routine (KnockKnock) from someday getting hung while
spinning on the ioresult field in the async PPCAccept or
PPCReject parameter block, a new switch was added for command
completion of a PPCAccept and PPCReject. The ppcInformBlk was
modified to include a PPCRejectParams structure
(ppcInformReplyPB). This structure is used for both the async
PPCReject and the async PPCAccept which occur when processing a
completion of a PPCInform. This change is isolated to
KnockKnock. (DFH) ¥3 Rework the DisableInterrupts and
EnableInterrupts macros to require fewer instruction for
compiler generated setup.
<12> 12/21/90 LN change constStrxxxParameter to ConstStrxxxParam as per change in
Types.h
<11> 12/21/90 DFH (gb) Added A5 setup in GetNextHighLevelEvent.
<10> 12/20/90 DFH (dba) Convert GetNextHighLevelEvent to a dispatch.
<9> 12/19/90 gbm (dba) Fix pointer arithmetic bugs throughout the file, including
the one that kept us from quitting applications with the 32-bit
Memory Manager.
<8> 12/14/90 JWM change rtrnReciptMsgID to rtrnReceiptMsgID
<7> 12/13/90 JWM Make the checkin comments look right
<6> 12/13/90 JWM DFH ¥1 Fix a bug with doDuplicateNameOpen. The termination of the loop
is wrong.
DFH ¥2 Added a call to Delete User Identity after a StartSecureSession
call. eppc doesn't need the User Identity. Deleting it frees up
resources inside the PPCToolbox.
DFH ¥3 StartSecureSession may cause the Dialog Manager to run. It is
best if our app's A5 is in the machine because user items in
dialog windows behind the authentication dialog(s) may execute.
Some user item procedures assume that A5 points to the application's
global not the Process Manager's globals.
DFH ¥4 The PPCToolbox as of 7.0B3 will no longer return errors through
PPCInit. References to ppcInitErr have been removed. In addition
a message was formatted and sent to the Shell if ppcInitErr did not
equal 0 (noErr). This message will be elimated in Startup.c.
DFH ¥5 BRC #78779-During PostHighLevelEvent to a remote machine,
if a proper port location is specified, but not a proper port name (i.e.
port name doesn't exist on the target machine), then the authentication
dialog is brought up. This behavior was fixed by making guestAllowed
return destPortErr (-906) if listPortPB.actualCount IS 0. getSessionID
will see the destPortErr and not call StartSecureSession.
DFH ¥6 Request from DTS to place TICKS in the when field of the event when
the message arrives. This is now done in KnockKnock when a read completes.
DFH ¥7 BRC # 78336ÐHighLevel event posted shortly before an application quits
does not get delivered to its target. This ÔbugÕ needs fixing. It is
due to closing the ppc port of an application during _ExitToShell. This
close causes all outstanding i/o for a port to go to completion. Any
post an application did between his last _WaitNextEvent and _ExitToShell
may not be delivered. This is due to the facts that processing the
i/o completion queues for PPCToolbox calls occurs during WaitNextEvent
if the system decides to call GetNextHLE and that PPCClose causes any
outstanding PPCWrites to end with an error. A goal for eppc has been
to shield the application writer from the async nature of the PPCToolbox
In this spirit, code was changed to keep ppc ports open after
application calls _ExitToShell if the port has PPCWrites which are
still pending. This allow the posts of a terminated application to
complete. Note: the PPCWrite still could error, but no one is
listening.
Changes to affect leaving the ppc port open after application
_ExitToShell involved storing the address of the last PPCWrite in
the session record, not breaking connections with outstanding
PPCWrites during _ExitToShell, closing the port when the last PPCWrite
completed and weaning all code paths in eppc that run after
_ExitToShell from there dependence on the EPPCBlk which goes away
at _ExitToShell.
DFH ¥8 An application posting a high level event from the background which
requires the network authenication dialog will receive an error
noUserInteractionAllowed (Ð610). This was done to fix the User Interface
NoÐNo of bringing up the a modal dialog when the application was
not the frontmost. Note: this situation occurs only on the first
post to a remote target. The application writer should use the
notification manager to indicate a need to interact with the user.
When the user brings the application forward, the application writer
should do the post again. After the post is sucessful, the
application can return to the background and post messages to the
remote target without the noUserInteractionAllowed error.
DFH ¥9 In eppc:getSessionID there were two error paths which did not release
the session table entry correctly.
<5> 11/26/90 DFH Change trespass() to dbmsg().
<4> 11/6/90 JWM ¥1 for the priority 1 bug dealing with an assert error when the
sender quits before the receiver has been delivered all his
messages.
¥2 you can now post messages of minimum unsigned long through
maximum unsigned long, if you have the memory.
¥3 In previous releases, when you didnÕt accept a message fully
an incorrect return message receipt was return to the sender.
This is now fixed and works as documented.
¥4 In previous releases three memory allocations and two memory
releases were necessary for each message received. In this
version each number has been reduced by one.
¥5 In previous releases, if the first PPCRead failed, the
connection would go dormant. In this release if the first
PPCRead fails for reasons other than sessionClosedErr or
noSessionErr, the connection is broken and the targetÕs session
table entry is updated to reflect the connection was explicitly
broken. If the reason was sessionClosedErr or noSessionErr, the
targetÕs connection table entry is updated to reflect the
connection was broken by the other end. Both these states are
seen from the programmerÕs view as sessionClosedErr if he tries
to post using for him what is a valid connection. He must
repost his message to establish a new connection with his
target.
¥6 In previous releases, if the second PPCRead failed, the
connection was broken and the targetÕs session table entry is
updated as in ¥4. However in the second read there are two
different classes of errors Ð memory errors and PPCToolbox
errors. In this release, memory errors cause the system to
restart the second read at a later time when, hopefully, the
memory configuration has changed. This change keeps the system
from breaking a session because of memory shortage. In the
previous release breaking the connection because of memory
failure caused the PPCWrites on the other end of the connection
to fail. Sometimes there were a lot of writes. For instance in
the Large Data Buffer test.
<2> 10/30/90 csd Update the symbols to match the headers on BBS.
<0> 9/02/89 JM New Today.
*/
#include <Errors.h>
#include <MFPrivate.h>
#include <Memory.h>
#include <MemoryMgrPriv.h>
#include <PPCToolBox.h>
#include <AppleEventsInternal.h>
#include <Errors.h>
#include <Resources.h>
#include <Values.h>
#include <Script.h>
#include "Data.h"
#include "Glue.h"
#include "Zone.h"
#include "Lomem.h"
#include "Patches.h"
#include "AppleEventExtensions.h"
#define IS ==
#define ISNT !=
#define AND &&
#define OR ||
#define ppcOpenCmd 1
#define ppcStartCmd 2
#define ppcInformCmd 3
#define ppcAcceptCmd 4
#define ppcRejectCmd 5
#define ppcWriteCmd 6
#define ppcReadCmd 7
#define ppcEndCmd 8
#define ppcCloseCmd 9
#define IPCListPortsCmd 10
typedef short ConnectionType;
#define eppcConnection (0)
#define systemReceiverConnection (1)
#define systemSenderConnection (2)
#define noEppcConnection (3)
#define badConnection (4)
#define noSessionID (0)
#define systemSessionID (-1)
#define INITIALIZED(eBlk) (eBlk->portID != 0)
#define TarEPPCBlkPtr(pPtr) (&(pPtr->eppcBlk))
#define CanDoEPPC(pPtr) (INITIALIZED(TarEPPCBlkPtr(pPtr)))
/* status bits in the Inform, Read and Write parameter blocks */
#define closeInProgress 0x8000
/* status bits in cTableEntry */
#define informConnection 0x8000
#define startConnection 0x4000
#define sessionClosed 0x2000
#define sessionEnded 0x1000
#define waitingForWrites 0x0800
#define HLEQDoesNotExist 0x0400
#define sizeOfPPCpbHeader (sizeof(LINKTYPE)+sizeof(Handle)+sizeof(EPPCBlkPtr)+2*sizeof(unsigned long))
#define sizeOfPPCWriteBlk (sizeOfPPCpbHeader+sizeof(PPCWritePBRec))
#define sizeOfPPCReadBlk (sizeOfPPCpbHeader+sizeof(PPCReadPBRec))
#define sizeOfPPCInformBlk (sizeOfPPCpbHeader+sizeof(PPCInformPBRec))
#define CheckForFatalError assert
#define SYNCHRONOUS 0
#define ASYNCHRONOUS 1
#define NROFINFORMPBS 3
#define MAXNROFTRIES 99
typedef struct ppcWriteBlk {
LINKTYPE ppcWriteBlkQ;
Handle pbHdl;
EPPCBlkPtr pbID;
unsigned long portID;
unsigned long status;
PPCWritePBRec ppcWritePB;
MFmsgBlk mfMsgBlk;
} ppcWriteBlk, *ppcWritePBPtr;
typedef struct ppcReadBlk {
LINKTYPE ppcReadBlkQ;
Handle pbHdl;
EPPCBlkPtr pbID;
unsigned long portID;
unsigned long status;
PPCReadPBRec ppcReadPB;
MFmsgBlk mfMsgBlk;
} ppcReadBlk, *ppcReadPBPtr;
typedef struct ppcInformBlk {
LINKTYPE ppcInformBlkQ;
Handle pbHdl;
EPPCBlkPtr pbID;
unsigned long portID;
unsigned long status;
PPCInformPBRec ppcInformPB;
PPCRejectPBRec ppcInformReplyPB; // <13>
PPCPortRec portName;
LocationNameRec portLoc;
} ppcInformBlk, *ppcInformPBPtr;
typedef struct cTableEntry {
LINKTYPE sLList; /* queue element */
Handle sHdl; /* handle for element */
unsigned long sportID; /* port id for _ExitToShell fix */
unsigned long sID; /* session id */
short status; /* status word */
PPCReadPBPtr srPB; /* outstanding read */
PPCWritePBPtr swPB; /* outstanding write */
EPPCBlkPtr sEppcBlkPtr;/* eppc control block */
PPCPortRec sPName; /* whose port name - depends on status */
LocationNameRec sLocName; /* whose location - depends on status */
} cTableEntry, *cTableEntryPtr, **cTableEntryHdl;
void KnockKnock(PPCParamBlockPtr ppcPB);
ConnectionType GetConnectionType(ProcessSerialNumberPtr postersPSN, unsigned long receiverID, unsigned long postingOptions, unsigned long *connectionID, ProcessSerialNumberPtr targetsPSN, OSErr *err);
unsigned long validateSessionID(ProcessSerialNumberPtr ownersPSN, unsigned long sessionID, OSErr *err);
Boolean PSNFromName(const PPCPortPtr portName, ProcessSerialNumberPtr pPSN);
Boolean PSNFromSignature(const ResType signature, ProcessSerialNumberPtr pPSN);
Boolean findElement(char *A, char *B);
Boolean byPBID(ppcWritePBPtr A, EPPCBlk *B);
MFmsgBlkPtr GetMsgEventBuffer (u_long lengthInBytes, OSErr *err);
unsigned long getSessionID(ProcessSerialNumberPtr sendersPSN, TargetIDPtr portN, OSErr *err);
EPPCBlkPtr EPPCBlkFromPSN(ProcessSerialNumberPtr psn);
PPCParamBlockPtr getPPCInformPB(EPPCBlkPtr refCon, unsigned short portRefNum, OSErr *err);
PPCParamBlockPtr getPPCReadPB(EPPCBlkPtr refCon, unsigned long sessionID, OSErr *err);
void releasePPCpb(PPCParamBlockPtr pb);
void postMsgToPPC(const MFmsgBlkPtr msgBlk, unsigned long sessionID, OSErr *err);
MFmsgBlkPtr fetchMsgData(EPPCBlkPtr eppcBlkPtr, MFmsgBlkPtr mfMsgBlk, OSErr *err);
void scanInformBQ(void);
void scanWriteBQ(void);
void scanReadBQ(void);
void getSessionRecord(SenderIDPtr senderID, EPPCBlk *recvrEppcB);
HighLevelEventMsgHdl
copyMsg(MFmsgBlkPtr aMsg, OSErr *err);
Boolean guestAllowed(TargetIDPtr portN, OSErr *err);
Boolean comparePortLocation(LocationNamePtr L1, LocationNamePtr L2);
Boolean comparePortNames(PPCPortPtr N1, PPCPortPtr N2);
void closePPCPort(unsigned long portID);
void openPPCPort(EPPCBlkPtr eppcB);
Boolean PortNameAndLocationAreEqual(PPCPortPtr portName, LocationNamePtr locName, TargetIDPtr targetID);
TargetID getTIDfromEppcBlk(EPPCBlkPtr eppcBPtr);
ppcReadPBPtr startPPCRead(EPPCBlkPtr eppcBPtr, unsigned long sessionID, OSErr *err);
ppcReadPBPtr restartPPCRead(ppcReadPBPtr rPB, OSErr *err);
PEntryPtr PEntryFromPSN(ProcessSerialNumberPtr pPSN);
cTableEntryPtr findSessionRecord(unsigned long sessionRefNum);
cTableEntryPtr findFreeSessionRecord(void);
void makePortName(EPPCBlkPtr eppcB);
void doDuplicateNameOpen(EPPCBlkPtr eppcB, PPCOpenPBPtr PPCpb);
void GetNextPortName(EPPCBlkPtr eppcB);
void DeleteBQEntries(unsigned long portID, LINKLIST *theBQ);
void qPPCPb(ppcWritePBPtr wPBPtr, LINKLIST *BQList);
void FlushHLEQueue(EPPCBlkPtr eppcBlk, Boolean sendReturnReceipts, short rtnMsg);
Boolean NameFromPSN(PPCPortPtr portName, ProcessSerialNumberPtr pPSN);
Boolean bySessionID(cTableEntryPtr A, unsigned long *B);
Boolean bysEppcBlkPtr(cTableEntryPtr A, EPPCBlk *B);
Boolean byportID(cTableEntryPtr A, unsigned long *B);
Boolean byportIDforPB(ppcWriteBlk *A, unsigned long *B);
short breakConnections(unsigned long portID);
void endSessionAfterAReadError(cTableEntryPtr sRec, OSErr err);
#pragma parameter DisableInterrupts(__A0)
void DisableInterrupts(unsigned short *oldMask) = {0x40D0, 0x007C, 0x0700};
/* move sr, (a0)
* ori.w 0x0700, sr
*/
#pragma parameter EnableInterrupts(__D0)
void EnableInterrupts(unsigned short oldMask) = {0x46C0};
/* move.w d0,sr
*/
typedef pascal Boolean
(*hleFilter_ptr)(unsigned long *params, Ptr msgBuff, SenderIDPtr senderID);
typedef pascal OSErr
(*StartSecureSession_ptr)(PPCStartPBPtr,
Str32 userName,
Boolean useDefault,
Boolean allowGuest,
Boolean *guestSelected,
ConstStr255Param prompt);
#pragma segment eppc_segment
StringPtr SSSPromptpStr; /* prompt string for StartSecureSession */
StringPtr MacOSPortNamepStr; /* port name of kSystemProcess */
StringPtr MacOSPortTypepStr; /* port type of kSystemProcess */
LINKLIST eppcBQ = {nil,nil,nil,0,0,sizeof(EPPCBlk),nil};
LINKLIST ppcInformBQ = {nil,nil,nil,0,0,sizeof(ppcInformBlk),nil};
LINKLIST ppcReadBQ = {nil,nil,nil,0,0,sizeof(ppcReadBlk),nil};
LINKLIST ppcWriteBQ = {nil,nil,nil,0,0,sizeof(ppcWriteBlk),nil};
LINKLIST connectionQ = {nil,nil,nil,0,0,sizeof(cTableEntry),nil};
/*____________________________________________________________________________________________*/
void
ePPCInit(void)
{
Handle hdl;
/* Ask the PPC Toolbox to initialize.*/
ppcInitErr = PPCInit();
/* Check for acceptable result, then try for our string resource */
if ((hdl = Get1Resource('STR#',-16415)) ISNT nil)
{
MoveHHi(hdl);
HLock(hdl);
SSSPromptpStr = *hdl+(Ptr)2;
MacOSPortNamepStr = (Ptr)SSSPromptpStr + (Ptr)(*(char *)SSSPromptpStr);
MacOSPortNamepStr += (Ptr)1;
MacOSPortTypepStr = (Ptr)MacOSPortNamepStr + (Ptr)(*(char *)MacOSPortNamepStr);
MacOSPortTypepStr += (Ptr)1;
}
}
/*____________________________________________________________________________________________*/
pascal OSErr
c_GetPortNameFromPSN(PPCPortPtr portName, ProcessSerialNumberPtr pPSN)
{
unsigned long olda5;
Boolean foundIt;
olda5 = ProcessMgrA5SimpleSetup();
foundIt = NameFromPSN(portName,pPSN);
A5SimpleRestore(olda5);
return (foundIt) ? noErr : noPortErr;
}
/*____________________________________________________________________________________________*/
pascal OSErr
c_GetPSNFromPortName(PPCPortPtr portName, ProcessSerialNumberPtr psnStorage)
{
unsigned long olda5;
Boolean foundIt;
olda5 = ProcessMgrA5SimpleSetup();
foundIt = PSNFromName(portName, psnStorage);
A5SimpleRestore(olda5);
return (foundIt) ? noErr : procNotFound;
}
/*____________________________________________________________________________________________*/
pascal Boolean
c_GetSpecificHighLevelEvent(ProcPtr aFilter, unsigned long *params, OSErr *err)
{
Boolean answer = false;
HighLevelEventMsgHdl msgCopyHandle;
unsigned long olda5;
MFmsgBlkPtr aMsg, bMsg, outstandingMsg;
Ptr theBuff;
EPPCBlkPtr eppcB;
int priority;
SenderID senderID;
olda5 = ProcessMgrA5SimpleSetup();
scanInformBQ();
scanReadBQ();
scanWriteBQ();
assert(pCurrentProcess ISNT pNullProcess);
assert(EqualPSNConst(kSystemProcess, &pCurrentProcess->p_serialNumber) IS false);
*err = noErr;
eppcB = &pCurrentProcess->eppcBlk; /* It maybe a bad assumption that pCurrentProcess an app */
if (eppcB->portID IS 0)
*err = noPortErr;
if (aFilter IS NULL)
*err = paramErr;
if (*err ISNT noErr)
{
A5SimpleRestore(olda5);
return(answer);
}
if (eppcB->mfMsgBlk ISNT nil)
{
aMsg = eppcB->mfMsgBlk;
msgCopyHandle = copyMsg(aMsg, err);
if (*err IS noErr) ;
else
{
A5SimpleRestore(olda5);
return(answer);
}
theBuff = *msgCopyHandle;
aMsg->msgStatus |= getSpecificHLE;
senderID.sessionID = (aMsg->sendermfid.sessionID IS systemSessionID) ? systemSessionID : aMsg->targetmfid.sessionID;
getSessionRecord(&senderID, eppcB);
A5SimpleRestore(olda5);
answer = CALL_FNC_PTR(hleFilter_ptr, aFilter, (params, theBuff, &senderID));
olda5 = ProcessMgrA5SimpleSetup();
DisposHandle(msgCopyHandle);
CheckForFatalError(MEMERROR IS noErr);
aMsg->msgStatus &= ~getSpecificHLE;
if (aMsg->msgStatus & msgPartialyAccepted)
{
aMsg->msgStatus &= ~msgPartialyAccepted;
aMsg->MFRefcon = 0;
if (aMsg->msgStatus & msgCompletelyAccepted)
flushMsg(aMsg,msgWasFullyAccepted);
else
flushMsg(aMsg,msgWasPartiallyAccepted);
eppcB->mfMsgBlk = nil;
}
if (answer != false)
{
A5SimpleRestore(olda5);
return(answer);
}
}
outstandingMsg = eppcB->mfMsgBlk;
for (priority=1;priority>=0;--priority) {
qset(&(eppcB->msgQ[priority]));
if (ll_length() IS 0)
aMsg = nil;
else
{
lltail();
llretrieve(&aMsg);
}
while (aMsg ISNT nil)
{
msgCopyHandle = copyMsg(aMsg, err);
if (*err IS noErr) ;
else {
answer = false;
eppcB->mfMsgBlk = outstandingMsg;
A5SimpleRestore(olda5);
return(answer);
}
theBuff = *msgCopyHandle;
eppcB->mfMsgBlk = aMsg;
aMsg->msgStatus |= getSpecificHLE;
senderID.sessionID = (aMsg->sendermfid.sessionID IS systemSessionID) ? systemSessionID : aMsg->targetmfid.sessionID;
getSessionRecord(&senderID, eppcB);
A5SimpleRestore(olda5);
answer = CALL_FNC_PTR(hleFilter_ptr, aFilter, (params, theBuff, &senderID));
olda5 = ProcessMgrA5SimpleSetup();
DisposHandle(msgCopyHandle);
CheckForFatalError(MEMERROR IS noErr);
aMsg->msgStatus &= ~getSpecificHLE;
qset(&(eppcB->msgQ[priority])); /* the user may have called something which
undid our call to qset(), so redo it */
if (aMsg->msgStatus & msgPartialyAccepted) /* was AcceptHighLevelEvent called? */
{
aMsg->msgStatus &= ~msgPartialyAccepted;
aMsg->MFRefcon = 0; /* in case we AcceptHLEvent this one again */
bMsg = aMsg->MFmsgBlkQ.previous;
lldelete();
flushMsg(aMsg,(aMsg->msgStatus & msgCompletelyAccepted) ? msgWasFullyAccepted:msgWasPartiallyAccepted);
qset(&(eppcB->msgQ[priority])); /* flushMsg may have called something which
undid our call to qset(), so redo it */
if (ll_length()) /* now find our next HLE */
if (bMsg ISNT nil)
{
eppcB->msgQ[priority].clp = bMsg;
aMsg = bMsg; /* llprevious() would return head of list */
}
else
{
aMsg = nil;
}
else
aMsg = nil; /* we deleted the last HLE */
}
else
{
if (llprevious())
llretrieve(&aMsg);
else
aMsg = nil;
}
if (answer != false)
break;
}
if (answer != false)
break;
} /* end of 'for loop' for each HLEvent Q attached to this port */
eppcB->mfMsgBlk = outstandingMsg;
A5SimpleRestore(olda5);
return(answer);
}
/*____________________________________________________________________________________________*/
HighLevelEventMsgHdl
copyMsg(MFmsgBlkPtr aMsg, OSErr *err)
{
HighLevelEventMsgHdl aHdl;
Ptr theCopyPtr;
aHdl = ProcessMgrHiNewHandle(sizeof(HighLevelEventMsg)+aMsg->eppcMsgBlk.msgLength, nil);
if ((*err = MEMERROR) != noErr)
return(aHdl);
HLock(aHdl);
theCopyPtr = *aHdl;
BlockMove(&aMsg->eppcMsgBlk,theCopyPtr,sizeof(HighLevelEventMsg));
theCopyPtr += (Ptr)sizeof(HighLevelEventMsg);
BlockMove(aMsg->addrOfMsg,theCopyPtr,aMsg->eppcMsgBlk.msgLength);
return(aHdl);
}
/*____________________________________________________________________________________________*/
pascal OSErr
c_AcceptHighLevelEvent(SenderIDPtr senderID, u_long *msgRefcon, Ptr msgBuff, u_long *msgLen)
{
unsigned long olda5;
short err;
unsigned long xferCount;
MFmsgBlkPtr mfMsgBlk;
/* save away the current caller's state, set major zone */
olda5 = ProcessMgrA5SimpleSetup();
err = noErr;
/* not sure about reference to pCurrentProcess being assumable. Bad things happen
* if it doesn't point to an app.
*/
if (pCurrentProcess->eppcBlk.mfMsgBlk) { /* is there a message for accepting? */
/* mfMsgBlk was set in event_common if we were delivering a pEvent.what = kHighLevelEvent or in
* GetSpecificHighLevelEvent.
*/
mfMsgBlk = (pCurrentProcess->eppcBlk.mfMsgBlk);
mfMsgBlk->msgStatus |= msgPartialyAccepted;
/* Can message fit into the user's buffer? */
if ((mfMsgBlk->eppcMsgBlk.msgLength - mfMsgBlk->MFRefcon) > *msgLen) {
/* message is larger than user's buffer */
xferCount = *msgLen; /* move what we can */
err = bufferIsSmall; /* tell the user his buffer is small */
/* tell the user how much is left to move */
*msgLen = mfMsgBlk->eppcMsgBlk.msgLength - mfMsgBlk->MFRefcon - *msgLen;
}
else {
/* message fits fine */
xferCount = mfMsgBlk->eppcMsgBlk.msgLength - mfMsgBlk->MFRefcon;
*msgLen = xferCount;
}
/* Move message */
if (msgBuff ISNT NULL && xferCount ISNT 0)
BlockMove(mfMsgBlk->addrOfMsg + (Ptr)mfMsgBlk->MFRefcon,msgBuff,xferCount);
*msgRefcon = mfMsgBlk->eppcMsgBlk.userRefcon;
senderID->sessionID = (mfMsgBlk->sendermfid.sessionID IS systemSessionID) ? systemSessionID : mfMsgBlk->targetmfid.sessionID;
getSessionRecord(senderID, &pCurrentProcess->eppcBlk);
/* Check to see if we moved all the message */
mfMsgBlk->MFRefcon += xferCount;
if (mfMsgBlk->MFRefcon >= mfMsgBlk->eppcMsgBlk.msgLength) {
/* message is all moved. Send the return receipt and release the memory */
mfMsgBlk->msgStatus |= msgCompletelyAccepted;
flushMsg(mfMsgBlk,msgWasFullyAccepted);
pCurrentProcess->eppcBlk.mfMsgBlk = nil;
}
}
else
err = noOutstandingHLE; /* HummÉ no High Level Event. His mistake or mine? */
/* restore caller's world */
A5SimpleRestore(olda5);
return(err);
}
/*____________________________________________________________________________________________*/
void
getSessionRecord(SenderIDPtr senderID, EPPCBlk *recvrEppcB)
{
cTableEntryPtr sRec;
MemClear(&senderID->location, sizeof(LocationNameRec));
MemClear(&senderID->name, sizeof(PPCPortRec));
MemClear(&senderID->recvrName, sizeof(PPCPortRec));
sRec = findSessionRecord(senderID->sessionID);
if (sRec ISNT nil)
{
senderID->name = sRec->sPName;
if (sRec->sLocName.locationKindSelector IS ppcNBPLocation)
senderID->location = sRec->sLocName;
else
senderID->location.locationKindSelector = ppcNoLocation;
senderID->recvrName = sRec->sEppcBlkPtr->nameOfMsgQ;
}
else
{
if (senderID->sessionID IS systemSessionID)
{
BlockMove(MacOSPortNamepStr, &senderID->name.name, *(char *)MacOSPortNamepStr+1);
senderID->name.portKindSelector = ppcByString;
BlockMove(MacOSPortTypepStr, &senderID->name.u.portTypeStr, *(char *)MacOSPortTypepStr+1);
senderID->recvrName = recvrEppcB->nameOfMsgQ;
}
}
}
/*____________________________________________________________________________________________*/
void
FlushHLEQueue(EPPCBlkPtr eppcBlk, Boolean sendReturnReceipts, short rtnMsg)
{
MFmsgBlkPtr aMsg=eppcBlk->mfMsgBlk;
int priority;
if (aMsg ISNT nil)
if (sendReturnReceipts)
flushMsg(aMsg, rtnMsg);
else
{
RelMsgEventBuffer(aMsg);
}
for (priority=1;priority>=0;priority--)
{
LINKLIST *theMsgQ = &(eppcBlk->msgQ[priority]);
qset(theMsgQ);
llhead();
while(ll_length() ISNT 0)
{
llretrieve(&aMsg);
lldelete();
if (sendReturnReceipts)
{
flushMsg(aMsg, rtnMsg);
qset(theMsgQ); /* flush queue may undo our qset, so redo it. */
}
else
{
RelMsgEventBuffer(aMsg);
}
}
}
}
/*____________________________________________________________________________________________*/
void
flushMsg(MFmsgBlkPtr msgBlk, unsigned short rtnModifiers)
{
if ((msgBlk->eppcMsgBlk.postingOptions & nReturnReceipt) AND ((msgBlk->msgStatus & getSpecificHLE) IS 0))
postReturnReceipt(msgBlk,rtnModifiers);
RelMsgEventBuffer(msgBlk);
}
/*_____________________________________fetchMsgData______________________________________*/
MFmsgBlkPtr
fetchMsgData(EPPCBlkPtr eppcBlkPtr, MFmsgBlkPtr mfMsgBlk, OSErr *err)
{
ppcReadPBPtr readPB;
Handle aHdl;
aHdl = GMBlk(sizeof(ppcReadBlk)+mfMsgBlk->eppcMsgBlk.msgLength, err);
if (*err ISNT noErr)
return nil;
HLock(aHdl);
readPB = *aHdl;
MemClear(readPB,sizeof(ppcReadBlk)+mfMsgBlk->eppcMsgBlk.msgLength);
readPB->pbHdl = aHdl;
readPB->pbID = eppcBlkPtr;
readPB->ppcReadPB.bufferPtr = readPB + 1; // <9>
readPB->ppcReadPB.bufferLength = mfMsgBlk->eppcMsgBlk.msgLength;
readPB->mfMsgBlk = *mfMsgBlk;
readPB->mfMsgBlk.addrOfMsg = readPB->ppcReadPB.bufferPtr;
readPB->mfMsgBlk.pbHandle = aHdl;
readPB->ppcReadPB.csCode = ppcReadCmd;
readPB->ppcReadPB.ioCompletion = nil;
readPB->ppcReadPB.sessRefNum = mfMsgBlk->targetmfid.sessionID;
if (mfMsgBlk->eppcMsgBlk.msgLength IS 0);
else
*err = PPCRead(&readPB->ppcReadPB,SYNCHRONOUS);
return (&readPB->mfMsgBlk);
}
/*____________________________________________________________________________________________*/
void
postReturnReceipt(const MFmsgBlkPtr mfMsgBlk, signed short rtnModifiers)
{
EventRecord theEvent;
u_long receiverID;
u_long msgRefcon;
u_long postingOptions;
if (mfMsgBlk->targetmfid.sessionID IS systemSessionID)
{
receiverID = &mfMsgBlk->sendermfid.localPSN;
postingOptions = receiverIDisPSN;
}
else
{
receiverID = mfMsgBlk->targetmfid.sessionID;
postingOptions = receiverIDisSessionID;
}
msgRefcon = mfMsgBlk->eppcMsgBlk.userRefcon;
theEvent.message = HighLevelEventMsgClass;
LONG_TO_PT(rtrnReceiptMsgID,&theEvent.where);
theEvent.modifiers = rtnModifiers;
PostHighLevelEvent(&theEvent, receiverID, msgRefcon, nil, 0, postingOptions);
}
/*____________________________________________________________________________________________*/
pascal short
c_PostHighLevelEvent(EventRecord *theEvent, u_long receiverID, u_long msgRefcon, Ptr msgBuff, u_long msgLen, u_long postingOptions)
{
ProcessSerialNumber SendersPSN;
unsigned long olda5, connectionID;
short err, anotherErr;
ConnectionType connectionType;
MFmsgBlkPtr msgBlk;
ProcessSerialNumber aPSN;
olda5 = ProcessMgrA5SimpleSetup();
err = noErr;
/* Make sure msgLen is reasonable */
if (msgBuff == nil)
msgLen = 0;
else if (msgLen > 0x0000ffffL)
{
A5SimpleRestore(olda5);
return(ddpLenErr);
}
(void) GetSystemClientProcess(&SendersPSN);
connectionType = GetConnectionType(&SendersPSN, receiverID, postingOptions, &connectionID, &aPSN, &err);
if (connectionType == badConnection)
{
A5SimpleRestore(olda5);
return(err);
}
/* badConnection has been weeded out. Try and get memory for message and necessary control info */
if ((msgBlk = GetMsgEventBuffer(msgLen, &anotherErr)) IS nil)
{
A5SimpleRestore(olda5);
return(anotherErr);
}
/* Set up the event record we'll give to the target */
msgBlk->eppcMsgBlk.theMsgEvent = *theEvent;
msgBlk->eppcMsgBlk.theMsgEvent.what = kHighLevelEvent;
msgBlk->eppcMsgBlk.theMsgEvent.when = TICKS;
/* Set up the message and refcon */
if (msgLen != 0)
BlockMove(msgBuff, msgBlk->addrOfMsg, msgLen);
msgBlk->eppcMsgBlk.userRefcon = msgRefcon;
msgBlk->MFRefcon = 0;
msgBlk->eppcMsgBlk.postingOptions = postingOptions;
/* Set up routing information */
msgBlk->sendermfid.localPSN = SendersPSN;
msgBlk->targetmfid.localPSN = aPSN;
msgBlk->sendermfid.sessionID = connectionID;
msgBlk->targetmfid.sessionID = noSessionID;
/* msgBlk is complete. Put it where it belongs based on connectionType. */
if (connectionType == eppcConnection)
{
postMsgToPPC(msgBlk, connectionID, &anotherErr);
if ((postingOptions & receiverIDMask) == receiverIDisTargetID)
((TargetIDPtr)receiverID)->sessionID = (anotherErr == noErr) ? connectionID : noSessionID;
err = anotherErr;
}
else if (connectionType == systemSenderConnection)
{
msgBlk->msgStatus |= localOnly;
postMsg(msgBlk);
}
else if (connectionType == systemReceiverConnection)
{
HandleSystemMessage(msgBlk, &err);
}
else if (connectionType == noEppcConnection)
TranslateAppleEvent(msgBlk, &err);
A5SimpleRestore(olda5);
return(err);
}
/*____________________________________________________________________________________________*/
void
CreateMsgQ(PEntryPtr pProc)
{
EPPCBlk eppcB;
short i;
MemClear(&eppcB,sizeof(EPPCBlk));
eppcB.pTablePtr = pProc;
qset(&eppcB.msgQ[0]);
qitemsize(sizeof(MFmsgBlk)); /* actually represents the header size only */
qset(&eppcB.msgQ[1]);
qitemsize(sizeof(MFmsgBlk)); /* actually represents the header size only */
/* Build eppcB port name from resource or p_taskmode */
makePortName(&eppcB);
openPPCPort(&eppcB);
if (eppcB.createErr IS noErr)
/* carve out a parameter block for a async PPCInform call and start the Informs*/
for (i=0;i<NROFINFORMPBS;i++)
{
eppcB.iPBs[i] = (PPCInformPBPtr)getPPCInformPB(&pProc->eppcBlk, eppcB.portID, &eppcB.createErr);
if (eppcB.createErr ISNT noErr)
break;
eppcB.createErr = PPCInform(eppcB.iPBs[i],ASYNCHRONOUS);
if (eppcB.createErr ISNT noErr)
break;
} /* end of for loop */
if (eppcB.createErr ISNT noErr)
closePPCPort(eppcB.portID);
pProc->eppcBlk = eppcB;
if (eppcB.createErr IS noErr)
{
qset(&eppcBQ);
q_push(&pProc->eppcBlk);
}
}
/*____________________________________________________________________________________________*/
void
makePortName(EPPCBlkPtr eppcB)
{
Ptr namePtr = nil;
short lenOfPortName;
Handle hdl;
if ((eppcB->pTablePtr->p_taskmode & modeHighLevelEventAware) AND ((hdl = (Handle)Get1Resource('eppc',0)) ISNT nil))
{
namePtr = *hdl;
eppcB->optionFlags = *(unsigned long *)namePtr;
if (eppcB->pTablePtr->p_taskmode & modeLocalAndRemoteHLEvents)
eppcB->optionFlags |= registerOnNetwork;
namePtr += (Ptr)sizeof(unsigned long);
eppcB->reserved1 = *(unsigned long *)namePtr;
namePtr += (Ptr)sizeof(unsigned long);
eppcB->nameOfMsgQ.nameScript = *(short *)namePtr;
namePtr += (Ptr)sizeof(short);
lenOfPortName = *(char *)namePtr;
namePtr += (Ptr)sizeof(char);
eppcB->nameOfMsgQ.name[0] = lenOfPortName;
BlockMove(namePtr,&eppcB->nameOfMsgQ.name[1],lenOfPortName);
eppcB->nameOfMsgQ.portKindSelector = ppcByString;
eppcB->nameOfMsgQ.u.portTypeStr[0] = 8;
BlockMove((Ptr)&eppcB->pTablePtr->p_signature,(Ptr)&eppcB->nameOfMsgQ.u.portTypeStr[1],4);
BlockMove("ep01",(Ptr)&eppcB->nameOfMsgQ.u.portTypeStr[5],4);
ReleaseResource(hdl);
return;
}
if (eppcB->pTablePtr->p_taskmode & modeHighLevelEventAware)
{
if (eppcB->pTablePtr->p_taskmode & modeLocalAndRemoteHLEvents)
eppcB->optionFlags |= registerOnNetwork;
eppcB->reserved1 = 0;
eppcB->nameOfMsgQ.nameScript = (short)GetEnvirons(smSysScript);
BlockMove(CURAPNAME, &eppcB->nameOfMsgQ.name, *((unsigned char *)CURAPNAME) + 1);
eppcB->nameOfMsgQ.portKindSelector = ppcByString;
eppcB->nameOfMsgQ.u.portTypeStr[0] = 8;
BlockMove((Ptr)&eppcB->pTablePtr->p_signature,(Ptr)&eppcB->nameOfMsgQ.u.portTypeStr[1],4);
BlockMove("ep01",(Ptr)&eppcB->nameOfMsgQ.u.portTypeStr[5],4);
return;
}
eppcB->createErr = -1;
}
/*____________________________________________________________________________________________*/
void
openPPCPort(EPPCBlkPtr eppcB) {
PPCOpenPBRec PPCpb;
if (eppcB->createErr ISNT noErr)
return;
MemClear(&PPCpb,sizeof(PPCOpenPBRec));
PPCpb.csCode = ppcOpenCmd;
PPCpb.ioCompletion = NULL;
PPCpb.serviceType = ppcServiceRealTime;
if (eppcB->optionFlags & registerOnNetwork)
PPCpb.networkVisible = true;
PPCpb.portName = &eppcB->nameOfMsgQ;
PPCpb.locationName = NULL;
eppcB->createErr = PPCOpen(&PPCpb, SYNCHRONOUS);
/* If error was name duplication, try adjusting the name until it works */
if (eppcB->createErr IS portNameExistsErr)
doDuplicateNameOpen(eppcB, &PPCpb);
/* Check result of PPCOpen or doDuplicateNameOpen */
if (eppcB->createErr IS noErr)
eppcB->portID = PPCpb.portRefNum;
}
/*____________________________________________________________________________________________*/
void
doDuplicateNameOpen(register EPPCBlkPtr eppcB, register PPCOpenPBPtr PPCpb)
{
register short nrOfTries;
nrOfTries = MAXNROFTRIES;
do {
GetNextPortName(eppcB);
eppcB->createErr = PPCOpen(PPCpb, SYNCHRONOUS);
} while ((eppcB->createErr IS portNameExistsErr) AND (--nrOfTries > 0));
}
/*____________________________________________________________________________________________*/
void
GetNextPortName(EPPCBlkPtr eppcB)
{
char lsd, msd;
lsd = StringByte(eppcB->nameOfMsgQ.u.portTypeStr, 7);
msd = StringByte(eppcB->nameOfMsgQ.u.portTypeStr, 6);
if (lsd IS '9')
{
lsd = '0';
if (msd IS '9')
msd = '0';
else
msd++;
}
else
lsd++;
StringByte(eppcB->nameOfMsgQ.u.portTypeStr, 7) = lsd;
StringByte(eppcB->nameOfMsgQ.u.portTypeStr, 6) = msd;
}
/*_________________________________________closePPCPort___________________________________*/
void
closePPCPort(unsigned long portID)
{
PPCClosePBRec closePB;
if (portID ISNT 0) {
closePB.csCode = ppcCloseCmd;
closePB.ioCompletion = nil;
closePB.portRefNum = portID;
PPCClose(&closePB, SYNCHRONOUS);
}
DeleteBQEntries(portID, &ppcReadBQ);
DeleteBQEntries(portID, &ppcWriteBQ);
DeleteBQEntries(portID, &ppcInformBQ);
return;
}
/*_____________________________________________DestroyMsgQ____________________________________*/
void
DestroyMsgQ(EPPCBlk *eppcBlk)
{
unsigned short oldMask = 0;
short i;
register PPCParamBlockPtr *pIPB;
if (INITIALIZED(eppcBlk) == false)
return;
/* tell KnockKnock to reject informs */
pIPB = &eppcBlk->iPBs[0];
DisableInterrupts(&oldMask);
for (i=NROFINFORMPBS; i > 0; i--)
{
ppcInformPBPtr thePB = (Ptr)*pIPB - sizeOfPPCpbHeader; // <9>
thePB->status |= closeInProgress;
pIPB++;
}
EnableInterrupts(oldMask);
/* take care of outstanding messages */
FlushHLEQueue(eppcBlk, false, 0);
if (breakConnections(eppcBlk->portID) IS 0)
closePPCPort(eppcBlk->portID);
/* remove the eppc block from the chain because its memory is going away*/
qset(&eppcBQ);
llsetMatch((ProcPtr)findElement);
llhead();
if (llcheck(eppcBlk))
lldelete();
else
assert(0);
}
/*_______________________________________breakConnections_________________________________*/
short
breakConnections(unsigned long portID)
{
OSErr ppcEndErr;
PPCEndPBRec endPB;
cTableEntryPtr sRec;
short nrOfSessionsLeftOpen = 0;
qset(&connectionQ);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)byportID);
llhead();
while (llcheck(&portID))
{
llretrieve(&sRec);
sRec->status |= HLEQDoesNotExist;
if (sRec->swPB IS nil)
{
if (!(sRec->status & (sessionClosed+sessionEnded)))
{
endPB.csCode = ppcEndCmd;
endPB.ioCompletion = nil;
endPB.sessRefNum = sRec->sID;
ppcEndErr = PPCEnd(&endPB,SYNCHRONOUS);
if (ppcEndErr IS noErr OR ppcEndErr IS noSessionErr) ;
else
assert(0);
}
lldelete();
DisposHandle(sRec->sHdl);
}
else
{
nrOfSessionsLeftOpen++;
sRec->status |= waitingForWrites;
if (!llnext())
break;
}
}
}
return(nrOfSessionsLeftOpen);
}
#if 0
/*____________________________________________DeleteBQEntries_________________________________*/
void
DeleteBQEntries(unsigned long portID, LINKLIST *theBQ)
{
ppcWritePBPtr pb;
unsigned short oldMask;
qset(theBQ);
DisableInterrupts(&oldMask);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)byportIDforPB);
llhead();
while (llcheck(&portID))
{
llretrieve(&pb);
lldelete();
DisposHandle(pb->pbHdl);
}
}
EnableInterrupts(oldMask);
}
#else
/*____________________________________________DeleteBQEntries_________________________________*/
void
DeleteBQEntries(unsigned long portID, LINKLIST *theBQ)
{
ppcWritePBPtr pb;
ppcWritePBPtr nextPB;
unsigned short oldMask;
LINKLIST aListToDisposeOf;
MemClear(&aListToDisposeOf, sizeof(LINKLIST));
qset(theBQ);
DisableInterrupts(&oldMask);
++theBQ->listInUse;
if (ll_length() ISNT 0)
{
llhead();
llretrieve(&pb);
while (pb ISNT nil)
{
if (llnext())
llretrieve(&nextPB);
else
nextPB = nil;
if (pb->portID IS portID)
{
theBQ->clp = pb;
lldelete();
qset(&aListToDisposeOf);
q_push(pb);
qset(theBQ);
}
if (nextPB ISNT nil)
theBQ->clp = nextPB;
pb = nextPB;
}
}
--theBQ->listInUse;
EnableInterrupts(oldMask);
qset(&aListToDisposeOf);
while (ll_length() ISNT 0)
{
q_pop(&pb);
DisposHandle(pb->pbHdl);
}
}
#endif
/*__________________________________GetMsgEventBuffer__________________________________________*/
MFmsgBlkPtr
GetMsgEventBuffer(u_long lengthInBytes, OSErr *err)
{
ppcWritePBPtr msgBlk = nil;
Handle aHdl = nil;
aHdl = GMBlk(sizeof(ppcWriteBlk)+lengthInBytes, err);
if (*err ISNT noErr)
return(nil);
HLock(aHdl);
msgBlk = *aHdl;
MemClear(msgBlk,sizeof(ppcWriteBlk)+lengthInBytes);
msgBlk->pbHdl = aHdl;
msgBlk->mfMsgBlk.pbHandle = aHdl;
msgBlk->mfMsgBlk.eppcMsgBlk.version = 3;
msgBlk->mfMsgBlk.eppcMsgBlk.HighLevelEventMsgHeaderLength = sizeof(HighLevelEventMsg);
msgBlk->mfMsgBlk.eppcMsgBlk.msgLength = lengthInBytes;
msgBlk->mfMsgBlk.addrOfMsg = (lengthInBytes IS 0) ? NULL : msgBlk + 1; // <9>
return(&msgBlk->mfMsgBlk);
}
/*_____________________________________RelMsgEventBuffer________________________________________*/
void
RelMsgEventBuffer (MFmsgBlkPtr mfMsgBlk)
{
if (mfMsgBlk->msgStatus & getSpecificHLE)
return;
DisposHandle(mfMsgBlk->pbHandle);
CheckForFatalError(MEMERROR IS noErr);
}
/*____________________________________________________________________________________________*/
void
postMsg(const MFmsgBlkPtr msgBlk)
{
PEntryPtr pTargetProc;
EPPCBlkPtr teppcBlk;
pTargetProc = PEntryFromPSN(&msgBlk->targetmfid.localPSN);
CheckForFatalError(pTargetProc != nil);
teppcBlk = TarEPPCBlkPtr(pTargetProc);
if (msgBlk->eppcMsgBlk.postingOptions & nAttnMsg)
qset(&(teppcBlk->msgQ[1]));
else
qset(&(teppcBlk->msgQ[0]));
q_push(msgBlk);
/* Wake him up, if necessary */
CancelSleep(pTargetProc);
}
/*____________________________________postMsgToPPC___________________________________________*/
void
postMsgToPPC(const MFmsgBlkPtr msgBlk, unsigned long sessionID, OSErr *err)
{
ppcWritePBPtr writePB;
unsigned long *along;
cTableEntryPtr sRec;
writePB = (ppcWritePBPtr) ((Ptr)msgBlk - sizeOfPPCWriteBlk); // <9>
writePB->ppcWritePB.csCode = ppcWriteCmd;
writePB->ppcWritePB.ioCompletion = KnockKnock;
writePB->ppcWritePB.sessRefNum = sessionID;
writePB->ppcWritePB.blockCreator = msgBlk->eppcMsgBlk.theMsgEvent.message;
along = &(msgBlk->eppcMsgBlk.theMsgEvent.where);
writePB->ppcWritePB.blockType = *along;
writePB->ppcWritePB.bufferLength = sizeof(HighLevelEventMsg)+msgBlk->eppcMsgBlk.msgLength;
writePB->ppcWritePB.bufferPtr = &msgBlk->eppcMsgBlk;
writePB->pbID = EPPCBlkFromPSN(&msgBlk->sendermfid.localPSN);
/* The following code was added to keep track of the last PPCWrite. */
/* A new field was created in the connection table entry record (session */
/* record) - swPB. Into this field we store the address of the ppcWritePBPtr*/
/* The swPB is used in ScanWriteBQ. This is part of the fix for lost post */
/* during _ExitToShell */
sRec = findSessionRecord(sessionID);
sRec->swPB = writePB;
writePB->portID = sRec->sportID;
*err = PPCWrite(&writePB->ppcWritePB,ASYNCHRONOUS);
}
/*_____________________________GetNextHighLevelEvent____________________________________________*/
pascal Boolean
c_GetNextHighLevelEvent(short eventmask, EventRecord *pReturnEvent, Boolean pullevent)
{
MFmsgBlkPtr aMsg, bMsg;
short priority;
unsigned long olda5;
Boolean retVal;
olda5 = ProcessMgrA5SimpleSetup();
retVal = false;
scanInformBQ();
scanReadBQ();
scanWriteBQ();
assert(pCurrentProcess ISNT pNullProcess);
assert(EqualPSNConst(kSystemProcess, &pCurrentProcess->p_serialNumber) IS false);
for (priority=1;priority>=0;priority--)
{
if (pCurrentProcess->eppcBlk.msgQ[priority].listlength ISNT 0 AND (eventmask & networkMask))
{
qset(&pCurrentProcess->eppcBlk.msgQ[priority]);
lltail();
llretrieve(&aMsg);
*pReturnEvent = aMsg->eppcMsgBlk.theMsgEvent;
if (pullevent)
{
q_pop(&bMsg);
CheckForFatalError(bMsg IS aMsg);
pCurrentProcess->eppcBlk.mfMsgBlk = (Handle)bMsg;
}
retVal = true;
break;
}
}
A5SimpleRestore(olda5);
return(retVal);
}
/*________________________________________scanInformBQ__________________________________________*/
void
scanInformBQ(void)
{
ppcInformPBPtr iPB;
PPCEndPBRec ppcEndPB;
OSErr err;
unsigned short oldMask = 0;
cTableEntryPtr sRec;
for (;;)
{
DisableInterrupts(&oldMask);
++ppcInformBQ.listInUse;
if (ppcInformBQ.listlength IS 0)
{
--ppcInformBQ.listInUse;
EnableInterrupts(oldMask);
break;
}
qset(&ppcInformBQ);
q_pop(&iPB);
--ppcInformBQ.listInUse;
EnableInterrupts(oldMask);
if (iPB->ppcInformPB.ioResult ISNT noErr)
{
releasePPCpb(&iPB->ppcInformPB);
continue;
}
if (iPB->ppcInformReplyPB.csCode IS ppcRejectCmd);
else
if ((sRec = findFreeSessionRecord()) ISNT nil)
{
sRec->status |= informConnection;
if (iPB->status & closeInProgress)
sRec->status |= HLEQDoesNotExist;
sRec->sID = iPB->ppcInformPB.sessRefNum;
sRec->sPName = *(iPB->ppcInformPB.portName);
if (iPB->ppcInformPB.requestType IS ppcRemoteOrigin)
sRec->sLocName = *(iPB->ppcInformPB.locationName);
else
sRec->sLocName.locationKindSelector = ppcNoLocation;
sRec->sEppcBlkPtr = iPB->pbID; /* pbID may no longer be valid */
sRec->sportID = iPB->portID;
sRec->srPB = startPPCRead(iPB->pbID, iPB->ppcInformPB.sessRefNum, &err);
if (err ISNT noErr)
DisposHandle(sRec->sHdl);
else
{
qset(&connectionQ);
q_push(sRec);
}
}
else
{
MemClear(&ppcEndPB,sizeof(PPCEndPBRec));
ppcEndPB.sessRefNum = iPB->ppcInformPB.sessRefNum;
ppcEndPB.csCode = ppcEndCmd;
ppcEndPB.ioCompletion = nil;
PPCEnd(&ppcEndPB,SYNCHRONOUS);
}
iPB->ppcInformPB.sessRefNum = 0;
MemClear(iPB->ppcInformPB.locationName, sizeof(LocationNameRec));
MemClear(iPB->ppcInformPB.portName, sizeof(PPCPortRec));
err = PPCInform(&iPB->ppcInformPB,ASYNCHRONOUS);
}
}
/*____________________________________scanWriteBQ_______________________________________________*/
void
scanWriteBQ(void)
{
ppcWritePBPtr wPB = nil;
ppcWritePBPtr lastwPB = nil;
unsigned short oldMask = 0;
cTableEntryPtr sRec;
PPCEndPBRec ppcEndPB;
for (;;)
{
DisableInterrupts(&oldMask);
++ppcWriteBQ.listInUse;
if (ppcWriteBQ.listlength IS 0)
{
--ppcWriteBQ.listInUse;
EnableInterrupts(oldMask);
break;
};
qset(&ppcWriteBQ);
q_pop(&wPB);
--ppcWriteBQ.listInUse;
EnableInterrupts(oldMask);
sRec = findSessionRecord(wPB->ppcWritePB.sessRefNum);
if (sRec IS nil)
{
releasePPCpb(&wPB->ppcWritePB);
continue;
}
if (wPB->ppcWritePB.ioResult IS noErr);
else
{
/* closing the port or ending a session causes writes to go to completion */
if (wPB->ppcWritePB.ioResult IS noSessionErr OR wPB->ppcWritePB.ioResult IS sessClosedErr)
sRec->status |= sessionClosed;
else
{
if (sRec->status & waitingForWrites);
else
{
MemClear(&ppcEndPB,sizeof(PPCEndPBRec));
ppcEndPB.sessRefNum = wPB->ppcWritePB.sessRefNum;
ppcEndPB.csCode = ppcEndCmd;
ppcEndPB.ioCompletion = nil;
PPCEnd(&ppcEndPB,SYNCHRONOUS);
sRec->status |= sessionEnded;
}
}
}
releasePPCpb(&wPB->ppcWritePB);
lastwPB = sRec->swPB;
if (lastwPB IS wPB)
{
sRec->swPB = nil;
if (sRec->status & waitingForWrites)
{
unsigned long portID = sRec->sportID;
sRec->status &= ~waitingForWrites;
if (breakConnections(portID) IS 0)
closePPCPort(portID);
}
}
}
}
/*_____________________________________scanReadBQ____________________________________________*/
void
scanReadBQ(void)
{
ppcReadPBPtr rPB;
OSErr err;
unsigned short oldMask = 0;
cTableEntryPtr sRec;
MFmsgBlkPtr aMFmsg;
/* Loop until the queue is empty. The queue is added to by KnockKnock, the
io completion for PPCToolbox calls. The loop is terminated when the queue
length goes to 0. Notice the calls to DisableInterrupts around the queue
manipulation logic.
*/
for (;;)
{
DisableInterrupts(&oldMask);
++ppcReadBQ.listInUse;
if (ppcReadBQ.listlength IS 0)
{
--ppcReadBQ.listInUse;
EnableInterrupts(oldMask);
break; /* the queue is empty. exit the for loop. */
}
qset(&ppcReadBQ);
q_pop(&rPB);
--ppcReadBQ.listInUse;
EnableInterrupts(oldMask);
sRec = findSessionRecord(rPB->ppcReadPB.sessRefNum);
if (sRec IS nil)
{
releasePPCpb(&rPB->ppcReadPB); /* could not find a session record match */
continue; /* for this completed read. This should */
} /* mean that DestroyMsgQ ended the */
/* session (i.e. the target has called */
/* _ExitToShell */
if (rPB->ppcReadPB.ioResult IS noErr);
else
{
endSessionAfterAReadError(sRec, rPB->ppcReadPB.ioResult);
continue;
}
/* set up to get the data portion of this High Level Event, if any. */
rPB->mfMsgBlk.targetmfid.sessionID = rPB->ppcReadPB.sessRefNum;
/* The next assignment determines the local target of the message. Note: you
cannot assume that the target is still alive (i.e. rPB->pbID->pTablePtr is
valid). If HLEQDoesNotExist is set, then the app called _ExitToShell and
we need to set localPSN accordingly.
*/
if (sRec->status & HLEQDoesNotExist)
{
SetPSN(kNoProcess, &rPB->mfMsgBlk.targetmfid.localPSN);
}
else
rPB->mfMsgBlk.targetmfid.localPSN = rPB->pbID->pTablePtr->p_serialNumber;
rPB->mfMsgBlk.addrOfMsg = nil;
aMFmsg = fetchMsgData(rPB->pbID, &rPB->mfMsgBlk, &err);
if (err IS noErr);
else /* There was an error during fetchMsgData */
{
if (aMFmsg IS nil) /* Was it a memory error? */
{
rPB->mfMsgBlk.msgStatus |= msgMemoryRestart;
qset(&ppcReadBQ);
DisableInterrupts(&oldMask);
++ppcReadBQ.listInUse;
if (ppcReadBQ.listlength IS 0)
q_push(rPB);
else
{
lltail();
lladd(rPB);
}
--ppcReadBQ.listInUse;
EnableInterrupts(oldMask);
break;
}
else /* it was not a memory error */
{
endSessionAfterAReadError(sRec, err);
if (aMFmsg)
RelMsgEventBuffer(aMFmsg);
continue;
}
}
if (sRec->status & HLEQDoesNotExist)
RelMsgEventBuffer(aMFmsg);
else
{
/* Put the HLE on the appropriate list */
if (rPB->mfMsgBlk.eppcMsgBlk.postingOptions & nAttnMsg)
qset(&(rPB->pbID->msgQ[1]));
else
qset(&(rPB->pbID->msgQ[0]));
q_push(aMFmsg);
/* Wake him up, if necessary */
CancelSleep(rPB->pbID->pTablePtr);
}
/* start the outstanding read on this session */
sRec->srPB = restartPPCRead(rPB, &err);
}
}
/*__________________________________endSessionAfterAReadError________________________________________*/
void
endSessionAfterAReadError(cTableEntryPtr sRec, OSErr err)
{
PPCEndPBRec ppcEndPB;
if (err IS noSessionErr OR err IS sessClosedErr)
sRec->status |= sessionClosed;
else
{
if (sRec->status & waitingForWrites);
else
{
MemClear(&ppcEndPB,sizeof(PPCEndPBRec));
ppcEndPB.sessRefNum = sRec->sID;
ppcEndPB.csCode = ppcEndCmd;
ppcEndPB.ioCompletion = nil;
PPCEnd(&ppcEndPB,SYNCHRONOUS);
sRec->status |= sessionEnded;
}
}
releasePPCpb(sRec->srPB);
sRec->srPB = nil;
}
/*____________________________________________________________________________________________*/
/* GetConnectionType. Returns the type of connection we can have from the sender to the receiver.
* It takes as input a SendersPSN, a receiverID, postion options.
* It returns, in addition to type of connection, connectionID, PSN of the target, and err.
* Results: eppcConnection means the connection is routed thru the PPCToolbox.
* noEppcConnection means that the receiver was located, but it cannot understand EPPC.
* systemReceiverConnection means the message was addressed to "the system"
* systemSenderConnection means the message was from the "the system"
* badConnection means that there is no possibility of connection.
* If the result is eppcConnection, the target is in connectionID. Otherwise, it is in targetsPSN.
* If the result is badConnection or noEppcConnection, the appropriate error code is returned
* in err. This error should be returned in the noEppcConnection case if the event can not
* be translated.
* Note: ¥ SendersPSN can be kSystemProcess, kNoProcess, or somebodies PSN.
* ¥ SendersPSN equal to kNoProcess is an error, I thought; I coded.
* ¥ SendersPSN equal to kSystemProcess means someone called us with SystemMode
* turned on.
* ¥ receiverID is the target identification. How receiverID is interpeted is determined
* by postingOptions. There are four options as of 7.0aA7.
* ¥ connectionID is the target identification. On output how connectionID is interpeted is
* determined by type of connection. There are five types as of 7.0aA7.
* ¥ targetsPSN is the target process serial number. On output how targetsPSN is
* interpeted is determined by type of connection. There are five options as of 7.0aA7.
* ¥ err is error. Means something.
*/
ConnectionType
GetConnectionType(ProcessSerialNumberPtr SendersPSN, unsigned long receiverID, unsigned long postingOptions, unsigned long *connectionID, ProcessSerialNumberPtr targetsPSN, OSErr *err)
{
PEntryPtr pReceiverProc;
EPPCBlkPtr eppcBPtr;
TargetID tID;
ProcessSerialNumber PSNFromSign;
ConnectionType retVal;
cTableEntryPtr sRec;
/* Assume the worst */
retVal = badConnection;
/* Sender must be able to do EPPC if a return receipt was requested
* NOTE: nReturnReceipt check is first because kSystemProcess may send
* an event when pCurrentProcess is nil.
*/
if ( ((postingOptions & nReturnReceipt) != 0) && (CanDoEPPC(pCurrentProcess) == false) )
{
*err = connectionInvalid;
*connectionID = noSessionID;
SetPSN(kNoProcess, targetsPSN);
return(retVal);
}
/* Determine specific connection type. */
if ((postingOptions &= receiverIDMask) IS receiverIDisPSN)
{
pReceiverProc = PEntryFromPSN((ProcessSerialNumberPtr) receiverID);
if (pReceiverProc != nil)
{
eppcBPtr = EPPCBlkFromPSN(&pReceiverProc->p_serialNumber); // <13>
if (eppcBPtr ISNT nil) { /* yes, target has eppcBlk */
/* Now consider the sender's identity */
if (EqualPSNConst(kSystemProcess, SendersPSN) == false)
{
tID = getTIDfromEppcBlk(eppcBPtr);
if ((*connectionID = getSessionID(SendersPSN,&tID, err)) != 0)
{
*targetsPSN = *((ProcessSerialNumberPtr) receiverID);
retVal = eppcConnection;
}
else /* we couldn't get a sessionID */
{
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
else
{ /* SendersPSN was kSystemProcess and target has eppcBlk */
*err = noErr;
*connectionID = systemSessionID;
*targetsPSN = *((ProcessSerialNumberPtr) receiverID);
retVal = systemSenderConnection;
}
}
else /* target had no eppcBlk. */
{
*err = destPortErr;
*connectionID = noSessionID;
*targetsPSN = *((ProcessSerialNumberPtr) receiverID);
retVal = noEppcConnection;
}
}
else /* we didn't find a process table for the target. See if it is a special case */
if ( EqualPSNConst(kSystemProcess, (ProcessSerialNumberPtr)(receiverID)) )
{
*err = noErr;
*connectionID = systemSessionID;
*targetsPSN = *((ProcessSerialNumberPtr) receiverID);
retVal = systemReceiverConnection;
}
else /* not a special case. Oh wellÉ not connection */
{
*err = procNotFound;
*connectionID = noSessionID;
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
if (postingOptions IS receiverIDisTargetID)
{
if ((*connectionID = getSessionID(SendersPSN,(TargetIDPtr)receiverID, err)) != 0)
{
retVal = eppcConnection;
}
}
if (postingOptions IS receiverIDisSignature)
{
if (PSNFromSignature((ResType)receiverID, &PSNFromSign))
{
eppcBPtr = EPPCBlkFromPSN(&PSNFromSign);
if (eppcBPtr ISNT nil) {
if (EqualPSNConst(kSystemProcess, SendersPSN) == false)
{
tID = getTIDfromEppcBlk(eppcBPtr);
if ((*connectionID = getSessionID(SendersPSN,&tID, err)) != 0)
{
*targetsPSN = PSNFromSign;
retVal = eppcConnection;
}
else
{
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
else /* SendersPSN was kSystemProcess and target has eppcBlk */
{
*err = noErr;
*connectionID = systemSessionID;
*targetsPSN = PSNFromSign;
retVal = systemSenderConnection;
}
}
else /* target had no eppcBlk. */
{
*err = destPortErr;
*connectionID = noSessionID;
*targetsPSN = PSNFromSign;
retVal = noEppcConnection;
}
}
else /* couldn't find a PSN from the signature. check special case */
{
if ( NULLPROC_SIGNATURE IS (ResType)receiverID )
{
*err = noErr;
*connectionID = systemSessionID;
SetPSN(kSystemProcess, targetsPSN);
retVal = systemReceiverConnection;
}
else /* not a special case. Oh wellÉ not connection */
{
*err = connectionInvalid;
*connectionID = noSessionID;
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
} /* end of receiverIDisSignature */
if (postingOptions IS receiverIDisSessionID)
{
sRec = findSessionRecord(receiverID);
if (sRec ISNT nil)
{
if (EqualPSNConst(kSystemProcess, SendersPSN) == false)
{
if ((*connectionID = validateSessionID(SendersPSN, receiverID, err)) != 0)
{
SetPSN(kNoProcess, targetsPSN);
retVal = eppcConnection;
}
else
{
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
else
{
*err = noErr;
*connectionID = systemSessionID;
*targetsPSN = sRec->sEppcBlkPtr->pTablePtr->p_serialNumber;
retVal = systemSenderConnection;
}
}
else /* no connection. check for special cases */
{
if (receiverID IS systemSessionID)
{
*err = noErr;
*connectionID = systemSessionID;
SetPSN(kSystemProcess, targetsPSN);
retVal = systemReceiverConnection;
}
else /* not a special case. Oh wellÉ not connection */
{
*err = connectionInvalid;
*connectionID = noSessionID;
SetPSN(kNoProcess, targetsPSN);
retVal = badConnection;
}
}
} /* end of receiverIDisSessionID */
return (retVal);
}
/*____________________________________________________________________________________________*/
unsigned long
validateSessionID(ProcessSerialNumberPtr ownersPSN, unsigned long sessionID, OSErr *err)
{
cTableEntryPtr sRec;
sRec = findSessionRecord(sessionID);
if (sRec ISNT nil)
if ( EqualPSN(ownersPSN, &sRec->sEppcBlkPtr->pTablePtr->p_serialNumber))
{
if (sRec->status & (sessionClosed+sessionEnded))
{
qset(&connectionQ);
llhead();
llsetMatch((ProcPtr)findElement);
if (llcheck(sRec))
{
lldelete();
DisposHandle(sRec->sHdl);
}
*err = sessClosedErr;
return(0);
}
*err = noErr;
return(sessionID);
}
*err = connectionInvalid;
return(0);
}
/*____________________________________________________________________________________________*/
Boolean
comparePortNames(PPCPortPtr N1, PPCPortPtr N2)
{
short i;
if (N1->nameScript ISNT N2->nameScript)
return(true);
if (N1->name[0] ISNT N2->name[0])
return(true);
if (N1->name[0]) {
for(i=1;i<=(N1->name[0]);++i)
if (N1->name[i] ISNT N2->name[i])
return(true);
}
if (N1->portKindSelector ISNT N2->portKindSelector)
return(true);
if (N1->portKindSelector IS ppcByString) {
if (N1->u.portTypeStr[0] ISNT N2->u.portTypeStr[0])
return(true);
if (N1->u.portTypeStr[0]) {
for(i=1;i<=(N1->u.portTypeStr[0]);++i)
if (N1->u.portTypeStr[i] ISNT N2->u.portTypeStr[i])
return(true);
}
}
else {
if (N1->u.port.creator ISNT N2->u.port.creator)
return(true);
if (N1->u.port.type ISNT N2->u.port.type)
return(true);
}
return(false);
}
/*____________________________________________________________________________________________*/
Boolean
comparePortLocation(LocationNamePtr L1, LocationNamePtr L2)
{
short i;
if (L1->locationKindSelector ISNT L2->locationKindSelector)
return(true);
if (L1->locationKindSelector IS ppcNoLocation)
return(false);
if (L1->u.nbpEntity.objStr[0] ISNT L2->u.nbpEntity.objStr[0])
return(true);
if (L1->u.nbpEntity.objStr[0]) {
for(i=1;i<=(L1->u.nbpEntity.objStr[0]);++i)
if (L1->u.nbpEntity.objStr[i] ISNT L2->u.nbpEntity.objStr[i])
return(true);
}
if (L1->u.nbpEntity.typeStr[0] ISNT L2->u.nbpEntity.typeStr[0])
return(true);
if (L1->u.nbpEntity.typeStr[0]) {
for(i=1;i<=(L1->u.nbpEntity.typeStr[0]);++i)
if (L1->u.nbpEntity.typeStr[i] ISNT L2->u.nbpEntity.typeStr[i])
return(true);
}
if (L1->u.nbpEntity.zoneStr[0] ISNT L2->u.nbpEntity.zoneStr[0])
return(true);
if (L1->u.nbpEntity.zoneStr[0]) {
for(i=1;i<=(L1->u.nbpEntity.zoneStr[0]);++i)
if (L1->u.nbpEntity.zoneStr[i] ISNT L2->u.nbpEntity.zoneStr[i])
return(true);
}
return(false);
}
/*____________________________________________________________________________________________*/
Boolean
PortNameAndLocationAreEqual(PPCPortPtr portName, LocationNamePtr locName, TargetIDPtr targetID)
{
Boolean different;
if ((different = comparePortNames(portName, &targetID->name)) == false)
if (locName IS nil)
return (targetID->location.locationKindSelector IS ppcNoLocation);
else
different = comparePortLocation(locName, &targetID->location);
return(!different);
}
/*____________________________________________________________________________________________*/
TargetID
getTIDfromEppcBlk(EPPCBlkPtr eppcBPtr)
{
TargetID tID;
MemClear(&tID,sizeof(TargetID));
tID.name = eppcBPtr->nameOfMsgQ;
tID.location.locationKindSelector = ppcNoLocation;
return(tID);
}
/*____________________________________________________________________________________________*/
cTableEntryPtr
findSessionRecord(unsigned long sessionRefNum)
{
cTableEntryPtr sRec = nil;
if (sessionRefNum IS 0 OR sessionRefNum IS -1)
return(sRec);
qset(&connectionQ);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)bySessionID);
llhead();
if (llcheck((char *)&sessionRefNum))
llretrieve((char *)&sRec);
}
return(sRec);
}
/*____________________________________________________________________________________________*/
#if !CubeE // <24> Keep the old version around for
cTableEntryPtr // <24> Non CubeE builds for now.
findFreeSessionRecord(void)
{
Handle aHdl;
OSErr err;
aHdl = GMBlk(sizeof(cTableEntry), &err);
if (err ISNT noErr)
return(nil);
HLock(aHdl);
MemClear(*aHdl,sizeof(cTableEntry));
((cTableEntryPtr)(*aHdl))->sHdl = aHdl;
return(*aHdl);
}
#endif
#if CubeE // <24> The ÒfixedÓ version from 7-Up
cTableEntryPtr
findFreeSessionRecord(void)
{
Handle aHdl;
aHdl = NewHandleSysClear(sizeof(cTableEntry)); // <24> Allocate the block in the System heap
if (!aHdl)
return(nil);
MoveHLow(aHdl); // <24> Move it low in the System heap
HLock(aHdl); // <24> before locking it.
((cTableEntryPtr)(*aHdl))->sHdl = aHdl;
return(*aHdl);
}
#endif
/*____________________________________getSessionID_____________________________________________*/
unsigned long
getSessionID(ProcessSerialNumberPtr sendersPSN, TargetIDPtr portN, OSErr *err)
{
PPCStartPBRec pbS;
cTableEntryPtr sRec;
TargetID senderTID;
Boolean portMatchfound = false;
Boolean enableGuestButton;
Str255 userName;
Boolean guestSelected;
EPPCBlkPtr postersEppcBlk;
Ptr funcPtr=StartSecureSession;
unsigned long myA5;
*err = noErr;
if ((postersEppcBlk = EPPCBlkFromPSN(sendersPSN)) IS nil)
{
*err = noPortErr;
return(0);
}
if (postersEppcBlk->createErr)
{
*err = postersEppcBlk->createErr;
return(0);
}
/* see if connection already exists */
senderTID = getTIDfromEppcBlk(postersEppcBlk);
qset(&connectionQ);
llhead();
if (ll_length() ISNT 0)
llretrieve(&sRec);
else
sRec = nil;
while (sRec)
{
if (sRec->sID)
{
if (sRec->sEppcBlkPtr IS postersEppcBlk)
if (PortNameAndLocationAreEqual(&sRec->sPName, &sRec->sLocName, portN))
{
portMatchfound = true;
break;
}
else
if (PortNameAndLocationAreEqual(&sRec->sEppcBlkPtr->nameOfMsgQ, nil, portN))
if (PortNameAndLocationAreEqual(&sRec->sPName, &sRec->sLocName, &senderTID))
{
portMatchfound = true;
break;
}
}
if (llnext())
llretrieve(&sRec);
else
sRec = nil;
}
if (portMatchfound)
if (sRec->status & (sessionClosed+sessionEnded))
{
*err = sessClosedErr;
lldelete();
DisposHandle(sRec->sHdl);
return(0);
}
else
return(sRec->sID);
/* connection didn't exist. See if we have a free one to give out. */
if ((sRec = findFreeSessionRecord()) IS nil)
{
*err = sessTableErr; /* session table has no free entries */
return (0);
}
MemClear(&pbS,sizeof(PPCStartPBRec));
pbS.csCode = ppcStartCmd;
pbS.portRefNum = postersEppcBlk->portID;
pbS.ioCompletion = nil;
pbS.serviceType = ppcServiceRealTime;
if (portN->location.locationKindSelector IS ppcNBPLocation)
pbS.locationName = &portN->location;
else
pbS.locationName = nil;
pbS.portName = &portN->name;
/* The following if statement causes noUserInteractionAllowed to be returned if we are */
/* establishing a remote connection and the application is not frontmost or is being */
/* coerced. */
/* For 7.0B6 the check for the application being coerced was removed. A background app */
/* may want to come to the foreground and post a HLE which requires a session to be */
/* established. The app may still be in the coerced state, yet pFrontProcess IS */
/* pCurrentProcess and the app is in the frontmost layer of windows. At this point */
/* it should be OK for authentication to occur. Hopefully.
/* This does not deal with the app filling in the NBP name of his local machine. */
#if 0
if ((pbS.locationName ISNT nil) AND ((pFrontProcess ISNT pCurrentProcess) OR (coercionState ISNT CS_DONE)))
#else
if ((pbS.locationName ISNT nil) AND (pFrontProcess ISNT pCurrentProcess))
#endif
{
DisposHandle(sRec->sHdl);
*err = noUserInteractionAllowed;
return(0);
}
enableGuestButton = guestAllowed(portN, err);
if (*err ISNT noErr)
{
DisposHandle(sRec->sHdl);
return (0);
}
MemClear(&userName,sizeof(Str255));
/* StartSecureSession may cause the Dialog Manager to run. It is best if our
app's A5 is in the machine because user items in dialog windows behind the
authentication dialog(s) may execute. Some user item procedure assume that
A5 points to the application's global not the Process Manager's globals.
*/
myA5 = CurrentA5SimpleSetup();
*err = CALL_FNC_PTR(StartSecureSession_ptr, funcPtr, (&pbS, &userName, false, enableGuestButton, &guestSelected, nil));
A5SimpleRestore(myA5);
/*
*err = StartSecureSession (&pbS, &userName, false, enableGuestButton, &guestSelected, nil);
*/
if (*err ISNT noErr)
{
DisposHandle(sRec->sHdl);
return (0);
}
/* delete user identity. eppc doesn't use it. deleting it frees PPCToolbox resources. */
DeleteUserIdentity (pbS.userRefNum);
/* fill in the connection entry */
sRec->status |= startConnection;
sRec->sID = pbS.sessRefNum;
sRec->sEppcBlkPtr = postersEppcBlk;
sRec->sportID = postersEppcBlk->portID;
sRec->sPName = portN->name;
if (portN->location.locationKindSelector IS ppcNBPLocation)
sRec->sLocName = portN->location;
else
sRec->sLocName.locationKindSelector = ppcNoLocation;
sRec->srPB = startPPCRead(postersEppcBlk, sRec->sID, err);
if (sRec->srPB IS nil)
{
DisposHandle(sRec->sHdl);
return(0);
}
else
{
qset(&connectionQ);
q_push(sRec);
}
return(sRec->sID);
}
/*_________________________________restartPPCRead_____________________________________________*/
PPCReadPBPtr
restartPPCRead(ppcReadPBPtr rPB, OSErr *err)
{
MemClear(&rPB->mfMsgBlk,sizeof(MFmsgBlk));
rPB->mfMsgBlk.pbHandle = rPB->pbHdl;
*err = PPCRead(&rPB->ppcReadPB,ASYNCHRONOUS);
return(&rPB->ppcReadPB);
}
/*___________________________________startPPCRead_____________________________________________*/
PPCReadPBPtr
startPPCRead(EPPCBlkPtr eppcBPtr, unsigned long sessionID, OSErr *err)
{
PPCReadPBPtr rPB;
rPB = getPPCReadPB(eppcBPtr, sessionID, err);
if (*err IS noErr)
*err = PPCRead(rPB,ASYNCHRONOUS);
return(rPB);
}
/*____________________________________________________________________________________________*/
Boolean
guestAllowed(TargetIDPtr portN, OSErr *err)
{
PortInfoRec infoBuff;
IPCListPortsPBRec listPortPB;
*err = noErr;
if (portN->location.locationKindSelector IS ppcNoLocation)
return(false);
MemClear(&listPortPB,sizeof(IPCListPortsPBRec));
listPortPB.csCode = IPCListPortsCmd;
listPortPB.requestCount = 1;
listPortPB.portName = &portN->name;
listPortPB.locationName = &portN->location;
listPortPB.bufferPtr = &infoBuff;
*err = IPCListPorts(&listPortPB,SYNCHRONOUS);
if (*err ISNT noErr)
return(false);
if (listPortPB.actualCount IS 1)
return (!infoBuff.authRequired);
else
{
*err = destPortErr;
return(false);
}
}
/*____________________________________________________________________________________________*/
Boolean
findElement(char *A, char *B)
{
return((*(LINKTYPE *)A).item IS (*(LINKTYPE *)B).item);
}
/*____________________________________________________________________________________________*/
Boolean
byPSN(EPPCBlkPtr A, ProcessSerialNumberPtr B)
{
return(EqualPSN(&(A->pTablePtr->p_serialNumber),B));
}
/*____________________________________________________________________________________________*/
Boolean
byPortName(EPPCBlkPtr A, PPCPortPtr B)
{
return(!comparePortNames(&A->nameOfMsgQ,B));
}
/*____________________________________________________________________________________________*/
Boolean
bySignature(EPPCBlkPtr A, ResType *B)
{
return(A->pTablePtr->p_signature IS *B);
}
/*____________________________________________________________________________________________*/
Boolean
byPBID(ppcWritePBPtr A, EPPCBlk *B)
{
return(A->pbID IS B);
}
/*____________________________________________________________________________________________*/
Boolean
bySessionID(cTableEntryPtr A, unsigned long *B)
{
return(A->sID IS *B);
}
/*____________________________________________________________________________________________*/
Boolean
bysEppcBlkPtr(cTableEntryPtr A, EPPCBlk *B)
{
return(A->sEppcBlkPtr IS B);
}
/*____________________________________________________________________________________________*/
Boolean
byportIDforPB(ppcWriteBlk *A, unsigned long *B)
{
return(A->portID IS *B);
}
/*____________________________________________________________________________________________*/
Boolean
byportID(cTableEntryPtr A, unsigned long *B)
{
return(A->sportID IS *B);
}
/*____________________________________________________________________________________________*/
Boolean
PSNFromName(PPCPortPtr portName, ProcessSerialNumberPtr pPSN)
{
EPPCBlkPtr eppcB = nil;
Boolean retVal;
qset(&eppcBQ);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)byPortName);
llhead();
if (llcheck(portName))
llretrieve((char *)&eppcB);
}
/* Set output parameter and function value */
if (retVal = (eppcB != nil))
*pPSN = eppcB->pTablePtr->p_serialNumber;
else
SetPSN(kNoProcess, pPSN);
return(retVal);
}
/*____________________________________________________________________________________________*/
Boolean
NameFromPSN(PPCPortPtr portName, ProcessSerialNumberPtr pPSN)
{
EPPCBlkPtr eppcB = nil;
Boolean retVal;
qset(&eppcBQ);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)byPSN);
llhead();
if (llcheck(pPSN))
llretrieve((char *)&eppcB);
}
/* Set output parameter and function value */
if (retVal = (eppcB != nil))
*portName = eppcB->nameOfMsgQ;
return(retVal);
}
/*____________________________________________________________________________________________*/
Boolean
PSNFromSignature(ResType qid, ProcessSerialNumberPtr pPSN)
{
EPPCBlkPtr eppcB = nil;
Boolean retVal;
qset(&eppcBQ);
if (ll_length() ISNT 0)
{
llsetMatch((ProcPtr)bySignature);
llhead();
if (llcheck((char *)&qid))
llretrieve((char *)&eppcB);
}
/* Set output parameter and function value */
if (retVal = (eppcB != nil))
*pPSN = eppcB->pTablePtr->p_serialNumber;
else
SetPSN(kNoProcess, pPSN);
return(retVal);
}
/*____________________________________________________________________________________________*/
EPPCBlkPtr
EPPCBlkFromPSN(ProcessSerialNumberPtr pPSN)
{
qset(&eppcBQ);
if (ll_length() IS 0)
return(nil);
llsetMatch((ProcPtr)byPSN);
llhead();
if (llcheck((char *)pPSN))
return(eppcBQ.clp);
else
return(nil);
}
/*____________________________________________________________________________________________*/
void
releasePPCpb(PPCParamBlockPtr pb)
{
if (pb IS nil)
return;
(Ptr)pb -= sizeOfPPCpbHeader;
DisposHandle(((ppcReadPBPtr)pb)->pbHdl);
CheckForFatalError(MEMERROR IS noErr);
}
/*____________________________________________________________________________________________*/
PPCInformPBPtr
getPPCInformPB(EPPCBlkPtr refCon, unsigned short portID, OSErr *err)
{
Handle aHdl;
ppcInformPBPtr ppcInformBlkPtr;
/* carve out a parameter block for a async PPC call */
aHdl = GMBlk(sizeof(ppcInformBlk), err);
if (*err ISNT noErr)
return(nil);
HLock(aHdl);
ppcInformBlkPtr = *aHdl;
MemClear(ppcInformBlkPtr,sizeof(ppcInformBlk));
ppcInformBlkPtr->pbID = refCon;
ppcInformBlkPtr->portID = portID;
ppcInformBlkPtr->pbHdl = aHdl;
ppcInformBlkPtr->ppcInformPB.portName = &ppcInformBlkPtr->portName;
ppcInformBlkPtr->ppcInformPB.locationName = &ppcInformBlkPtr->portLoc;
ppcInformBlkPtr->ppcInformPB.csCode = ppcInformCmd;
ppcInformBlkPtr->ppcInformPB.ioCompletion = KnockKnock;
ppcInformBlkPtr->ppcInformPB.portRefNum = portID;
ppcInformBlkPtr->ppcInformPB.autoAccept = false;
return(&ppcInformBlkPtr->ppcInformPB);
}
/*____________________________________________________________________________________________*/
PPCReadPBPtr
getPPCReadPB(EPPCBlkPtr refCon, unsigned long sessionID, OSErr *err)
{
Handle aHdl;
ppcReadPBPtr ppcReadBlkPtr;
/* carve out a parameter block for a async PPC call */
aHdl = GMBlk(sizeof(ppcReadBlk), err);
if (*err ISNT noErr)
return(nil);
HLock(aHdl);
ppcReadBlkPtr = *aHdl;
MemClear(ppcReadBlkPtr,sizeof(ppcReadBlk));
ppcReadBlkPtr->pbID = refCon;
ppcReadBlkPtr->portID = refCon->portID;
ppcReadBlkPtr->pbHdl = aHdl;
ppcReadBlkPtr->ppcReadPB.bufferPtr = &ppcReadBlkPtr->mfMsgBlk.eppcMsgBlk;
ppcReadBlkPtr->ppcReadPB.bufferLength = sizeof(HighLevelEventMsg);
ppcReadBlkPtr->ppcReadPB.csCode = ppcReadCmd;
ppcReadBlkPtr->ppcReadPB.ioCompletion = KnockKnock;
ppcReadBlkPtr->ppcReadPB.sessRefNum = sessionID;
return(&ppcReadBlkPtr->ppcReadPB);
}
/*____________________________________________________________________________________________*/
extern LINKLIST *list;
/*____________________________________________________________________________________________*/
void
qPPCPb(ppcWritePBPtr wPBPtr, LINKLIST *BQList)
{
LINKLIST *oldList = list;
assert(!BQList->listInUse);
qset(BQList);
q_push(wPBPtr);
list = oldList;
}
/*____________________________________________________________________________________________*/
void
KnockKnock( PPCParamBlockPtr ppcPb )
{
ppcWritePBPtr wPBPtr;
ppcInformPBPtr iPBPtr;
wPBPtr = (ppcWritePBPtr)((Ptr)ppcPb - (Ptr)sizeOfPPCpbHeader);
switch(ppcPb->startParam.csCode) {
case ppcInformCmd:
/* closing the port causes all async calls to go to completion */
if (ppcPb->informParam.ioResult ISNT noErr)
{
qPPCPb(wPBPtr, &ppcInformBQ);
break;
}
iPBPtr = (ppcInformPBPtr)wPBPtr;
iPBPtr->ppcInformReplyPB.ioCompletion = KnockKnock;
iPBPtr->ppcInformReplyPB.sessRefNum = ppcPb->informParam.sessRefNum; // <13>
/* if close is in progress on the port reject the incoming session */
if (wPBPtr->status & closeInProgress)
PPCReject(&iPBPtr->ppcInformReplyPB,ASYNCHRONOUS);
else
PPCAccept(&iPBPtr->ppcInformReplyPB,ASYNCHRONOUS);
break;
case ppcWriteCmd:
qPPCPb(wPBPtr, &ppcWriteBQ);
break;
case ppcReadCmd:
wPBPtr->mfMsgBlk.eppcMsgBlk.theMsgEvent.when = TICKS;
qPPCPb(wPBPtr, &ppcReadBQ);
break;
case ppcAcceptCmd:
case ppcRejectCmd:
iPBPtr = (ppcInformPBPtr)((Ptr)wPBPtr - (Ptr)sizeof(PPCInformPBRec));
qPPCPb(iPBPtr, &ppcInformBQ);
break;
case ppcOpenCmd:
case ppcStartCmd:
case ppcEndCmd:
case ppcCloseCmd:
case IPCListPortsCmd:
default:
dbmsg("Eppc completion routine called for unknown command.");
break;
}
}