/* -*- Mode: C++; tab-width: 6; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * 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 Mozilla XPCOM. * * The Initial Developer of the Original Code is * Benjamin Smedberg * * Portions created by the Initial Developer are Copyright (C) 2005 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsGlueLinking.h" #include "nsXPCOMGlue.h" #include "nsStringAPI.h" #include #include #include #include struct DependentLib { HINSTANCE libHandle; DependentLib *next; }; static DependentLib *sTop; HINSTANCE sXULLibrary; static void AppendDependentLib(HINSTANCE libHandle) { DependentLib *d = new DependentLib; if (!d) return; d->next = sTop; d->libHandle = libHandle; sTop = d; } static void ReadDependentCB(const char *aDependentLib) { HINSTANCE h = LoadLibraryExW(NS_ConvertUTF8toUTF16(aDependentLib).get(), NULL, LOAD_WITH_ALTERED_SEARCH_PATH); if (!h) return; AppendDependentLib(h); } // like strpbrk but finds the *last* char, not the first static char* ns_strrpbrk(char *string, const char *strCharSet) { char *found = NULL; for (; *string; ++string) { for (const char *search = strCharSet; *search; ++search) { if (*search == *string) { found = string; // Since we're looking for the last char, we save "found" // until we're at the end of the string. } } } return found; } // like strpbrk but finds the *last* char, not the first static wchar_t* ns_wcspbrk(wchar_t *string, const wchar_t *strCharSet) { wchar_t *found = NULL; for (; *string; ++string) { for (const wchar_t *search = strCharSet; *search; ++search) { if (*search == *string) { found = string; // Since we're looking for the last char, we save "found" // until we're at the end of the string. } } } return found; } bool ns_isRelPath(wchar_t* path) { #ifdef WINCE if (path[0] == '\\') return false; #else if (path[1] == ':') return false; #endif return true; } GetFrozenFunctionsFunc XPCOMGlueLoad(const char *aXpcomFile) { wchar_t xpcomFile[MAXPATHLEN]; MultiByteToWideChar(CP_ACP, 0, aXpcomFile,-1, xpcomFile, MAXPATHLEN); if (xpcomFile[0] == '.' && xpcomFile[1] == '\0') { wcscpy(xpcomFile, LXPCOM_DLL); } else { wchar_t xpcomDir[MAXPATHLEN]; if (ns_isRelPath(xpcomFile)) { _wfullpath(xpcomDir, xpcomFile, sizeof(xpcomDir)/sizeof(wchar_t)); } else { wcscpy(xpcomDir, xpcomFile); } wchar_t *lastSlash = ns_wcspbrk(xpcomDir, L"/\\"); if (lastSlash) { *lastSlash = '\0'; char xpcomDir_narrow[MAXPATHLEN]; WideCharToMultiByte(CP_ACP, 0, xpcomDir,-1, xpcomDir_narrow, MAX_PATH, NULL, NULL); XPCOMGlueLoadDependentLibs(xpcomDir_narrow, ReadDependentCB); _snwprintf(lastSlash, MAXPATHLEN - wcslen(xpcomDir), L"\\" LXUL_DLL); sXULLibrary = LoadLibraryExW(xpcomDir, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); #ifdef DEBUG if (!sXULLibrary) { LPVOID lpMsgBuf; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL ); wprintf(L"Error loading xul.dll: %s\n", lpMsgBuf); } #endif //DEBUG } } HINSTANCE h = LoadLibraryExW(xpcomFile, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); if (!h) { #ifdef DEBUG LPVOID lpMsgBuf; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL ); wprintf(L"Error loading xpcom.dll: %s\n", lpMsgBuf); #endif return nsnull; } AppendDependentLib(h); GetFrozenFunctionsFunc sym = (GetFrozenFunctionsFunc) GetProcAddress(h, "NS_GetFrozenFunctions"); if (!sym) XPCOMGlueUnload(); return sym; } void XPCOMGlueUnload() { while (sTop) { FreeLibrary(sTop->libHandle); DependentLib *temp = sTop; sTop = sTop->next; delete temp; } if (sXULLibrary) { FreeLibrary(sXULLibrary); sXULLibrary = nsnull; } } nsresult XPCOMGlueLoadXULFunctions(const nsDynamicFunctionLoad *symbols) { if (!sXULLibrary) return NS_ERROR_NOT_INITIALIZED; nsresult rv = NS_OK; while (symbols->functionName) { *symbols->function = (NSFuncPtr) GetProcAddress(sXULLibrary, symbols->functionName); if (!*symbols->function) rv = NS_ERROR_LOSS_OF_SIGNIFICANT_DATA; ++symbols; } return rv; }