mirror of
https://github.com/sheumann/AFPBridge.git
synced 2024-11-01 01:05:06 +00:00
6be698833d
We indicate that AFPBridge is installed by creating a message in the message center (which also includes its version, in case we want to distinguish between different versions in the future).
176 lines
4.3 KiB
C
176 lines
4.3 KiB
C
#pragma rtl
|
|
|
|
#include <appletalk.h>
|
|
#include <locator.h>
|
|
#include <misctool.h>
|
|
#include <tcpip.h>
|
|
#include <desk.h>
|
|
#include <orca.h>
|
|
#include <gsos.h>
|
|
#include <string.h>
|
|
#include "installcmds.h"
|
|
#include "aspinterface.h"
|
|
#include "asmglue.h"
|
|
|
|
const char bootInfoString[] = "AFPBridge v1.0b1";
|
|
LongWord version = 0x01006001; /* in rVersion format */
|
|
|
|
const char versionMessageString[] = "\pSTH~AFPBridge~Version~";
|
|
|
|
typedef struct VersionMessageRec {
|
|
Word blockLen;
|
|
char nameString[23];
|
|
LongWord dataBlock;
|
|
} VersionMessageRec;
|
|
|
|
extern Word *unloadFlagPtr;
|
|
extern void resetRoutine(void);
|
|
|
|
void pollTask(void);
|
|
void notificationProc(void);
|
|
|
|
static struct RunQRec {
|
|
Long reserved1;
|
|
Word period;
|
|
Word signature;
|
|
Long reserved2;
|
|
Byte jml;
|
|
void (*proc)(void);
|
|
} runQRec;
|
|
|
|
static struct NotificationProcRec {
|
|
Long reserved1;
|
|
Word reserved2;
|
|
Word Signature;
|
|
Long Event_flags;
|
|
Long Event_code;
|
|
Byte jml;
|
|
void (*proc)(void);
|
|
} notificationProcRec;
|
|
|
|
#define SoftResetPtr ((LongWord *)0xE11010)
|
|
extern LongWord oldSoftReset;
|
|
|
|
#define JML 0x5C
|
|
|
|
void setUnloadFlag(void) {
|
|
if (*unloadFlagPtr == 0)
|
|
*unloadFlagPtr = 1;
|
|
}
|
|
|
|
int main(void) {
|
|
unsigned int i;
|
|
FSTInfoRecGS fstInfoRec;
|
|
NotifyProcRecGS addNotifyProcRec;
|
|
VersionMessageRec versionMessageRec;
|
|
|
|
/*
|
|
* Check for presence of AppleShare FST. We error out and unload
|
|
* if it's not present. Our code doesn't directly depend on the
|
|
* AppleShare FST, but in practice it's not useful without it.
|
|
* This also ensures lower-level AppleTalk stuff is present.
|
|
*/
|
|
fstInfoRec.pCount = 2;
|
|
fstInfoRec.fileSysID = 0;
|
|
for (i = 1; fstInfoRec.fileSysID != appleShareFSID; i++) {
|
|
fstInfoRec.fstNum = i;
|
|
GetFSTInfoGS(&fstInfoRec);
|
|
if (toolerror() == paramRangeErr)
|
|
goto error;
|
|
}
|
|
|
|
/*
|
|
* Load Marinetti.
|
|
* We may get an error if the TCPIP init isn't loaded yet, but we ignore it.
|
|
* The tool stub is still loaded in that case, which is enough for now.
|
|
*/
|
|
LoadOneTool(54, 0x0200);
|
|
if (toolerror() && toolerror() != terrINITNOTFOUND)
|
|
goto error;
|
|
|
|
versionMessageRec.blockLen = sizeof(versionMessageRec);
|
|
memcpy(versionMessageRec.nameString, versionMessageString,
|
|
sizeof(versionMessageRec.nameString));
|
|
versionMessageRec.dataBlock = version;
|
|
MessageByName(TRUE, (Pointer)&versionMessageRec);
|
|
if (toolerror())
|
|
goto error;
|
|
|
|
ShowBootInfo(bootInfoString, NULL);
|
|
|
|
/*
|
|
* Put Marinetti in the default TPT so its tool stub won't be unloaded,
|
|
* even if UnloadOneTool is called on it. Programs may still call
|
|
* TCPIPStartUp and TCPIPShutDown, but those don't actually do
|
|
* anything, so the practical effect is that Marinetti will always
|
|
* be available once its init has loaded (which may not have happened
|
|
* yet when this init loads).
|
|
*/
|
|
SetDefaultTPT();
|
|
|
|
RamForbid();
|
|
installCmds();
|
|
RamPermit();
|
|
|
|
runQRec.period = 1;
|
|
runQRec.signature = 0xA55A;
|
|
runQRec.jml = JML;
|
|
runQRec.proc = pollTask;
|
|
AddToRunQ((Pointer)&runQRec);
|
|
|
|
notificationProcRec.Signature = 0xA55A;
|
|
notificationProcRec.Event_flags = 0x20; /* shutdown */
|
|
notificationProcRec.jml = JML;
|
|
notificationProcRec.proc = notificationProc;
|
|
addNotifyProcRec.pCount = 1;
|
|
addNotifyProcRec.procPointer = (ProcPtr)¬ificationProcRec;
|
|
AddNotifyProcGS(&addNotifyProcRec);
|
|
|
|
oldSoftReset = *SoftResetPtr;
|
|
*SoftResetPtr = ((LongWord)&resetRoutine << 8) | JML;
|
|
|
|
return;
|
|
|
|
error:
|
|
setUnloadFlag();
|
|
return;
|
|
}
|
|
|
|
|
|
#pragma databank 1
|
|
void pollTask(void) {
|
|
Word stateReg;
|
|
|
|
IncBusyFlag();
|
|
stateReg = ForceRomIn();
|
|
|
|
if (OS_KIND == KIND_GSOS)
|
|
PollAllSessions();
|
|
runQRec.period = 4*60;
|
|
|
|
RestoreStateReg(stateReg);
|
|
DecBusyFlag();
|
|
}
|
|
#pragma databank 0
|
|
|
|
|
|
/*
|
|
* Notification procedure called at shutdown time.
|
|
* We try to notify the servers that we're closing the connections.
|
|
* This only works if Marinetti is still active, i.e. if its own
|
|
* shutdown notification procedure hasn't run yet.
|
|
*/
|
|
#pragma databank 1
|
|
void notificationProc(void) {
|
|
Word stateReg;
|
|
|
|
IncBusyFlag();
|
|
stateReg = ForceRomIn();
|
|
|
|
CloseAllSessions(0);
|
|
|
|
RestoreStateReg(stateReg);
|
|
DecBusyFlag();
|
|
}
|
|
#pragma databank 0
|