//:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: // // npmac.cpp // //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: #include #include #include #include #include #include #include #define XP_MAC 1 // // A4Stuff.h contains the definition of EnterCodeResource and // EnterCodeResource, used for setting up the code resourceÕs // globals for 68K (analagous to the function SetCurrentA5 // defined by the toolbox). // // A4Stuff does not exist as of CW 7. Define them to nothing. // #if (defined(__MWERKS__) && (__MWERKS__ >= 0x2400)) || defined(__GNUC__) #define EnterCodeResource() #define ExitCodeResource() #else #include #endif #include "npapi.h" // // The Mixed Mode procInfos defined in npupp.h assume Think C- // style calling conventions. These conventions are used by // Metrowerks with the exception of pointer return types, which // in Metrowerks 68K are returned in A0, instead of the standard // D0. Thus, since NPN_MemAlloc and NPN_UserAgent return pointers, // Mixed Mode will return the values to a 68K plugin in D0, but // a 68K plugin compiled by Metrowerks will expect the result in // A0. The following pragma forces Metrowerks to use D0 instead. // #ifdef __MWERKS__ #ifndef powerc #pragma pointers_in_D0 #endif #endif #include "npupp.h" #ifdef __MWERKS__ #ifndef powerc #pragma pointers_in_A0 #endif #endif // The following fix for static initializers (which fixes a previous // incompatibility with some parts of PowerPlant, was submitted by // Jan Ulbrich. #ifdef __MWERKS__ #ifdef __cplusplus extern "C" { #endif #ifndef powerc extern void __InitCode__(void); #else extern void __sinit(void); #define __InitCode__ __sinit #endif extern void __destroy_global_chain(void); #ifdef __cplusplus } #endif // __cplusplus #endif // __MWERKS__ // // Define PLUGIN_TRACE to 1 to have the wrapper functions emit // DebugStr messages whenever they are called. // //#define PLUGIN_TRACE 1 #if PLUGIN_TRACE #define PLUGINDEBUGSTR(msg) ::DebugStr(msg) #else #define PLUGINDEBUGSTR(msg) ((void) 0) #endif #ifdef __ppc__ // glue for mapping outgoing Macho function pointers to TVectors struct TFPtoTVGlue{ void* glue[2]; }; struct { TFPtoTVGlue newp; TFPtoTVGlue destroy; TFPtoTVGlue setwindow; TFPtoTVGlue newstream; TFPtoTVGlue destroystream; TFPtoTVGlue asfile; TFPtoTVGlue writeready; TFPtoTVGlue write; TFPtoTVGlue print; TFPtoTVGlue event; TFPtoTVGlue urlnotify; TFPtoTVGlue getvalue; TFPtoTVGlue setvalue; TFPtoTVGlue shutdown; } gPluginFuncsGlueTable; static inline void* SetupFPtoTVGlue(TFPtoTVGlue* functionGlue, void* fp) { functionGlue->glue[0] = fp; functionGlue->glue[1] = 0; return functionGlue; } #define PLUGIN_TO_HOST_GLUE(name, fp) (SetupFPtoTVGlue(&gPluginFuncsGlueTable.name, (void*)fp)) // glue for mapping netscape TVectors to Macho function pointers struct TTVtoFPGlue { uint32 glue[6]; }; struct { TTVtoFPGlue geturl; TTVtoFPGlue posturl; TTVtoFPGlue requestread; TTVtoFPGlue newstream; TTVtoFPGlue write; TTVtoFPGlue destroystream; TTVtoFPGlue status; TTVtoFPGlue uagent; TTVtoFPGlue memalloc; TTVtoFPGlue memfree; TTVtoFPGlue memflush; TTVtoFPGlue reloadplugins; TTVtoFPGlue getJavaEnv; TTVtoFPGlue getJavaPeer; TTVtoFPGlue geturlnotify; TTVtoFPGlue posturlnotify; TTVtoFPGlue getvalue; TTVtoFPGlue setvalue; TTVtoFPGlue invalidaterect; TTVtoFPGlue invalidateregion; TTVtoFPGlue forceredraw; TTVtoFPGlue pushpopupsenabledstate; TTVtoFPGlue poppopupsenabledstate; } gNetscapeFuncsGlueTable; static void* SetupTVtoFPGlue(TTVtoFPGlue* functionGlue, void* tvp) { static const TTVtoFPGlue glueTemplate = { 0x3D800000, 0x618C0000, 0x800C0000, 0x804C0004, 0x7C0903A6, 0x4E800420 }; memcpy(functionGlue, &glueTemplate, sizeof(TTVtoFPGlue)); functionGlue->glue[0] |= ((UInt32)tvp >> 16); functionGlue->glue[1] |= ((UInt32)tvp & 0xFFFF); ::MakeDataExecutable(functionGlue, sizeof(TTVtoFPGlue)); return functionGlue; } #define HOST_TO_PLUGIN_GLUE(name, fp) (SetupTVtoFPGlue(&gNetscapeFuncsGlueTable.name, (void*)fp)) #else #define PLUGIN_TO_HOST_GLUE(name, fp) (fp) #define HOST_TO_PLUGIN_GLUE(name, fp) (fp) #endif /* __ppc__ */ #pragma mark - //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: // // Globals // //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: #if !TARGET_API_MAC_CARBON QDGlobals* gQDPtr; // Pointer to NetscapeÕs QuickDraw globals #endif short gResFile; // Refnum of the pluginÕs resource file NPNetscapeFuncs gNetscapeFuncs; // Function table for procs in Netscape called by plugin //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: // // Wrapper functions for all calls from the plugin to Netscape. // These functions let the plugin developer just call the APIs // as documented and defined in npapi.h, without needing to know // about the function table and call macros in npupp.h. // //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: void NPN_Version(int* plugin_major, int* plugin_minor, int* netscape_major, int* netscape_minor) { *plugin_major = NP_VERSION_MAJOR; *plugin_minor = NP_VERSION_MINOR; *netscape_major = gNetscapeFuncs.version >> 8; // Major version is in high byte *netscape_minor = gNetscapeFuncs.version & 0xFF; // Minor version is in low byte } NPError NPN_GetURLNotify(NPP instance, const char* url, const char* window, void* notifyData) { int navMinorVers = gNetscapeFuncs.version & 0xFF; NPError err; if( navMinorVers >= NPVERS_HAS_NOTIFICATION ) { err = CallNPN_GetURLNotifyProc(gNetscapeFuncs.geturlnotify, instance, url, window, notifyData); } else { err = NPERR_INCOMPATIBLE_VERSION_ERROR; } return err; } NPError NPN_GetURL(NPP instance, const char* url, const char* window) { return CallNPN_GetURLProc(gNetscapeFuncs.geturl, instance, url, window); } NPError NPN_PostURLNotify(NPP instance, const char* url, const char* window, uint32 len, const char* buf, NPBool file, void* notifyData) { int navMinorVers = gNetscapeFuncs.version & 0xFF; NPError err; if( navMinorVers >= NPVERS_HAS_NOTIFICATION ) { err = CallNPN_PostURLNotifyProc(gNetscapeFuncs.posturlnotify, instance, url, window, len, buf, file, notifyData); } else { err = NPERR_INCOMPATIBLE_VERSION_ERROR; } return err; } NPError NPN_PostURL(NPP instance, const char* url, const char* window, uint32 len, const char* buf, NPBool file) { return CallNPN_PostURLProc(gNetscapeFuncs.posturl, instance, url, window, len, buf, file); } NPError NPN_RequestRead(NPStream* stream, NPByteRange* rangeList) { return CallNPN_RequestReadProc(gNetscapeFuncs.requestread, stream, rangeList); } NPError NPN_NewStream(NPP instance, NPMIMEType type, const char* window, NPStream** stream) { int navMinorVers = gNetscapeFuncs.version & 0xFF; NPError err; if( navMinorVers >= NPVERS_HAS_STREAMOUTPUT ) { err = CallNPN_NewStreamProc(gNetscapeFuncs.newstream, instance, type, window, stream); } else { err = NPERR_INCOMPATIBLE_VERSION_ERROR; } return err; } int32 NPN_Write(NPP instance, NPStream* stream, int32 len, void* buffer) { int navMinorVers = gNetscapeFuncs.version & 0xFF; NPError err; if( navMinorVers >= NPVERS_HAS_STREAMOUTPUT ) { err = CallNPN_WriteProc(gNetscapeFuncs.write, instance, stream, len, buffer); } else { err = NPERR_INCOMPATIBLE_VERSION_ERROR; } return err; } NPError NPN_DestroyStream(NPP instance, NPStream* stream, NPError reason) { int navMinorVers = gNetscapeFuncs.version & 0xFF; NPError err; if( navMinorVers >= NPVERS_HAS_STREAMOUTPUT ) { err = CallNPN_DestroyStreamProc(gNetscapeFuncs.destroystream, instance, stream, reason); } else { err = NPERR_INCOMPATIBLE_VERSION_ERROR; } return err; } void NPN_Status(NPP instance, const char* message) { CallNPN_StatusProc(gNetscapeFuncs.status, instance, message); } const char* NPN_UserAgent(NPP instance) { return CallNPN_UserAgentProc(gNetscapeFuncs.uagent, instance); } void* NPN_MemAlloc(uint32 size) { return CallNPN_MemAllocProc(gNetscapeFuncs.memalloc, size); } void NPN_MemFree(void* ptr) { CallNPN_MemFreeProc(gNetscapeFuncs.memfree, ptr); } uint32 NPN_MemFlush(uint32 size) { return CallNPN_MemFlushProc(gNetscapeFuncs.memflush, size); } void NPN_ReloadPlugins(NPBool reloadPages) { CallNPN_ReloadPluginsProc(gNetscapeFuncs.reloadplugins, reloadPages); } #ifdef OJI JRIEnv* NPN_GetJavaEnv(void) { return CallNPN_GetJavaEnvProc( gNetscapeFuncs.getJavaEnv ); } jobject NPN_GetJavaPeer(NPP instance) { return CallNPN_GetJavaPeerProc( gNetscapeFuncs.getJavaPeer, instance ); } #endif NPError NPN_GetValue(NPP instance, NPNVariable variable, void *value) { return CallNPN_GetValueProc( gNetscapeFuncs.getvalue, instance, variable, value); } NPError NPN_SetValue(NPP instance, NPPVariable variable, void *value) { return CallNPN_SetValueProc( gNetscapeFuncs.setvalue, instance, variable, value); } void NPN_InvalidateRect(NPP instance, NPRect *rect) { CallNPN_InvalidateRectProc( gNetscapeFuncs.invalidaterect, instance, rect); } void NPN_InvalidateRegion(NPP instance, NPRegion region) { CallNPN_InvalidateRegionProc( gNetscapeFuncs.invalidateregion, instance, region); } void NPN_ForceRedraw(NPP instance) { CallNPN_ForceRedrawProc( gNetscapeFuncs.forceredraw, instance); } void NPN_PushPopupsEnabledState(NPP instance, NPBool enabled) { CallNPN_PushPopupsEnabledStateProc( gNetscapeFuncs.pushpopupsenabledstate, instance, enabled); } void NPN_PopPopupsEnabledState(NPP instance) { CallNPN_PopPopupsEnabledStateProc( gNetscapeFuncs.poppopupsenabledstate, instance); } #pragma mark - //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: // // Wrapper functions for all calls from Netscape to the plugin. // These functions let the plugin developer just create the APIs // as documented and defined in npapi.h, without needing to // install those functions in the function table or worry about // setting up globals for 68K plugins. // //:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: NPError Private_Initialize(void); void Private_Shutdown(void); NPError Private_New(NPMIMEType pluginType, NPP instance, uint16 mode, int16 argc, char* argn[], char* argv[], NPSavedData* saved); NPError Private_Destroy(NPP instance, NPSavedData** save); NPError Private_SetWindow(NPP instance, NPWindow* window); NPError Private_NewStream(NPP instance, NPMIMEType type, NPStream* stream, NPBool seekable, uint16* stype); NPError Private_DestroyStream(NPP instance, NPStream* stream, NPError reason); int32 Private_WriteReady(NPP instance, NPStream* stream); int32 Private_Write(NPP instance, NPStream* stream, int32 offset, int32 len, void* buffer); void Private_StreamAsFile(NPP instance, NPStream* stream, const char* fname); void Private_Print(NPP instance, NPPrint* platformPrint); int16 Private_HandleEvent(NPP instance, void* event); void Private_URLNotify(NPP instance, const char* url, NPReason reason, void* notifyData); jobject Private_GetJavaClass(void); NPError Private_Initialize(void) { NPError err; EnterCodeResource(); PLUGINDEBUGSTR("\pInitialize;g;"); err = NPP_Initialize(); ExitCodeResource(); return err; } void Private_Shutdown(void) { EnterCodeResource(); PLUGINDEBUGSTR("\pShutdown;g;"); NPP_Shutdown(); #ifdef __MWERKS__ __destroy_global_chain(); #endif ExitCodeResource(); } NPError Private_New(NPMIMEType pluginType, NPP instance, uint16 mode, int16 argc, char* argn[], char* argv[], NPSavedData* saved) { EnterCodeResource(); NPError ret = NPP_New(pluginType, instance, mode, argc, argn, argv, saved); PLUGINDEBUGSTR("\pNew;g;"); ExitCodeResource(); return ret; } NPError Private_Destroy(NPP instance, NPSavedData** save) { NPError err; EnterCodeResource(); PLUGINDEBUGSTR("\pDestroy;g;"); err = NPP_Destroy(instance, save); ExitCodeResource(); return err; } NPError Private_SetWindow(NPP instance, NPWindow* window) { NPError err; EnterCodeResource(); PLUGINDEBUGSTR("\pSetWindow;g;"); err = NPP_SetWindow(instance, window); ExitCodeResource(); return err; } NPError Private_NewStream(NPP instance, NPMIMEType type, NPStream* stream, NPBool seekable, uint16* stype) { NPError err; EnterCodeResource(); PLUGINDEBUGSTR("\pNewStream;g;"); err = NPP_NewStream(instance, type, stream, seekable, stype); ExitCodeResource(); return err; } int32 Private_WriteReady(NPP instance, NPStream* stream) { int32 result; EnterCodeResource(); PLUGINDEBUGSTR("\pWriteReady;g;"); result = NPP_WriteReady(instance, stream); ExitCodeResource(); return result; } int32 Private_Write(NPP instance, NPStream* stream, int32 offset, int32 len, void* buffer) { int32 result; EnterCodeResource(); PLUGINDEBUGSTR("\pWrite;g;"); result = NPP_Write(instance, stream, offset, len, buffer); ExitCodeResource(); return result; } void Private_StreamAsFile(NPP instance, NPStream* stream, const char* fname) { EnterCodeResource(); PLUGINDEBUGSTR("\pStreamAsFile;g;"); NPP_StreamAsFile(instance, stream, fname); ExitCodeResource(); } NPError Private_DestroyStream(NPP instance, NPStream* stream, NPError reason) { NPError err; EnterCodeResource(); PLUGINDEBUGSTR("\pDestroyStream;g;"); err = NPP_DestroyStream(instance, stream, reason); ExitCodeResource(); return err; } int16 Private_HandleEvent(NPP instance, void* event) { int16 result; EnterCodeResource(); PLUGINDEBUGSTR("\pHandleEvent;g;"); result = NPP_HandleEvent(instance, event); ExitCodeResource(); return result; } void Private_Print(NPP instance, NPPrint* platformPrint) { EnterCodeResource(); PLUGINDEBUGSTR("\pPrint;g;"); NPP_Print(instance, platformPrint); ExitCodeResource(); } void Private_URLNotify(NPP instance, const char* url, NPReason reason, void* notifyData) { EnterCodeResource(); PLUGINDEBUGSTR("\pURLNotify;g;"); NPP_URLNotify(instance, url, reason, notifyData); ExitCodeResource(); } #ifdef OJI jobject Private_GetJavaClass(void) { EnterCodeResource(); PLUGINDEBUGSTR("\pGetJavaClass;g;"); jobject clazz = NPP_GetJavaClass(); ExitCodeResource(); if (clazz) { JRIEnv* env = NPN_GetJavaEnv(); return (jobject)JRI_NewGlobalRef(env, clazz); } return NULL; } #endif void SetUpQD(void); void SetUpQD(void) { #if !TARGET_API_MAC_CARBON ProcessSerialNumber PSN; FSSpec myFSSpec; Str63 name; ProcessInfoRec infoRec; OSErr result = noErr; CFragConnectionID connID; Str255 errName; #endif // // Memorize the pluginÕs resource file // refnum for later use. // gResFile = CurResFile(); #if !TARGET_API_MAC_CARBON // // Ask the system if CFM is available. // long response; OSErr err = Gestalt(gestaltCFMAttr, &response); Boolean hasCFM = BitTst(&response, 31-gestaltCFMPresent); ProcessInfoRec infoRec; if (hasCFM) { // // GetProcessInformation takes a process serial number and // will give us back the name and FSSpec of the application. // See the Process Manager in IM. // Str63 name; FSSpec myFSSpec; infoRec.processInfoLength = sizeof(ProcessInfoRec); infoRec.processName = name; infoRec.processAppSpec = &myFSSpec; ProcessSerialNumber PSN; PSN.highLongOfPSN = 0; PSN.lowLongOfPSN = kCurrentProcess; result = GetProcessInformation(&PSN, &infoRec); if (result != noErr) PLUGINDEBUGSTR("\pFailed in GetProcessInformation"); } else // // If no CFM installed, assume it must be a 68K app. // result = -1; CFragConnectionID connID; if (result == noErr) { // // Now that we know the app name and FSSpec, we can call GetDiskFragment // to get a connID to use in a subsequent call to FindSymbol (it will also // return the address of ÒmainÓ in app, which we ignore). If GetDiskFragment // returns an error, we assume the app must be 68K. // Ptr mainAddr; Str255 errName; result = GetDiskFragment(infoRec.processAppSpec, 0L, 0L, infoRec.processName, kLoadCFrag, &connID, (Ptr*)&mainAddr, errName); } if (result == noErr) { // // The app is a PPC code fragment, so call FindSymbol // to get the exported ÒqdÓ symbol so we can access its // QuickDraw globals. // CFragSymbolClass symClass; result = FindSymbol(connID, "\pqd", (Ptr*)&gQDPtr, &symClass); if (result != noErr) { // this fails if we are in NS 6 gQDPtr = &qd; // so we default to the standard QD globals } } else { // // The app is 68K, so use its A5 to compute the address // of its QuickDraw globals. // gQDPtr = (QDGlobals*)(*((long*)SetCurrentA5()) - (sizeof(QDGlobals) - sizeof(GrafPtr))); } #endif } #ifdef __GNUC__ // gcc requires that main have an 'int' return type int main(NPNetscapeFuncs* nsTable, NPPluginFuncs* pluginFuncs, NPP_ShutdownUPP* unloadUpp); #else NPError main(NPNetscapeFuncs* nsTable, NPPluginFuncs* pluginFuncs, NPP_ShutdownUPP* unloadUpp); #endif #if !TARGET_API_MAC_CARBON #pragma export on #if GENERATINGCFM RoutineDescriptor mainRD = BUILD_ROUTINE_DESCRIPTOR(uppNPP_MainEntryProcInfo, main); #endif #pragma export off #endif /* !TARGET_API_MAC_CARBON */ #ifdef __GNUC__ DEFINE_API_C(int) main(NPNetscapeFuncs* nsTable, NPPluginFuncs* pluginFuncs, NPP_ShutdownUPP* unloadUpp) #else DEFINE_API_C(NPError) main(NPNetscapeFuncs* nsTable, NPPluginFuncs* pluginFuncs, NPP_ShutdownUPP* unloadUpp) #endif { EnterCodeResource(); PLUGINDEBUGSTR("\pmain"); #ifdef __MWERKS__ __InitCode__(); #endif NPError err = NPERR_NO_ERROR; // // Ensure that everything Netscape passed us is valid! // if ((nsTable == NULL) || (pluginFuncs == NULL) || (unloadUpp == NULL)) err = NPERR_INVALID_FUNCTABLE_ERROR; // // Check the ÒmajorÓ version passed in NetscapeÕs function table. // We wonÕt load if the major version is newer than what we expect. // Also check that the function tables passed in are big enough for // all the functions we need (they could be bigger, if Netscape added // new APIs, but thatÕs OK with us -- weÕll just ignore them). // if (err == NPERR_NO_ERROR) { if ((nsTable->version >> 8) > NP_VERSION_MAJOR) // Major version is in high byte err = NPERR_INCOMPATIBLE_VERSION_ERROR; } if (err == NPERR_NO_ERROR) { // // Copy all the fields of NetscapeÕs function table into our // copy so we can call back into Netscape later. Note that // we need to copy the fields one by one, rather than assigning // the whole structure, because the Netscape function table // could actually be bigger than what we expect. // int navMinorVers = nsTable->version & 0xFF; gNetscapeFuncs.version = nsTable->version; gNetscapeFuncs.size = nsTable->size; gNetscapeFuncs.posturl = (NPN_PostURLUPP)HOST_TO_PLUGIN_GLUE(posturl, nsTable->posturl); gNetscapeFuncs.geturl = (NPN_GetURLUPP)HOST_TO_PLUGIN_GLUE(geturl, nsTable->geturl); gNetscapeFuncs.requestread = (NPN_RequestReadUPP)HOST_TO_PLUGIN_GLUE(requestread, nsTable->requestread); gNetscapeFuncs.newstream = (NPN_NewStreamUPP)HOST_TO_PLUGIN_GLUE(newstream, nsTable->newstream); gNetscapeFuncs.write = (NPN_WriteUPP)HOST_TO_PLUGIN_GLUE(write, nsTable->write); gNetscapeFuncs.destroystream = (NPN_DestroyStreamUPP)HOST_TO_PLUGIN_GLUE(destroystream, nsTable->destroystream); gNetscapeFuncs.status = (NPN_StatusUPP)HOST_TO_PLUGIN_GLUE(status, nsTable->status); gNetscapeFuncs.uagent = (NPN_UserAgentUPP)HOST_TO_PLUGIN_GLUE(uagent, nsTable->uagent); gNetscapeFuncs.memalloc = (NPN_MemAllocUPP)HOST_TO_PLUGIN_GLUE(memalloc, nsTable->memalloc); gNetscapeFuncs.memfree = (NPN_MemFreeUPP)HOST_TO_PLUGIN_GLUE(memfree, nsTable->memfree); gNetscapeFuncs.memflush = (NPN_MemFlushUPP)HOST_TO_PLUGIN_GLUE(memflush, nsTable->memflush); gNetscapeFuncs.reloadplugins = (NPN_ReloadPluginsUPP)HOST_TO_PLUGIN_GLUE(reloadplugins, nsTable->reloadplugins); if( navMinorVers >= NPVERS_HAS_LIVECONNECT ) { gNetscapeFuncs.getJavaEnv = (NPN_GetJavaEnvUPP)HOST_TO_PLUGIN_GLUE(getJavaEnv, nsTable->getJavaEnv); gNetscapeFuncs.getJavaPeer = (NPN_GetJavaPeerUPP)HOST_TO_PLUGIN_GLUE(getJavaPeer, nsTable->getJavaPeer); } if( navMinorVers >= NPVERS_HAS_NOTIFICATION ) { gNetscapeFuncs.geturlnotify = (NPN_GetURLNotifyUPP)HOST_TO_PLUGIN_GLUE(geturlnotify, nsTable->geturlnotify); gNetscapeFuncs.posturlnotify = (NPN_PostURLNotifyUPP)HOST_TO_PLUGIN_GLUE(posturlnotify, nsTable->posturlnotify); } gNetscapeFuncs.getvalue = (NPN_GetValueUPP)HOST_TO_PLUGIN_GLUE(getvalue, nsTable->getvalue); gNetscapeFuncs.setvalue = (NPN_SetValueUPP)HOST_TO_PLUGIN_GLUE(setvalue, nsTable->setvalue); gNetscapeFuncs.invalidaterect = (NPN_InvalidateRectUPP)HOST_TO_PLUGIN_GLUE(invalidaterect, nsTable->invalidaterect); gNetscapeFuncs.invalidateregion = (NPN_InvalidateRegionUPP)HOST_TO_PLUGIN_GLUE(invalidateregion, nsTable->invalidateregion); gNetscapeFuncs.forceredraw = (NPN_ForceRedrawUPP)HOST_TO_PLUGIN_GLUE(forceredraw, nsTable->forceredraw); gNetscapeFuncs.pushpopupsenabledstate = (NPN_PushPopupsEnabledStateUPP)HOST_TO_PLUGIN_GLUE(pushpopupsenabledstate, nsTable->pushpopupsenabledstate); gNetscapeFuncs.poppopupsenabledstate = (NPN_PopPopupsEnabledStateUPP)HOST_TO_PLUGIN_GLUE(poppopupsenabledstate, nsTable->poppopupsenabledstate); // // Set up the plugin function table that Netscape will use to // call us. Netscape needs to know about our version and size // and have a UniversalProcPointer for every function we implement. // pluginFuncs->version = (NP_VERSION_MAJOR << 8) + NP_VERSION_MINOR; pluginFuncs->size = sizeof(NPPluginFuncs); pluginFuncs->newp = NewNPP_NewProc(PLUGIN_TO_HOST_GLUE(newp, Private_New)); pluginFuncs->destroy = NewNPP_DestroyProc(PLUGIN_TO_HOST_GLUE(destroy, Private_Destroy)); pluginFuncs->setwindow = NewNPP_SetWindowProc(PLUGIN_TO_HOST_GLUE(setwindow, Private_SetWindow)); pluginFuncs->newstream = NewNPP_NewStreamProc(PLUGIN_TO_HOST_GLUE(newstream, Private_NewStream)); pluginFuncs->destroystream = NewNPP_DestroyStreamProc(PLUGIN_TO_HOST_GLUE(destroystream, Private_DestroyStream)); pluginFuncs->asfile = NewNPP_StreamAsFileProc(PLUGIN_TO_HOST_GLUE(asfile, Private_StreamAsFile)); pluginFuncs->writeready = NewNPP_WriteReadyProc(PLUGIN_TO_HOST_GLUE(writeready, Private_WriteReady)); pluginFuncs->write = NewNPP_WriteProc(PLUGIN_TO_HOST_GLUE(write, Private_Write)); pluginFuncs->print = NewNPP_PrintProc(PLUGIN_TO_HOST_GLUE(print, Private_Print)); pluginFuncs->event = NewNPP_HandleEventProc(PLUGIN_TO_HOST_GLUE(event, Private_HandleEvent)); if( navMinorVers >= NPVERS_HAS_NOTIFICATION ) { pluginFuncs->urlnotify = NewNPP_URLNotifyProc(PLUGIN_TO_HOST_GLUE(urlnotify, Private_URLNotify)); } #ifdef OJI if( navMinorVers >= NPVERS_HAS_LIVECONNECT ) { pluginFuncs->javaClass = (JRIGlobalRef) Private_GetJavaClass(); } #else pluginFuncs->javaClass = NULL; #endif *unloadUpp = NewNPP_ShutdownProc(PLUGIN_TO_HOST_GLUE(shutdown, Private_Shutdown)); SetUpQD(); err = Private_Initialize(); } ExitCodeResource(); return err; }