/* ***** 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 libXUL embedding. * * The Initial Developer of the Original Code is * Benjamin Smedberg * * Portions created by the Initial Developer are Copyright (C) 2005 * the Mozilla Foundation. 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 "nsXULAppAPI.h" #include #include "prenv.h" #include "nsIAppShell.h" #include "nsIAppStartupNotifier.h" #include "nsIDirectoryService.h" #include "nsILocalFile.h" #include "nsIToolkitChromeRegistry.h" #include "nsIToolkitProfile.h" #include "nsAppDirectoryServiceDefs.h" #include "nsAppRunner.h" #include "nsAutoRef.h" #include "nsDirectoryServiceDefs.h" #include "nsStaticComponents.h" #include "nsString.h" #include "nsThreadUtils.h" #include "nsWidgetsCID.h" #include "nsXPFEComponentsCID.h" #include "nsXREDirProvider.h" #ifdef MOZ_IPC #include "base/at_exit.h" #include "base/command_line.h" #include "base/message_loop.h" #include "chrome/common/child_process.h" #include "mozilla/ipc/GeckoChildProcessHost.h" #include "mozilla/ipc/GeckoThread.h" #include "ScopedXREEmbed.h" #include "mozilla/plugins/PluginThreadChild.h" #include "ContentProcessThread.h" #include "ContentProcessParent.h" #include "ContentProcessChild.h" #include "mozilla/ipc/TestShellParent.h" #include "mozilla/ipc/XPCShellEnvironment.h" #include "mozilla/test/TestParent.h" #include "mozilla/test/TestProcessParent.h" #include "mozilla/test/TestThreadChild.h" #include "mozilla/Monitor.h" using mozilla::ipc::GeckoChildProcessHost; using mozilla::ipc::GeckoThread; using mozilla::ipc::ScopedXREEmbed; using mozilla::plugins::PluginThreadChild; using mozilla::dom::ContentProcessThread; using mozilla::dom::ContentProcessParent; using mozilla::dom::ContentProcessChild; using mozilla::ipc::TestShellParent; using mozilla::ipc::XPCShellEnvironment; using mozilla::test::TestParent; using mozilla::test::TestProcessParent; using mozilla::test::TestThreadChild; using mozilla::Monitor; using mozilla::MonitorAutoEnter; #endif static NS_DEFINE_CID(kAppShellCID, NS_APPSHELL_CID); void XRE_GetStaticComponents(nsStaticModuleInfo const **aStaticComponents, PRUint32 *aComponentCount) { *aStaticComponents = kPStaticModules; *aComponentCount = kStaticModuleCount; } nsresult XRE_LockProfileDirectory(nsILocalFile* aDirectory, nsISupports* *aLockObject) { nsCOMPtr lock; nsresult rv = NS_LockProfilePath(aDirectory, nsnull, nsnull, getter_AddRefs(lock)); if (NS_SUCCEEDED(rv)) NS_ADDREF(*aLockObject = lock); return rv; } static nsStaticModuleInfo *sCombined; static PRInt32 sInitCounter; nsresult XRE_InitEmbedding(nsILocalFile *aLibXULDirectory, nsILocalFile *aAppDirectory, nsIDirectoryServiceProvider *aAppDirProvider, nsStaticModuleInfo const *aStaticComponents, PRUint32 aStaticComponentCount) { // Initialize some globals to make nsXREDirProvider happy static char* kNullCommandLine[] = { nsnull }; gArgv = kNullCommandLine; gArgc = 0; NS_ENSURE_ARG(aLibXULDirectory); if (++sInitCounter > 1) // XXXbsmedberg is this really the right solution? return NS_OK; if (!aAppDirectory) aAppDirectory = aLibXULDirectory; nsresult rv; new nsXREDirProvider; // This sets gDirServiceProvider if (!gDirServiceProvider) return NS_ERROR_OUT_OF_MEMORY; rv = gDirServiceProvider->Initialize(aAppDirectory, aLibXULDirectory, aAppDirProvider); if (NS_FAILED(rv)) return rv; // Combine the toolkit static components and the app components. PRUint32 combinedCount = kStaticModuleCount + aStaticComponentCount; sCombined = new nsStaticModuleInfo[combinedCount]; if (!sCombined) return NS_ERROR_OUT_OF_MEMORY; memcpy(sCombined, kPStaticModules, sizeof(nsStaticModuleInfo) * kStaticModuleCount); memcpy(sCombined + kStaticModuleCount, aStaticComponents, sizeof(nsStaticModuleInfo) * aStaticComponentCount); rv = NS_InitXPCOM3(nsnull, aAppDirectory, gDirServiceProvider, sCombined, combinedCount); if (NS_FAILED(rv)) return rv; // We do not need to autoregister components here. The CheckUpdateFile() // bits in NS_InitXPCOM3 check for an .autoreg file. If the app wants // to autoregister every time (for instance, if it's debug), it can do // so after we return from this function. nsCOMPtr startupNotifier (do_CreateInstance(NS_APPSTARTUPNOTIFIER_CONTRACTID)); if (!startupNotifier) return NS_ERROR_FAILURE; startupNotifier->Observe(nsnull, APPSTARTUP_TOPIC, nsnull); return NS_OK; } void XRE_NotifyProfile() { NS_ASSERTION(gDirServiceProvider, "XRE_InitEmbedding was not called!"); gDirServiceProvider->DoStartup(); } void XRE_TermEmbedding() { if (--sInitCounter != 0) return; NS_ASSERTION(gDirServiceProvider, "XRE_TermEmbedding without XRE_InitEmbedding"); gDirServiceProvider->DoShutdown(); NS_ShutdownXPCOM(nsnull); delete [] sCombined; delete gDirServiceProvider; } const char* XRE_ChildProcessTypeToString(GeckoProcessType aProcessType) { return (aProcessType < GeckoProcessType_End) ? kGeckoProcessTypeString[aProcessType] : nsnull; } GeckoProcessType XRE_StringToChildProcessType(const char* aProcessTypeString) { for (int i = 0; i < (int) NS_ARRAY_LENGTH(kGeckoProcessTypeString); ++i) { if (!strcmp(kGeckoProcessTypeString[i], aProcessTypeString)) { return static_cast(i); } } return GeckoProcessType_Invalid; } #ifdef MOZ_IPC static GeckoProcessType sChildProcessType = GeckoProcessType_Default; nsresult XRE_InitChildProcess(int aArgc, char* aArgv[], GeckoProcessType aProcess) { NS_ENSURE_ARG_MIN(aArgc, 1); NS_ENSURE_ARG_POINTER(aArgv); NS_ENSURE_ARG_POINTER(aArgv[0]); if (PR_GetEnv("MOZ_DEBUG_CHILD_PROCESS")) { #ifdef OS_POSIX printf("\n\nCHILDCHILDCHILDCHILD\n debug me @%d\n\n", getpid()); sleep(30); #elif defined(OS_WIN) Sleep(30000); #endif } base::AtExitManager exitManager; CommandLine::Init(aArgc, aArgv); MessageLoopForIO mainMessageLoop; { ChildThread* mainThread; switch (aProcess) { case GeckoProcessType_Default: mainThread = new GeckoThread(); break; case GeckoProcessType_Plugin: mainThread = new PluginThreadChild(); break; case GeckoProcessType_Content: mainThread = new ContentProcessThread(); break; case GeckoProcessType_TestHarness: mainThread = new TestThreadChild(); break; default: NS_RUNTIMEABORT("Unknown main thread class"); } sChildProcessType = aProcess; ChildProcess process(mainThread); // Do IPC event loop MessageLoop::current()->Run(); } return NS_OK; } GeckoProcessType XRE_GetProcessType() { return sChildProcessType; } namespace { class MainFunctionRunnable : public nsRunnable { public: NS_DECL_NSIRUNNABLE MainFunctionRunnable(MainFunction aFunction, void* aData) : mFunction(aFunction), mData(aData) { NS_ASSERTION(aFunction, "Don't give me a null pointer!"); } private: MainFunction mFunction; void* mData; }; } /* anonymous namespace */ NS_IMETHODIMP MainFunctionRunnable::Run() { mFunction(mData); return NS_OK; } nsresult XRE_InitParentProcess(int aArgc, char* aArgv[], MainFunction aMainFunction, void* aMainFunctionData) { NS_ENSURE_ARG_MIN(aArgc, 1); NS_ENSURE_ARG_POINTER(aArgv); NS_ENSURE_ARG_POINTER(aArgv[0]); CommandLine::Init(aArgc, aArgv); ScopedXREEmbed embed; { embed.Start(); nsCOMPtr appShell(do_GetService(kAppShellCID)); NS_ENSURE_TRUE(appShell, NS_ERROR_FAILURE); if (aMainFunction) { nsCOMPtr runnable = new MainFunctionRunnable(aMainFunction, aMainFunctionData); NS_ENSURE_TRUE(runnable, NS_ERROR_OUT_OF_MEMORY); nsresult rv = NS_DispatchToCurrentThread(runnable); NS_ENSURE_SUCCESS(rv, rv); } // Do event loop if (NS_FAILED(appShell->Run())) { NS_WARNING("Failed to run appshell"); return NS_ERROR_FAILURE; } } return NS_OK; } NS_SPECIALIZE_TEMPLATE class nsAutoRefTraits : public nsPointerRefTraits { public: void Release(XPCShellEnvironment* aEnv) { XPCShellEnvironment::DestroyEnvironment(aEnv); } }; namespace { class QuitRunnable : public nsRunnable { public: NS_IMETHOD Run() { nsCOMPtr appShell(do_GetService(kAppShellCID)); NS_ENSURE_TRUE(appShell, NS_ERROR_FAILURE); return appShell->Exit(); } }; int TestShellMain(int argc, char** argv) { // Make sure we quit when we exit this function. nsIRunnable* quitRunnable = new QuitRunnable(); NS_ENSURE_TRUE(quitRunnable, 1); nsresult rv = NS_DispatchToCurrentThread(quitRunnable); NS_ENSURE_SUCCESS(rv, 1); nsAutoRef env(XPCShellEnvironment::CreateEnvironment()); NS_ENSURE_TRUE(env, 1); ContentProcessParent* childProcess = ContentProcessParent::GetSingleton(); NS_ENSURE_TRUE(childProcess, 1); TestShellParent* testShellParent = childProcess->CreateTestShell(); NS_ENSURE_TRUE(testShellParent, 1); testShellParent->SetXPCShell(env); bool ok = env->DefineIPCCommands(testShellParent); NS_ENSURE_TRUE(ok, 1); const char* filename = argc > 1 ? argv[1] : nsnull; env->Process(filename); testShellParent->SetXPCShell(nsnull); return env->ExitCode(); } struct TestShellData { int* result; int argc; char** argv; }; void TestShellMainWrapper(void* aData) { NS_ASSERTION(aData, "Don't give me a null pointer!"); TestShellData& testShellData = *static_cast(aData); *testShellData.result = TestShellMain(testShellData.argc, testShellData.argv); } } /* anonymous namespace */ int XRE_RunTestShell(int aArgc, char* aArgv[]) { int result; TestShellData data = { &result, aArgc, aArgv }; nsresult rv = XRE_InitParentProcess(aArgc, aArgv, TestShellMainWrapper, &data); NS_ENSURE_SUCCESS(rv, 1); return result; } //----------------------------------------------------------------------------- // TestHarness static void IPCTestHarnessMain(void* data) { TestProcessParent* subprocess = new TestProcessParent(); // leaks bool launched = subprocess->SyncLaunch(); NS_ASSERTION(launched, "can't launch subprocess"); TestParent* parent = new TestParent(); // leaks parent->Open(subprocess->GetChannel()); parent->DoStuff(); } int XRE_RunIPCTestHarness(int aArgc, char* aArgv[]) { nsresult rv = XRE_InitParentProcess( aArgc, aArgv, IPCTestHarnessMain, NULL); NS_ENSURE_SUCCESS(rv, 1); return 0; } nsresult XRE_RunAppShell() { nsCOMPtr appShell(do_GetService(kAppShellCID)); NS_ENSURE_TRUE(appShell, NS_ERROR_FAILURE); return appShell->Run(); } template<> struct RunnableMethodTraits { static void RetainCallee(ContentProcessChild* obj) { } static void ReleaseCallee(ContentProcessChild* obj) { } }; void XRE_ShutdownChildProcess(MessageLoop* aUILoop) { if (aUILoop) { NS_ASSERTION(!NS_IsMainThread(), "Wrong thread!"); aUILoop->PostTask(FROM_HERE, NewRunnableMethod(ContentProcessChild::GetSingleton(), &ContentProcessChild::Quit)); return; } NS_ASSERTION(NS_IsMainThread(), "Wrong thread!"); nsCOMPtr appShell(do_GetService(kAppShellCID)); if (appShell) { appShell->Exit(); } } ContentProcessParent* XRE_GetContentProcessParent() { if (XRE_GetProcessType() == GeckoProcessType_Default) { return ContentProcessParent::GetSingleton(); } NS_WARNING("Called XRE_GetContentProcessParent in child process!"); return nsnull; } ContentProcessChild* XRE_GetContentProcessChild() { if (XRE_GetProcessType() != GeckoProcessType_Default) { return ContentProcessChild::GetSingleton(); } NS_WARNING("Called XRE_GetContentProcessChild in parent process!"); return nsnull; } #endif