mac-rom/ProcessMgr/ScrapCoercion.c

1132 lines
34 KiB
C
Raw Normal View History

/*
File: ScrapCoercion.c
Contains: Routines that implement scrap coercion on foreground process switches.
Written by: Phil Goldman and Erich Ringewald
Copyright: <EFBFBD> 1986-1992 by Apple Computer, Inc., all rights reserved.
Change History (most recent first):
<17> 4/9/92 KST #1027029,<JH>: Inform TSM when a non-TSM aware application is
getting suspended so TSM can pull down input method's menu.
<16> 3/23/92 JSM OSEvents.h is obsolete, use Events.h.
<15> 2/25/91 DFH DC,WS#DFH-910225a: Fixed so GetScrap works even if called before
first event call. IsScrapOwnedByCurrentProcess did not allow for
case where MoveScrap() has not been called on the current app.
<14> 1/28/91 DFH JSM,#81718:Fixed OrphanScrapIntoPartition to deal with case
where SCRAPHANDLE is nil because scrap is on the disk.
<12> 1/8/91 DFH (rdd) Changed StartForegroundSwitch to allow for case where
current front process is suspended for high-level debugging.
<11> 12/21/90 DFH Fixed SetFirstDstCoercionState to set srcMyScrapInfo.pScrapOwner
= pCurrentProcess when dst scrap is already up to date.
<10> 12/18/90 DFH Added GetScrap patch support routine.
<9> 12/18/90 DFH Removed fgAppsUnexecuted mechanism, since front switch queue
works better.
<8> 12/17/90 DFH Fixed scrap retention devices.
<7> 12/15/90 DFH Propagate correct scrap, even if some destination process
doesn't have enough memory for it. True even if real scrap owner
ExitToShell's.
<6> 12/11/90 DFH CancelQuitAll now called only when FG switching out of intended
victim. Other switches can be normal (e.g. when intended victim
is a BG-only application).
<5> 12/10/90 DFH Compiler bug workaround in MoveScrap. Use SCRAPCOUNT += 1
instead of SCRAPCOUNT++.
<4> 11/14/90 DFH Change ScrapIO to use PBOpenDF instead of PBOpen.
<3> 11/6/90 DFH Use structure defs for menu bar access in CS_INIT.
<0> x/xx/86 PYG New Today.
*/
#include <types.h>
#include <memory.h>
#include <desk.h>
#include <menus.h>
#include <quickdraw.h>
#include <palette.h>
#include <osutils.h>
#include <events.h>
#include <windows.h>
#include <files.h>
#include <errors.h>
#include <TSMprivate.h>
#include <TextServices.h>
#include "Glue.h"
#include "Lomem.h"
#include "Data.h"
#include "SysMisc.h"
#include "Patches.h"
#include "Puppet.h"
#include "AppleEventExtensions.h"
#pragma segment kernel_segment
static long resume_message; /* the message field of the resume event */
static short eatMouseTimeout; /* counter of gne tries to eat mouse-down in CS_EATMOUSE */
static short clickDAWindTimeout; /* counter of gne tries to force click in fake DA window */
/* Function prototypes that shoule be in (yet another) header file */
void BringProcessToFront(PEntryPtr);
/* Function prototypes defined and used in this file */
void SetThirdSrcCoercionState(void);
void SetFirstDstCoercionState(void);
short FindCutCopyItems(MenuHandle);
void MoveScrap(void);
void ScrapIO(short, StringPtr, Ptr, u_size);
void ActivateTopWindow(short);
void DisposeOrphanedScrap(void);
/* ActivateTopWindow. Invert the hiliting the front window when the current application
* is the kind that does the rest of the deactivate/activate itself.
* NOTE: Assumes A5 = PROCESSMGRGLOBALS
*/
void
ActivateTopWindow(short activateMods)
{
WindowPtr frontWindow;
EventRecord dummyEvent;
assert(pCurrentProcess->p_layer != nil);
if ((frontWindow = FrontWindowIn(pCurrentProcess->p_layer)) != nil)
{
/* Highlight the window and make sure the palette agrees. */
HiliteWindow(frontWindow, (activateMods != 0));
if (Colorized && ((activateMods & activeFlag) != 0))
ActivatePalette(frontWindow);
/* If it's a DA window and activate events are allowed, give the DA an activate event */
if (((WindowPeek) frontWindow)->windowKind < 0 && SEVTENB != 0)
{
dummyEvent.what = activateEvt;
dummyEvent.message = (unsigned long)frontWindow;
dummyEvent.modifiers = activateMods;
(void) SystemEvent(&dummyEvent);
}
}
}
/* PutUpFakeWindow. Puts up a window way offscreen to generate de-activate events.
* NOTE: Have to have at least one character in name string just in case someone
* tries to stick this name in a menu (null strings are not allowed there).
* An app that does this: Insight Accounts Receivable
* Assumes A5 == PROCESSMGRGLOBALS
*/
void
PutUpFakeWindow(void)
{
WindowPtr fakeWindowPtr;
Rect fakeWindowRect;
short nullski = TWO_CHARS_TO_SHORT('\01',' ');
/* NOTE: Should use global here */
SET_RECT(&fakeWindowRect,16000,16000,16004,16004);
fakeWindowPtr = NewWindow(NULL, &fakeWindowRect,
&nullski, true, (short) 0, (WindowPtr) -1, true, 0L);
/* Tag it as a fake window by sticking a funny value in the windowKind field */
((WindowPeek) fakeWindowPtr)->windowKind = SCRAP_COERCION_FAKE_DA_WINDOW_ID;
pCurrentProcess->p_wptr = fakeWindowPtr;
}
/* TakeDownFakeWindow. Takes down the fake DA window we put up earlier. Nothing fancy.
* Assumes A5 == PROCESSMGRGLOBALS
*/
void
TakeDownFakeWindow(void)
{
register WindowPeek fakeWindowPtr;
register PEntryPtr pCurrProc;
/* Does the window exist? If so, get rid of it. */
pCurrProc = pCurrentProcess;
if ((fakeWindowPtr = pCurrProc->p_wptr) != nil)
{
DisposeWindow(fakeWindowPtr);
pCurrProc->p_wptr = nil;
}
}
#define SCRAP_TRANSFER_SIZE 1024
#define DO_SCRAP_READ ((short) 0)
#define DO_SCRAP_WRITE ((short) 1)
/* MoveScrap. Moves the scrap from the source task to the destination task. The scrap
* of the source task is described by srcMyScrapInfo. The scrap of the destination
* task is described in lomem.
*/
void
MoveScrap(void)
{
u_size size;
/* Say we've tried at least once to give this process a scrap. It means that the
* scrap copied over during LaunchApplication is no longer to be used.
*/
pCurrentProcess->p_scrapIsNotFromLaunch = true;
/* Get/Set scrap size */
size = SCRAPINFO = srcMyScrapInfo.size;
/* See if the dst task scrap is in memory */
if (SCRAPHANDLE != nil)
{
/* Check whether the src task scrap is in mem */
if (srcMyScrapInfo.memHandle != nil)
{
if (PtrToXHand(*srcMyScrapInfo.memHandle, SCRAPHANDLE, size) != noErr)
{
SCRAPINFO = 0;
SetHandleSize(SCRAPHANDLE, 0);
}
}
/* The src task scrap is on disk */
else
{
SetHandleSize(SCRAPHANDLE, size);
if (MEMERROR == noErr)
ScrapIO(DO_SCRAP_READ, srcMyScrapInfo.fileName, *SCRAPHANDLE, size);
else
{
SCRAPINFO = 0;
SetHandleSize(SCRAPHANDLE, 0);
}
}
}
/* The dst task scrap is on disk */
else
{
/* See whether the src task scrap is in memory */
if (srcMyScrapInfo.memHandle != nil)
ScrapIO(DO_SCRAP_WRITE, SCRAPNAME, *srcMyScrapInfo.memHandle, size);
#ifdef DISK_TO_DISK_SCRAP
/* The src task scrap is on disk. I give up!
* NOTE: srcMyScrapInfo.vRefNum is never set. ScrapCopyOverVolumes doesn't exist.
*/
else
{
short vRefNumDst = BOOTVOL;
/* Do I/O if on separate volumes, otherwise just change size */
if (srcMyScrapInfo.vRefNum != vRefNumDst)
{
/* Files on different volumes */
ScrapCopyOverVolumes(srcMyScrapInfo.vRefNum,vRefNumDst);
}
}
#endif DISK_TO_DISK_SCRAP
}
/* If it worked, indicate that the current process' scrap is good, and get rid
* of the orphaned scrap (its job is done!).
*/
if (size == SCRAPINFO)
{
srcMyScrapInfo.pScrapOwner = pCurrentProcess;
DisposeOrphanedScrap();
}
/* Invalidate the scrap, and sync up local counter with global. */
SCRAPCOUNT += 1;
pCurrentProcess->p_cutcopycount = cutCopyCount;
}
/* IsScrapOwnedByCurrentProcess. Return whether we think that the current SCRAPINFO
* is valid. Essentially, the check is whether the most recent MoveScrap() failed
* and no new scrap has been made.
*/
Boolean
IsScrapOwnedByCurrentProcess(void)
{
PEntryPtr pCurrProc;
pCurrProc = pCurrentProcess;
return ((pCurrProc == srcMyScrapInfo.pScrapOwner) ||
(pCurrProc->p_cutcopycount != cutCopyCount) ||
(pCurrProc->p_scrapIsNotFromLaunch == false));
}
/* ScrapIO. Generic scrap file I/O based on the first parameter passed in */
void
ScrapIO(short ioMode, StringPtr scrapFileName, Ptr scrapPtr, u_size size)
{
IOParam IOParamLocal;
/* Do the open */
IOParamLocal.ioNamePtr = scrapFileName;
IOParamLocal.ioVRefNum = BOOTVOL; /* scrap is always on boot volume */
IOParamLocal.ioVersNum = (char) 0;
IOParamLocal.ioMisc = NULL;
IOParamLocal.ioPermssn = (ioMode == DO_SCRAP_READ) ? fsRdPerm : fsWrPerm;
(void) PBOpenDF((ParmBlkPtr) &IOParamLocal, SyncHFS);
/* Do the I/O */
IOParamLocal.ioBuffer = scrapPtr;
IOParamLocal.ioReqCount = size;
IOParamLocal.ioPosMode = fsFromStart;
IOParamLocal.ioPosOffset = 0L;
if (ioMode == DO_SCRAP_READ)
(void) PBRead((ParmBlkPtr) &IOParamLocal, SyncHFS);
else
(void) PBWrite((ParmBlkPtr) &IOParamLocal, SyncHFS);
/* Do the close */
(void) PBClose((ParmBlkPtr) &IOParamLocal, SyncHFS);
}
/* Coercion_State_Engine. This is the engine that drives the scrap coercion that
* brackets user-initiated switches between a source task (src) and a destination task
* (dst). It is procedurally driven, rather than table driven, based on the variable
* coercionState. The sequence of values that coercionState will have is a function of
* the PEntry of the src and dst tasks, in specific the modeNeedSuspendResume (in
* p_taskMode) and the p_cutcopycount field. These two values are used to determine new
* values of coercionState at certain key points in the scrap coercion. In between
* these key points new coercionState values are predetermined from the old ones.
*
* See the flowchart in file 'Scrap Coercion Flowchart' for a more visual explanation.
*
* The return value of the function is a word-sized boolean indicating whether the
* given EventRecord has been filled in with a non-nullEvent (i.e. like an event call).
*
* NOTE: Assumes A5 = PROCESSMGRGLOBALS
*/
#define APP_EVENTS_ONLY_MASK (app1Mask+app2Mask+app3Mask)
short
Coercion_State_Engine(unsigned short eventmask, EventRecord *theevent, Ptr oldtrap, Boolean pullevent)
{
short retval;
switch(coercionState)
{
/* First state seen by the incoming process */
case CS_ENTERING:
{
SetFirstDstCoercionState();
retval = wFalse;
break;
}
/* Outgoing app has finished coercion. We're ready to switch */
case CS_EXITING:
{
PEntryPtr pProc;
src_scrap_setup();
coercionState = CS_ENTERING;
pProc = pNewFrontProcess;
pNewFrontProcess = nil;
/* Make pProc frontmost, and switch to it */
BringProcessToFront(pProc);
retval = wFalse;
break;
}
/* Generate the deactivate, activate events caused by putting up window */
case CS_GENERATE_DEACTIVATE1:
case CS_GENERATE_DEACTIVATE2:
{
if (pullevent)
SetThirdSrcCoercionState();
retval = CALL_FNC_PTR(Eventcommon_ptr,oldtrap,(eventmask & activMask, theevent));
break;
}
case CS_MOVE_SCRAP:
{
MoveScrap();
nullTimeout = NULL_TIMEOUT_INIT;
coercionState = ((pCurrentProcess->p_taskmode & modeNeedSuspendResume) != 0)
? CS_RESUME : CS_GENERATE_NULLS_THEN_CMD_C;
/* sync up next time thru */
retval = wFalse;
break;
}
case CS_GENERATE_NULLS_THEN_CMD_V:
case CS_GENERATE_NULLS_THEN_CMD_C:
{
#define CMD_V (0x976)
#define CMD_C (0x863)
retval = wFalse;
/* Generating a few null events to make apps like Microsoft File happy */
if (--nullTimeout > NULL_TIMEOUT_CUTOFF)
break;
/* If it's OK to, convert the event to a cmd-C or cmd-V. First off,
* system events aren't allowed inside GNE if SEVTENB != 0.
*/
if ((pullevent == false) || (SEVTENB == (char) 0 && nullTimeout > 0))
{
/* Must have caller that wants key-downs */
if ((eventmask & keyDownMask) != 0)
{
/* Try to get outgoing app to post its scrap */
if (coercionState == CS_GENERATE_NULLS_THEN_CMD_V)
{
if (pullevent)
coercionState = CS_EXITING;
theevent->message = CMD_V;
}
/* Try to get new app to load the scrap. */
else
{
if (pullevent)
{
clickDAWindTimeout = CLICK_DA_WIND_TIMEOUT_INIT;
coercionState = CS_CLICK_IN_WINDOW;
}
theevent->message = CMD_C;
}
/* force a cmd-<something>
* NOTE: Why do we also say the mouse button is down?
*/
theevent->what = keyDown;
theevent->modifiers = cmdKey+btnState;
retval = wTrue;
}
}
/* We timed out waiting for SEVTENB to be clear. Abandon the attempt. */
else
{
/* See if we were in destination trying to load the scrap. */
if (coercionState == CS_GENERATE_NULLS_THEN_CMD_C)
{
coercionState = CS_CLICK_IN_WINDOW;
clickDAWindTimeout = CLICK_DA_WIND_TIMEOUT_INIT;
}
/* We were trying to get the outgoing app to post the scrap. */
else
coercionState = CS_EXITING;
}
break;
}
/* Fabricate a mouse-down in the fake DA window so we can close the window when
* the app calls _SystemClick (i.e. as if the user clicked the close box).
*/
case CS_CLICK_IN_WINDOW:
{
WindowPtr fakeDAWindowPtr;
retval = wFalse;
/* Must have caller that wants mousedown. Timeout if necessary. */
if ((eventmask & mDownMask) == 0)
{
if (--clickDAWindTimeout == 0)
coercionState = CS_TAKE_DOWN_WINDOW;
break;
}
/* Return the event */
coercionState = CS_TAKE_DOWN_WINDOW;
if ((fakeDAWindowPtr = pCurrentProcess->p_wptr) != nil)
{
theevent->what = mouseDown;
PT_TO_PT(&(*(((WindowPeek) fakeDAWindowPtr)->strucRgn))->rgnBBox.top, &theevent->where);
theevent->modifiers = 0;
retval = wTrue;
}
break;
}
/* We just gave a mousedown in the DA window. Either, the app called SystemClick
* and we closed the window, or he has made another event call and we can take
* down the window now. Look for an activate event on another window.
* NOTE: If we DO give back an activate, we'll come through here again, since we
* don't change the coercionState.
*/
case CS_TAKE_DOWN_WINDOW:
{
/* Just in case the app never calls _SystemClick */
TakeDownFakeWindow();
/* Try and force thru the activate event that should be generated since we just
* took down the fake window
*/
retval = CALL_FNC_PTR(Eventcommon_ptr,oldtrap,(eventmask & activMask, theevent));
/* Either there was no activate to get (e.g. no windows left) or we got it.
* In both cases forge ahead.
* NOTE: The 32-bit dirty check of the CURACTIVATE high bit checks whether the
* window update was canceled (i.e. <= 0 means no window update in progress).
*/
if ((long)CURACTIVATE <= 0)
{
eatMouseTimeout = EAT_MOUSE_TIMEOUT;
coercionState = CS_EATMOUSE;
}
break;
}
case CS_PRESUSPEND_DEACTIVATE:
/* Deactivate the top window and return a suspend event. This also
* results in a deactivate event later on.
*/
ActivateTopWindow(0);
/* and fall thru */
case CS_SUSPEND:
{
retval = wFalse;
/* fabricate a suspend event */
if ((eventmask & app4Mask) != 0)
{
theevent->what = app4Evt;
theevent->message = SUSPEND_MESSAGE;
retval = wTrue;
}
/* actually switch next time */
if (pullevent)
coercionState = CS_EXITING;
break;
}
case CS_RESUME:
{
/* Make transition to next state */
if (pullevent)
{
if ((pCurrentProcess->p_taskmode & modeDoesActivateOnFGSwitch) != 0)
{
/* treat resume as a activate evt too */
ActivateTopWindow(activeFlag);
eatMouseTimeout = EAT_MOUSE_TIMEOUT;
coercionState = CS_EATMOUSE;
}
else
coercionState = CS_TAKE_DOWN_WINDOW;
}
/* Generate a resume event for the app. Added bonus: Throw in scrap
* coercion flags and puppet string results into the .message field.
*/
retval = wFalse;
if ((eventmask & app4Mask) != 0)
{
theevent->what = app4Evt;
theevent->message = resume_message;
retval = wTrue;
}
break;
}
case CS_EATMOUSE:
{
/* About to leave scrap coercion so check if a mouse event
* should be flushed. Flush if mouse down in content rgn,
* close box, or zoom box of front window. This is most
* likely the mouse click that caused the FG switch so long ago,
* but not necessarily (since the FG switch might be programmatic
* or via Apple menu process list item).
* NOTE: Technically, w->hilited check should be w == FrontWindow(),
* but since many apps (e.g. SuperPaint) have tool windows, and
* highlight a menu farther along in the list, we do this hack.
* Question: What does this do in PageMaker, which has 2
* active windows?
*/
short windowPart;
WindowPtr pWindow;
Boolean allDone = false;
/* If there is a mousedown, process it */
if (retval = CALL_FNC_PTR(Eventcommon_ptr, oldtrap,(eventmask & mDownMask, theevent)))
{
windowPart = FindWindow(theevent->where, &pWindow);
if ((pWindow != nil)
&& (windowPart != inDrag)
&& (windowPart != inDesk)
&& (((WindowPeek) pWindow)->hilited)
&& (((windowPart == inGoAway) || (windowPart == inZoomIn) || (windowPart == inZoomOut))
|| (pCurrentProcess->p_taskmode & modeGetFrontClicks) == 0) )
{
if (pullevent == false)
FlushEvents(mDownMask, mUpMask);
FlushEvents(mUpMask, everyEvent-mUpMask);
retval = wFalse;
}
/* Indicate that this click brought the app to the front */
else
theevent->modifiers |= activeFlag;
/* That's all folks! */
allDone = pullevent;
}
/* We're done if this was GetNextEvent, or we timed out with EventAvails */
else
allDone = (pullevent || ((eventmask & mDownMask) != 0) || (--eatMouseTimeout == 0));
/* High-level switch is complete. If we've switched into a scratchpad task,
* now is a good time to set its state to receive the puppet strings.
*/
if (allDone)
{
/* Let process see real events for a change */
coercionState = CS_DONE;
/* Well<6C> maybe a few more fake ones! */
if (pCurrentProcess->p_puppetvars.puppetMasterProc != nil)
pCurrentProcess->p_state = PRPUPPET;
/* Are we in the middle of a system QuitAll? */
if (pSystemQuitAllMsg != nil)
{
PEntryPtr pTargetProc;
/* Can continue the quitall if we are switching back to the
* originator after a victim has died. Allow initial switch to
* intended victim.
*/
pTargetProc = PEntryFromPSN(&(pSystemQuitAllMsg->targetmfid.localPSN));
if ( (pTargetProc == nil) && EqualPSN(&(pSystemQuitAllMsg->sendermfid.localPSN), &pCurrentProcess->p_serialNumber) )
ContinueQuitAll();
}
}
break;
}
/* Initial coercion state of a process. This code should only be reached the
* very first time an app makes an event call in the foreground.
*/
case CS_INIT:
{
register MenuPair *pThisMenuPair;
register Ptr limitPtr;
MenuHandle menuHdl;
/* Check if process can do cut or copies, but only if there is a menu list. */
pCurrentProcess->p_haseditcmdkeys = 0;
if ((MENULIST != nil) && (*MENULIST != nil))
{
/* Check likely edit menu first. Scan remainder only on no-find. */
if ((menuHdl = GetMHandle(3)) && FindCutCopyItems(menuHdl))
pCurrentProcess->p_haseditcmdkeys = 1;
else
{
pThisMenuPair = &((MenuListPtr) (*MENULIST))->menuPairs[0];
limitPtr = *MENULIST + ((MenuListPtr) (*MENULIST))->lastMenu;
while (pThisMenuPair <= limitPtr)
{
/* Check this menu */
if (FindCutCopyItems(pThisMenuPair->menuHdl))
{
pCurrentProcess->p_haseditcmdkeys = 1;
break;
}
/* Look at next menu */
pThisMenuPair += 1;
}
}
}
/* Set up process with the current scrap. */
MoveScrap();
/* Put process in state to receive scratchpad events if the lad has
* already acquired a master. Poor guy.
*/
coercionState = CS_DONE;
if (pCurrentProcess->p_puppetvars.puppetMasterProc != nil)
pCurrentProcess->p_state = PRPUPPET;
retval = CALL_FNC_PTR(Eventcommon_ptr,oldtrap,(eventmask, theevent));
break;
}
}
return(retval);
}
#pragma segment Main
/* AddToFrontProcessQueue. Puts pProc at the end of the waiting list for being switched
* to the foreground. Uses a pretty crude queue, on the assumption that we're not going
* to have a very long list.
* NOTE: This could be made more VM friendly, by a) skipping RemoveFromFrontProcessQueue
* if we could tell that pProc wasn't in the queue, and b) using a tail pointer.
*/
void
AddToFrontProcessQueue(PEntryPtr pProc)
{
PEntryPtr pCurr, pPrev;
/* Remove pProc if it is already queued */
RemoveFromFrontProcessQueue(pProc);
/* Scan the list to find the end */
pPrev = nil;
pCurr = pFrontProcessQueue;
while (pCurr != nil)
{
pPrev = pCurr;
pCurr = pCurr->p_nextQueuedFrontProcess;
}
/* Now, append to the list */
pProc->p_nextQueuedFrontProcess = nil;
if (pPrev == nil)
pFrontProcessQueue = pProc;
else
pPrev->p_nextQueuedFrontProcess = pProc;
}
/* PopFrontProcessQueue. Extract and return the first process to bring to the front. */
PEntryPtr
PopFrontProcessQueue(void)
{
PEntryPtr pProc;
/* Look at the queue */
if ((pProc = pFrontProcessQueue) != nil)
{
pFrontProcessQueue = pProc->p_nextQueuedFrontProcess;
pProc->p_nextQueuedFrontProcess = nil;
}
return(pProc);
}
/* RemoveFromFrontProcessQueue. "Cancels" pending foreground switch to pProc. */
void
RemoveFromFrontProcessQueue(PEntryPtr pProc)
{
PEntryPtr pCurr, pPrev;
pPrev = nil;
pCurr = pFrontProcessQueue;
while (pCurr != nil)
{
if (pCurr == pProc)
{
PEntryPtr pNext;
pNext = pProc->p_nextQueuedFrontProcess;
if (pPrev == nil)
pFrontProcessQueue = pNext;
else
pPrev->p_nextQueuedFrontProcess = pNext;
pProc->p_nextQueuedFrontProcess = nil;
break;
}
pPrev = pCurr;
pCurr = pCurr->p_nextQueuedFrontProcess;
}
}
/* OrphanScrapIntoPartition. This is a great one! When the APPLZONE is about to be
* nuked, we make sure that the SCRAPHANDLE is not taken with it, if that scrap is one
* that we'll want to pass on to other processes (i.e. realScrapOwner is true). We save
* it off by copying it to the beginning of the application partition, then resizing the
* partition to the correct length. This avoids excessive stack or heap use. We
* indicate that SCRAPHANDLE is an orphan, so that it gets disposed when the copying
* (MoveScrap()) is complete.
* If realScrapOwner is false, we make sure the scrap gets disposed even if it is not
* inside the partition that is going away.
* NOTE: Requires valid APPLZONE, but then invalidates it! Yea!
*/
void
OrphanScrapIntoPartition(void)
{
register Handle scrapHdl, procHdl;
unsigned long scrapSize;
Boolean scrapInZone;
/* Nothing to do if scrap is purged.
* NOTE: This has side-effect that a purged scrap handle from heap other than
* APPLZONE or one of its subzones will not get disposed. Too risky to try, tho!
*/
if ((scrapHdl = SCRAPHANDLE) != nil)
if (*scrapHdl == nil)
return;
/* Remember whether handle is from APPLZONE.
* NOTE: Will be <EFBFBD>false<EFBFBD> for nil handle, which is what we want.
*/
scrapInZone = ( (scrapHdl >= APPLZONE) && (scrapHdl < APPLZONE->bkLim) );
/* Don't save/copy the scrap if dying process doesn't have the scrap of interest.
* Dispose the scrap if it is not going to be implicitly disposed when the
* partition is. E.g. The DA Handler's scrap is in the system heap.
*/
if ((pCurrentProcess != srcMyScrapInfo.pScrapOwner) && (pCurrentProcess->p_cutcopycount == cutCopyCount))
{
if ((scrapInZone == false) && (scrapHdl != nil))
DisposHandle(scrapHdl);
return;
}
/* Copy scrap to beginning of partition, then size down the partition handle.
* We nil out APPLZONE since we are putting it out of commission.
* NOTE: We know that scrapSize < GetHandleSize(procHdl) since the SCRAPHANDLE
* is fully contained by a heap within the partition.
*/
if (scrapInZone)
{
procHdl = pCurrentProcess->p_zoneh;
scrapSize = GetHandleSize(scrapHdl);
APPLZONE = nil;
BlockMove(*scrapHdl, *procHdl, scrapSize);
SetHandleSize(procHdl, scrapSize);
/* Now replace the lomem, and forget that handle was the partition */
pCurrentProcess->p_zoneh = nil;
HUnlock(procHdl);
SCRAPHANDLE = procHdl;
}
/* Set up coercion info, and remember to chuck the scrap when we're done. This needs to
* be done even if scrapHdl is nil, since that just means the scrap is on disk.
*/
src_scrap_setup();
srcMyScrapInfo.pScrapOwner = nil;
srcMyScrapInfo.orphanedScrap = true;
}
/* DisposeOrphanedScrap. Dispose the source scrap if it was meant to be. Zero
* memHandle field for safety.
* NOTE: The orphaned scrap exists until a new scrap is made. A new scrap is made
* (or at least committed to) when a) an application calls ZeroScrap, PutScrap,
* SysEdit (cut or copy), or MenuSelect (cut or copy), or b) we call MoveScrap()
* toward the end of a foreground switch. At these times, DisposeOrphanedScrap is
* called because we'll no longer need the orphan. In (a) cases, the orphan was
* never needed (oh well). In (b), the orphan was used to give the new front app
* a scrap, so the orphan's job is complete. Give 'im an extra lump of coal!
*/
void
DisposeOrphanedScrap(void)
{
if (srcMyScrapInfo.orphanedScrap)
{
srcMyScrapInfo.orphanedScrap = false;
if (srcMyScrapInfo.memHandle != nil)
{
assert(*(srcMyScrapInfo.memHandle) != 0);
DisposHandle(srcMyScrapInfo.memHandle);
srcMyScrapInfo.memHandle = nil;
}
}
}
/* src_scrap_setup. Save the outgoing app's scrap state information. It will be
* used later when MoveScrap is called in the context of the new foreground
* application.
*/
void
src_scrap_setup(void)
{
assert (pCurrentProcess != nil);
/* Don't change scrap info if the current process has an out-of-date scrap.
* Decrement the p_cutcopycount so we'll try again to copy the scrap when
* twitching this process back in.
*/
if ((pCurrentProcess != srcMyScrapInfo.pScrapOwner) && (pCurrentProcess->p_cutcopycount == cutCopyCount))
{
pCurrentProcess->p_cutcopycount--;
return;
}
/* Dump orphan, if he's around */
DisposeOrphanedScrap();
/* Set up the src scrap record */
srcMyScrapInfo.size = SCRAPINFO;
/* if SCRAPHANDLE == nil, src scrap is on disk */
if ((srcMyScrapInfo.memHandle = SCRAPHANDLE) == nil)
srcMyScrapInfo.fileName = SCRAPNAME;
/* Assume this scrap is attached still. Caller can change this, if he wants. */
srcMyScrapInfo.orphanedScrap = false;
/* Claim scrap and synch the cutcopycount since global was just taken from the app */
srcMyScrapInfo.pScrapOwner = pCurrentProcess;
pCurrentProcess->p_cutcopycount = cutCopyCount;
}
/* StartForegroundSwitch. Initiate foreground switch. */
void
StartForegroundSwitch(PEntryPtr pProc)
{
register PEntryPtr pFront, pTargetProc;
assert(coercionState == CS_DONE);
pNewFrontProcess = pProc;
pFront = pFrontProcess;
/* Are we in the middle of a system QuitAll? Cancel it if we're switching out of
* the currently intended victim.
* NOTE: This assumes that StartForegroundSwitch is not called when switching to
* a new front process after the current front process has ExitToShell'ed. Otherwise,
* we would be canceling the aeQuitAll just because we got someone to quit! Pretty
* safe assumption, though, since StartForegroundSwitch deals with pCurrentProcess,
* which is completely invalid after ExitToShell.
*/
if ( (pSystemQuitAllMsg != nil)
&& (pFront == (pTargetProc = PEntryFromPSN(&(pSystemQuitAllMsg->targetmfid.localPSN)))) )
CancelQuitAll(&pTargetProc->p_serialNumber);
/* Check whether FG switching out of a process that has been suspended for
* high-level debugging. If so, we must forego the source's half of the state
* machine. We go straight to the new FG process.
*/
if (pFront->p_state == PRNULL)
{
coercionState = CS_ENTERING;
BringProcessToFront(pProc);
// Inform TSM that a non-TSM aware app got swapped to the front ...
//if ( !pFront->p_inlineAware && (tsmLayerOwner != nil))
// InformTSM(kMsgResumeApp, nil);
return;
}
/* Start the ball rolling<6E> */
if ((pFront->p_taskmode & (modeNeedSuspendResume | modeDoesActivateOnFGSwitch))
== (modeNeedSuspendResume | modeDoesActivateOnFGSwitch))
{
coercionState = CS_PRESUSPEND_DEACTIVATE;
}
else
{
coercionState = (WINDOWLIST != nil) ? CS_GENERATE_DEACTIVATE1 : CS_GENERATE_DEACTIVATE2;
PutUpFakeWindow();
}
// Inform TSM that a non-TSM aware app got suspended ...
if ( !pFront->p_inlineAware && (tsmLayerOwner != nil))
InformTSM(kMsgSuspendApp, nil);
}
/* SetThirdSrcCoercionState. Decide where to go after CS_GENERATE_DEACTIVATE1
* or CS_GENERATE_DEACTIVATE2. Being in CS_GENERATE_DEACTIVATE2 means that
* we're expecting to see an activate event for the fake DA window. It's time
* to put the outgoing app on ice.
*/
void
SetThirdSrcCoercionState(void)
{
/* See if we're expecting deactivate-then-activate (i.e. whether the fake window
* was not the only window in the WINDOWLIST).
*/
if (coercionState == CS_GENERATE_DEACTIVATE1)
{
coercionState = CS_GENERATE_DEACTIVATE2;
return;
}
/* If we can't see cut/copies, assume it happened */
if (pCurrentProcess->p_haseditcmdkeys == false)
cutCopyCount++;
/* Set the next state. This will determine how the outgoing app will be told/forced
* to post its scrap and get switched out.
*/
/* Try to use modern suspend event that tells app to do it all */
if ((pCurrentProcess->p_taskmode & modeNeedSuspendResume) != 0)
coercionState = CS_SUSPEND;
/* We have dumb app that doesn't understand suspend events. */
else
{
/* Force scrap coercion if necessary */
if (pCurrentProcess->p_cutcopycount != cutCopyCount)
{
nullTimeout = NULL_TIMEOUT_INIT;
coercionState = CS_GENERATE_NULLS_THEN_CMD_V;
}
/* Otherwise, start the move to the new app. */
else
coercionState = CS_EXITING;
}
}
/* SetFirstDstCoercionState. Start the the track that the dst task will take through
* coercion state space. Called only by engine in CS_ENTERING state, this can be
* pulled back inline for optimization.
*/
void
SetFirstDstCoercionState(void)
{
PEntryPtr pCurrProc = pCurrentProcess;
if (pCurrProc->p_slices == 1)
{
coercionState = CS_INIT;
return;
}
/* NOTE: Have to set bit 1 dynamically */
resume_message = RESUME_MESSAGE;
/* Check whether returning from scratchpad switch */
if (pCurrProc->p_puppetvars.masterPuppetProc != nil)
{
resume_message |= (1 << bitResumeScratch);
*((char *) &resume_message + 1) = pCurrProc->p_puppetvars.masterResumeMsg;
pCurrProc->p_puppetvars.masterResumeMsg = resumeFlag;
pCurrProc->p_puppetvars.masterPuppetProc = nil;
}
/* Check whether we need to copy clipboard. If so, the process needs also to
* coerce the clipboard into its scrap when it gets its resume event.
*/
if (pCurrProc->p_cutcopycount != cutCopyCount)
{
resume_message |= convertClipboardFlag;
coercionState = CS_MOVE_SCRAP;
}
/* No clipboard copying needed. Take the easy way out. */
else
{
/* Pass the honor of scrap ownership to the new front process */
srcMyScrapInfo.pScrapOwner = pCurrentProcess;
/* Remove orphaned scrap, if that's the case */
DisposeOrphanedScrap();
/* Shortcut to resume event for the bright, young apps */
if ((pCurrProc->p_taskmode & modeNeedSuspendResume) != 0)
coercionState = CS_RESUME;
else
{
coercionState = CS_CLICK_IN_WINDOW;
clickDAWindTimeout = CLICK_DA_WIND_TIMEOUT_INIT;
}
}
}
/* CheckForCutCopySysEditCmd. Given the edit command passed to _SysEdit, determine whether
* it is a "cut" or "copy" type of command. If it is, increment the "cutcopycount"
* addressed by pCutCopyCount. If said count is the global cutCopyCount, call
* DisposeOrphanScrap to free the memory now that we won't need the orphan.
*/
void
CheckForCutCopySysEditCmd(short theEditCmd, short *pCutCopyCount)
{
/* Edit command passed to _SysEdit has not yet been biased by accUndo */
if ( (theEditCmd != (accCut-accUndo)) && (theEditCmd != (accCopy-accUndo)) )
return;
(*pCutCopyCount)++;
if (pCutCopyCount == &cutCopyCount)
DisposeOrphanedScrap();
}
/* CheckForCutCopyMenuKey. Given the character that is the key equivalent for a menu
* item, determine whether the item is a "cut" or "copy" type of command.
* If it is, increment the "cutcopycount" addressed by pCutCopyCount. If said count is
* the global cutCopyCount, call DisposeOrphanScrap to free the memory now that we won't
* need the orphan.
*/
void
CheckForCutCopyMenuKey(unsigned char ch, short *pCutCopyCount)
{
/* Is it a cut or copy char? */
switch(ch)
{
case (unsigned char)'X':
case (unsigned char)'x':
case (unsigned char)'<EFBFBD>': /* Option-x */
case (unsigned char)'C':
case (unsigned char)'c':
case (unsigned char)'<EFBFBD>': /* Option-c */
case (unsigned char)'Z':
case (unsigned char)'z':
case (unsigned char)'<EFBFBD>': /* Option-z */
{
(*pCutCopyCount)++;
if (pCutCopyCount == &cutCopyCount)
DisposeOrphanedScrap();
}
}
}
/* FindCutCopyItems. Check the menu with Handle menuHdl to see if it has any cut
* or copy items that we can find. Return result of search. This does nothing to
* relocate menuHdl so it needn't be locked.
* NOTE: Uses speedy but structure-dependant search through menu data.
*/
short
FindCutCopyItems(MenuHandle menuHdl)
{
MenuPtr menuPtr;
register Ptr chPtr;
char nameLen;
/* Assume menu is useless if no menu info available */
if (menuHdl == nil || (menuPtr = *menuHdl) == nil)
return(wFalse);
/* Jump over menu title */
chPtr = &(menuPtr->menuData);
nameLen = *chPtr++;
chPtr += nameLen;
if (*chPtr == (char) 0)
return(wFalse);
for (;;)
{
nameLen = *chPtr++; /* get length of item name */
if (nameLen == (char) 0)
return(wFalse); /* ran off end of list */
chPtr += nameLen + 4; /* skip over name and flags */
/* Is it a cut or copy char? */
switch(chPtr[-3])
{
case 'X':
case 'x':
case 'C':
case 'c':
return(wTrue);
}
}
}