/* * The contents of this file are subject to the Mozilla Public License * Version 1.1 (the "License"); you may not use this file except in * compliance with the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the * License for the specific language governing rights and limitations * under the License. * * The Original Code is the Mozilla OS/2 libraries. * * The Initial Developer of the Original Code is John Fairhurst, * . Portions created by John Fairhurst are * Copyright (C) 1998 John Fairhurst. All Rights Reserved. * * Contributor(s): * */ // OS/2 plugin-loading code. #define INCL_DOS #define INCL_DOSERRORS #include #include "nsPluginsDir.h" #include "prlink.h" #include "plstr.h" #include "prmem.h" #include "nsSpecialSystemDirectory.h" #include "nsPluginDefs.h" /* Load a string stored as RCDATA in a resource segment */ /* Returned string needs to be PR_Free'd by caller */ static char *LoadRCDATAString( HMODULE hMod, ULONG resid) { APIRET rc; ULONG ulSize = 0; char *string = 0; rc = DosQueryResourceSize( hMod, RT_RCDATA, resid, &ulSize); if( rc == NO_ERROR) { char *readOnlyString = 0; rc = DosGetResource( hMod, RT_RCDATA, resid, (void**) &readOnlyString); /* allow for 0-termination if user hasn't got it right */ if( readOnlyString[ ulSize - 1] != '\0') ulSize++; if( rc == NO_ERROR) { /* copy string & zero-terminate */ string = (char*) PR_Malloc( ulSize); memcpy( string, readOnlyString, ulSize - 1); string[ ulSize - 1] = '\0'; DosFreeResource( readOnlyString); } } return string; } static PRUint32 CalculateVariantCount(char* mimeTypes) { PRUint32 variants = 1; if(mimeTypes == nsnull) return 0; char* index = mimeTypes; while (*index) { if (*index == '|') variants++; ++index; } return variants; } static char** MakeStringArray(PRUint32 variants, char* data) { if((variants <= 0) || (data == nsnull)) return nsnull; char ** array = (char **)PR_Calloc(variants, sizeof(char *)); if(array == nsnull) return nsnull; char * start = data; for(PRUint32 i = 0; i < variants; i++) { char * p = PL_strchr(start, '|'); if(p != nsnull) *p = 0; array[i] = PL_strdup(start); start = ++p; } return array; } static void FreeStringArray(PRUint32 variants, char ** array) { if((variants == 0) || (array == nsnull)) return; for(PRUint32 i = 0; i < variants; i++) { if(array[i] != nsnull) { PL_strfree(array[i]); array[i] = nsnull; } } PR_Free(array); } // nsPluginsDir class nsPluginsDir::nsPluginsDir(PRUint16 location) { // XXX This isn't right for the embedded case, but it's as close // as we can do right now. nsSpecialSystemDirectory appdir( nsSpecialSystemDirectory::OS_CurrentProcessDirectory); appdir += "plugins"; if( !appdir.Exists()) appdir.CreateDirectory(); *(nsFileSpec*)this = appdir; } nsPluginsDir::~nsPluginsDir() { } PRBool nsPluginsDir::IsPluginFile( const nsFileSpec &fileSpec) { PRBool rc = PR_FALSE; char *leafname = fileSpec.GetLeafName(); if( nsnull != leafname) { int len = strlen( leafname); if( len > 6 && // np*.dll (0 == strnicmp( &(leafname[len - 4]), ".dll", 4)) && (0 == strnicmp( leafname, "np", 2))) { rc = PR_TRUE; } delete [] leafname; } return rc; } // nsPluginFile implementation nsPluginFile::nsPluginFile( const nsFileSpec &spec) : nsFileSpec(spec) {} nsPluginFile::~nsPluginFile() {} // Loads the plugin into memory using NSPR's shared-library loading nsresult nsPluginFile::LoadPlugin( PRLibrary *&outLibrary) { nsNSPRPath nsprpath( *this); outLibrary = PR_LoadLibrary( nsprpath); return outLibrary == nsnull ? NS_ERROR_FAILURE : NS_OK; } // Obtains all of the information currently available for this plugin. nsresult nsPluginFile::GetPluginInfo( nsPluginInfo &info) { nsresult rc = NS_ERROR_FAILURE; HMODULE hPlug = 0; // Need a HMODULE to query resource statements char failure[ CCHMAXPATH] = ""; APIRET ret; const char* path = this->GetCString(); ret = DosLoadModule( failure, CCHMAXPATH, path, &hPlug); while( ret == NO_ERROR) { info.fPluginInfoSize = sizeof( nsPluginInfo); char *leaf = GetLeafName(); info.fName = PL_strdup( leaf); delete [] leaf; // get description (doesn't matter if it's missing)... info.fDescription = LoadRCDATAString( hPlug, NS_INFO_FileDescription); char * mimeType = LoadRCDATAString( hPlug, NS_INFO_MIMEType); if( nsnull == mimeType) break; char * mimeDescription = LoadRCDATAString( hPlug, NS_INFO_FileOpenName); if( nsnull == mimeDescription) break; char * extensions = LoadRCDATAString( hPlug, NS_INFO_FileExtents); if( nsnull == extensions) break; info.fVariantCount = CalculateVariantCount(mimeType); info.fMimeTypeArray = MakeStringArray(info.fVariantCount, mimeType); if( info.fMimeTypeArray == nsnull) break; info.fMimeDescriptionArray = MakeStringArray(info.fVariantCount, mimeDescription); if( nsnull == info.fMimeDescriptionArray) break; info.fExtensionArray = MakeStringArray(info.fVariantCount, extensions); if( nsnull == info.fExtensionArray) break; info.fFileName = PL_strdup(path); rc = NS_OK; break; } if( 0 != hPlug) DosFreeModule( hPlug); return rc; } nsresult nsPluginFile::FreePluginInfo(nsPluginInfo& info) { if(info.fName != nsnull) PL_strfree(info.fName); if(info.fDescription != nsnull) PL_strfree(info.fDescription); if(info.fMimeTypeArray != nsnull) FreeStringArray(info.fVariantCount, info.fMimeTypeArray); if(info.fMimeDescriptionArray != nsnull) FreeStringArray(info.fVariantCount, info.fMimeDescriptionArray); if(info.fExtensionArray != nsnull) FreeStringArray(info.fVariantCount, info.fExtensionArray); memset((void *)&info, 0, sizeof(info)); return NS_OK; }