/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "mozilla/ipc/ProcessChild.h" #include "Endpoint.h" #include "nsDebug.h" #ifdef XP_WIN # include // for _exit() # include #else # include // for _exit() # include # include "base/eintr_wrapper.h" # include "prenv.h" #endif #include "nsAppRunner.h" #include "mozilla/AppShutdown.h" #include "mozilla/ipc/IOThread.h" #include "mozilla/ipc/ProcessUtils.h" #include "mozilla/GeckoArgs.h" namespace mozilla { namespace ipc { ProcessChild* ProcessChild::gProcessChild; StaticMutex ProcessChild::gIPCShutdownStateLock; MOZ_CONSTINIT nsCString ProcessChild::gIPCShutdownStateAnnotation; ProcessChild::ProcessChild(IPC::Channel::ChannelHandle aClientChannel, ProcessId aParentPid, const nsID& aMessageChannelId) : mUILoop(MessageLoop::current()), mParentPid(aParentPid), mMessageChannelId(aMessageChannelId), mChildThread( MakeUnique(std::move(aClientChannel), aParentPid)) { MOZ_ASSERT(mUILoop, "UILoop should be created by now"); MOZ_ASSERT(!gProcessChild, "should only be one ProcessChild"); CrashReporter::RegisterAnnotationNSCString( CrashReporter::Annotation::IPCShutdownState, &gIPCShutdownStateAnnotation); gProcessChild = this; } /* static */ void ProcessChild::AddPlatformBuildID(geckoargs::ChildProcessArgs& aExtraArgs) { nsCString parentBuildID(mozilla::PlatformBuildID()); geckoargs::sParentBuildID.Put(parentBuildID.get(), aExtraArgs); } /* static */ bool ProcessChild::InitPrefs(int aArgc, char* aArgv[]) { Maybe prefsHandle = geckoargs::sPrefsHandle.Get(aArgc, aArgv); Maybe prefMapHandle = geckoargs::sPrefMapHandle.Get(aArgc, aArgv); if (prefsHandle.isNothing() || prefMapHandle.isNothing()) { return false; } SharedPreferenceDeserializer deserializer; return deserializer.DeserializeFromSharedMemory(std::move(*prefsHandle), std::move(*prefMapHandle)); } #ifdef ENABLE_TESTS // Allow tests to cause a synthetic delay/"hang" during child process // shutdown by setting environment variables. # ifdef XP_UNIX static void ReallySleep(int aSeconds) { struct ::timespec snooze = {aSeconds, 0}; HANDLE_EINTR(nanosleep(&snooze, &snooze)); } # else static void ReallySleep(int aSeconds) { ::Sleep(aSeconds * 1000); } # endif // Unix/Win static void SleepIfEnv(const char* aName) { if (auto* value = PR_GetEnv(aName)) { ReallySleep(atoi(value)); } } #else // not tests static void SleepIfEnv(const char* aName) {} #endif ProcessChild::~ProcessChild() { #ifdef NS_FREE_PERMANENT_DATA // In this case, we won't early-exit and we'll wait indefinitely for // child processes to terminate. This sleep is late enough that, in // content processes, it won't block parent process shutdown, so // we'll get into late IPC shutdown with processes still running. SleepIfEnv("MOZ_TEST_CHILD_EXIT_HANG"); #endif gIPCShutdownStateAnnotation = ""_ns; gProcessChild = nullptr; } /* static */ void ProcessChild::NotifiedImpendingShutdown() { AppShutdown::SetImpendingShutdown(); ProcessChild::AppendToIPCShutdownStateAnnotation( "NotifiedImpendingShutdown"_ns); } /* static */ void ProcessChild::QuickExit() { #ifndef NS_FREE_PERMANENT_DATA // In this case, we're going to terminate the child process before // we get to ~ProcessChild above (and terminate the parent process // before the shutdown hook in ProcessWatcher). Instead, blocking // earlier will let us exercise ProcessWatcher's kill timer. SleepIfEnv("MOZ_TEST_CHILD_EXIT_HANG"); #endif AppShutdown::DoImmediateExit(); } UntypedEndpoint ProcessChild::TakeInitialEndpoint() { return UntypedEndpoint{PrivateIPDLInterface{}, mChildThread->TakeInitialPort(), mMessageChannelId, EndpointProcInfo::Current(), EndpointProcInfo{.mPid = mParentPid, .mChildID = 0}}; } } // namespace ipc } // namespace mozilla