/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* 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 "GMPParent.h" #include "mozilla/Logging.h" #include "nsComponentManagerUtils.h" #include "nsComponentManagerUtils.h" #include "nsIInputStream.h" #include "nsILineInputStream.h" #include "nsNetUtil.h" #include "nsCharSeparatedTokenizer.h" #include "nsThreadUtils.h" #include "nsIRunnable.h" #include "nsIWritablePropertyBag2.h" #include "mozIGeckoMediaPluginService.h" #include "mozilla/ipc/GeckoChildProcessHost.h" #include "mozilla/SyncRunnable.h" #include "mozilla/unused.h" #include "nsIObserverService.h" #include "GMPTimerParent.h" #include "runnable_utils.h" #if defined(XP_LINUX) && defined(MOZ_GMP_SANDBOX) #include "mozilla/SandboxInfo.h" #endif #include "GMPContentParent.h" #include "mozilla/dom/CrashReporterParent.h" using mozilla::dom::CrashReporterParent; using mozilla::ipc::GeckoChildProcessHost; #ifdef MOZ_CRASHREPORTER #include "nsPrintfCString.h" using CrashReporter::AnnotationTable; using CrashReporter::GetIDFromMinidump; #endif #include "mozilla/Telemetry.h" namespace mozilla { #undef LOG #undef LOGD extern LogModule* GetGMPLog(); #define LOG(level, x, ...) MOZ_LOG(GetGMPLog(), (level), (x, ##__VA_ARGS__)) #define LOGD(x, ...) LOG(mozilla::LogLevel::Debug, "GMPParent[%p|childPid=%d] " x, this, mChildPid, ##__VA_ARGS__) #ifdef __CLASS__ #undef __CLASS__ #endif #define __CLASS__ "GMPParent" namespace gmp { GMPParent::GMPParent() : mState(GMPStateNotLoaded) , mProcess(nullptr) , mDeleteProcessOnlyOnUnload(false) , mAbnormalShutdownInProgress(false) , mIsBlockingDeletion(false) , mCanDecrypt(false) , mGMPContentChildCount(0) , mAsyncShutdownRequired(false) , mAsyncShutdownInProgress(false) , mChildPid(0) , mHoldingSelfRef(false) { LOGD("GMPParent ctor"); mPluginId = GeckoChildProcessHost::GetUniqueID(); } GMPParent::~GMPParent() { // Can't Close or Destroy the process here, since destruction is MainThread only MOZ_ASSERT(NS_IsMainThread()); LOGD("GMPParent dtor"); MOZ_ASSERT(!mProcess); } nsresult GMPParent::CloneFrom(const GMPParent* aOther) { MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); MOZ_ASSERT(aOther->mDirectory && aOther->mService, "null plugin directory"); return Init(aOther->mService, aOther->mDirectory); } nsresult GMPParent::Init(GeckoMediaPluginServiceParent* aService, nsIFile* aPluginDir) { MOZ_ASSERT(aPluginDir); MOZ_ASSERT(aService); MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); mService = aService; mDirectory = aPluginDir; // aPluginDir is // // where should be gmp-gmpopenh264 nsCOMPtr parent; nsresult rv = aPluginDir->GetParent(getter_AddRefs(parent)); if (NS_FAILED(rv)) { return rv; } nsAutoString parentLeafName; rv = parent->GetLeafName(parentLeafName); if (NS_FAILED(rv)) { return rv; } LOGD("%s: for %s", __FUNCTION__, NS_LossyConvertUTF16toASCII(parentLeafName).get()); MOZ_ASSERT(parentLeafName.Length() > 4); mName = Substring(parentLeafName, 4); return ReadGMPMetaData(); } void GMPParent::Crash() { if (mState != GMPStateNotLoaded) { Unused << SendCrashPluginNow(); } } nsresult GMPParent::LoadProcess() { MOZ_ASSERT(mDirectory, "Plugin directory cannot be NULL!"); MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); MOZ_ASSERT(mState == GMPStateNotLoaded); nsAutoString path; if (NS_FAILED(mDirectory->GetPath(path))) { return NS_ERROR_FAILURE; } LOGD("%s: for %s", __FUNCTION__, NS_ConvertUTF16toUTF8(path).get()); if (!mProcess) { mProcess = new GMPProcessParent(NS_ConvertUTF16toUTF8(path).get()); if (!mProcess->Launch(30 * 1000)) { LOGD("%s: Failed to launch new child process", __FUNCTION__); mProcess->Delete(); mProcess = nullptr; return NS_ERROR_FAILURE; } mChildPid = base::GetProcId(mProcess->GetChildProcessHandle()); LOGD("%s: Launched new child process", __FUNCTION__); bool opened = Open(mProcess->GetChannel(), base::GetProcId(mProcess->GetChildProcessHandle())); if (!opened) { LOGD("%s: Failed to open channel to new child process", __FUNCTION__); mProcess->Delete(); mProcess = nullptr; return NS_ERROR_FAILURE; } LOGD("%s: Opened channel to new child process", __FUNCTION__); bool ok = SendSetNodeId(mNodeId); if (!ok) { LOGD("%s: Failed to send node id to child process", __FUNCTION__); return NS_ERROR_FAILURE; } LOGD("%s: Sent node id to child process", __FUNCTION__); // Intr call to block initialization on plugin load. ok = CallStartPlugin(); if (!ok) { LOGD("%s: Failed to send start to child process", __FUNCTION__); return NS_ERROR_FAILURE; } LOGD("%s: Sent StartPlugin to child process", __FUNCTION__); } mState = GMPStateLoaded; // Hold a self ref while the child process is alive. This ensures that // during shutdown the GMPParent stays alive long enough to // terminate the child process. MOZ_ASSERT(!mHoldingSelfRef); mHoldingSelfRef = true; AddRef(); return NS_OK; } // static void GMPParent::AbortWaitingForGMPAsyncShutdown(nsITimer* aTimer, void* aClosure) { NS_WARNING("Timed out waiting for GMP async shutdown!"); GMPParent* parent = reinterpret_cast(aClosure); MOZ_ASSERT(parent->mService); #if defined(MOZ_CRASHREPORTER) parent->mService->SetAsyncShutdownPluginState(parent, 'G', NS_LITERAL_CSTRING("Timed out waiting for async shutdown")); #endif parent->mService->AsyncShutdownComplete(parent); } nsresult GMPParent::EnsureAsyncShutdownTimeoutSet() { MOZ_ASSERT(mAsyncShutdownRequired); if (mAsyncShutdownTimeout) { return NS_OK; } nsresult rv; mAsyncShutdownTimeout = do_CreateInstance(NS_TIMER_CONTRACTID, &rv); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // Set timer to abort waiting for plugin to shutdown if it takes // too long. rv = mAsyncShutdownTimeout->SetTarget(mGMPThread); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t timeout = GMP_DEFAULT_ASYNC_SHUTDONW_TIMEOUT; RefPtr service = GeckoMediaPluginServiceParent::GetSingleton(); if (service) { timeout = service->AsyncShutdownTimeoutMs(); } rv = mAsyncShutdownTimeout->InitWithFuncCallback( &AbortWaitingForGMPAsyncShutdown, this, timeout, nsITimer::TYPE_ONE_SHOT); Unused << NS_WARN_IF(NS_FAILED(rv)); return rv; } bool GMPParent::RecvPGMPContentChildDestroyed() { --mGMPContentChildCount; if (!IsUsed()) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'E', NS_LITERAL_CSTRING("Last content child destroyed")); } #endif CloseIfUnused(); } #if defined(MOZ_CRASHREPORTER) else { if (mService) { mService->SetAsyncShutdownPluginState(this, 'F', nsPrintfCString("Content child destroyed, remaining: %u", mGMPContentChildCount)); } } #endif return true; } void GMPParent::CloseIfUnused() { MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); LOGD("%s: mAsyncShutdownRequired=%d", __FUNCTION__, mAsyncShutdownRequired); if ((mDeleteProcessOnlyOnUnload || mState == GMPStateLoaded || mState == GMPStateUnloading) && !IsUsed()) { // Ensure all timers are killed. for (uint32_t i = mTimers.Length(); i > 0; i--) { mTimers[i - 1]->Shutdown(); } if (mAsyncShutdownRequired) { if (!mAsyncShutdownInProgress) { LOGD("%s: sending async shutdown notification", __FUNCTION__); #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'H', NS_LITERAL_CSTRING("Sent BeginAsyncShutdown")); } #endif mAsyncShutdownInProgress = true; if (!SendBeginAsyncShutdown()) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'I', NS_LITERAL_CSTRING("Could not send BeginAsyncShutdown - Aborting async shutdown")); } #endif AbortAsyncShutdown(); } else if (NS_FAILED(EnsureAsyncShutdownTimeoutSet())) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'J', NS_LITERAL_CSTRING("Could not start timer after sending BeginAsyncShutdown - Aborting async shutdown")); } #endif AbortAsyncShutdown(); } } } else { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'K', NS_LITERAL_CSTRING("No (more) async-shutdown required")); } #endif // No async-shutdown, kill async-shutdown timer started in CloseActive(). AbortAsyncShutdown(); // Any async shutdown must be complete. Shutdown GMPStorage. for (size_t i = mStorage.Length(); i > 0; i--) { mStorage[i - 1]->Shutdown(); } Shutdown(); } } } void GMPParent::AbortAsyncShutdown() { MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); LOGD("%s", __FUNCTION__); if (mAsyncShutdownTimeout) { mAsyncShutdownTimeout->Cancel(); mAsyncShutdownTimeout = nullptr; } if (!mAsyncShutdownRequired || !mAsyncShutdownInProgress) { return; } RefPtr kungFuDeathGrip(this); mService->AsyncShutdownComplete(this); mAsyncShutdownRequired = false; mAsyncShutdownInProgress = false; CloseIfUnused(); } void GMPParent::CloseActive(bool aDieWhenUnloaded) { LOGD("%s: state %d", __FUNCTION__, mState); MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); if (aDieWhenUnloaded) { mDeleteProcessOnlyOnUnload = true; // don't allow this to go back... } if (mState == GMPStateLoaded) { mState = GMPStateUnloading; } if (mState != GMPStateNotLoaded && IsUsed()) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'A', nsPrintfCString("Sent CloseActive, content children to close: %u", mGMPContentChildCount)); } #endif if (!SendCloseActive()) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'B', NS_LITERAL_CSTRING("Could not send CloseActive - Aborting async shutdown")); } #endif AbortAsyncShutdown(); } else if (IsUsed()) { // We're expecting RecvPGMPContentChildDestroyed's -> Start async-shutdown timer now if needed. if (mAsyncShutdownRequired && NS_FAILED(EnsureAsyncShutdownTimeoutSet())) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'C', NS_LITERAL_CSTRING("Could not start timer after sending CloseActive - Aborting async shutdown")); } #endif AbortAsyncShutdown(); } } else { // We're not expecting any RecvPGMPContentChildDestroyed // -> Call CloseIfUnused() now, to run async shutdown if necessary. // Note that CloseIfUnused() may have already been called from a prior // RecvPGMPContentChildDestroyed(), however depending on the state at // that time, it might not have proceeded with shutdown; And calling it // again after shutdown is fine because after the first one we'll be in // GMPStateNotLoaded. #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'D', NS_LITERAL_CSTRING("Content children already destroyed")); } #endif CloseIfUnused(); } } } void GMPParent::MarkForDeletion() { mDeleteProcessOnlyOnUnload = true; mIsBlockingDeletion = true; } bool GMPParent::IsMarkedForDeletion() { return mIsBlockingDeletion; } void GMPParent::Shutdown() { LOGD("%s", __FUNCTION__); MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); MOZ_ASSERT(!mAsyncShutdownTimeout, "Should have canceled shutdown timeout"); if (mAbnormalShutdownInProgress) { return; } MOZ_ASSERT(!IsUsed()); if (mState == GMPStateNotLoaded || mState == GMPStateClosing) { return; } RefPtr self(this); DeleteProcess(); // XXX Get rid of mDeleteProcessOnlyOnUnload and this code when // Bug 1043671 is fixed if (!mDeleteProcessOnlyOnUnload) { // Destroy ourselves and rise from the fire to save memory mService->ReAddOnGMPThread(self); } // else we've been asked to die and stay dead MOZ_ASSERT(mState == GMPStateNotLoaded); } class NotifyGMPShutdownTask : public nsRunnable { public: explicit NotifyGMPShutdownTask(const nsAString& aNodeId) : mNodeId(aNodeId) { } NS_IMETHOD Run() { MOZ_ASSERT(NS_IsMainThread()); nsCOMPtr obsService = mozilla::services::GetObserverService(); MOZ_ASSERT(obsService); if (obsService) { obsService->NotifyObservers(nullptr, "gmp-shutdown", mNodeId.get()); } return NS_OK; } nsString mNodeId; }; void GMPParent::ChildTerminated() { RefPtr self(this); nsIThread* gmpThread = GMPThread(); if (!gmpThread) { // Bug 1163239 - this can happen on shutdown. // PluginTerminated removes the GMP from the GMPService. // On shutdown we can have this case where it is already been // removed so there is no harm in not trying to remove it again. LOGD("%s::%s: GMPThread() returned nullptr.", __CLASS__, __FUNCTION__); } else { gmpThread->Dispatch(NS_NewRunnableMethodWithArg>( mService, &GeckoMediaPluginServiceParent::PluginTerminated, self), NS_DISPATCH_NORMAL); } } void GMPParent::DeleteProcess() { LOGD("%s", __FUNCTION__); if (mState != GMPStateClosing) { // Don't Close() twice! // Probably remove when bug 1043671 is resolved mState = GMPStateClosing; Close(); } mProcess->Delete(NS_NewRunnableMethod(this, &GMPParent::ChildTerminated)); LOGD("%s: Shut down process", __FUNCTION__); mProcess = nullptr; mState = GMPStateNotLoaded; NS_DispatchToMainThread( new NotifyGMPShutdownTask(NS_ConvertUTF8toUTF16(mNodeId)), NS_DISPATCH_NORMAL); if (mHoldingSelfRef) { Release(); mHoldingSelfRef = false; } } GMPState GMPParent::State() const { return mState; } // Not changing to use mService since we'll be removing it nsIThread* GMPParent::GMPThread() { if (!mGMPThread) { nsCOMPtr mps = do_GetService("@mozilla.org/gecko-media-plugin-service;1"); MOZ_ASSERT(mps); if (!mps) { return nullptr; } // Not really safe if we just grab to the mGMPThread, as we don't know // what thread we're running on and other threads may be trying to // access this without locks! However, debug only, and primary failure // mode outside of compiler-helped TSAN is a leak. But better would be // to use swap() under a lock. mps->GetThread(getter_AddRefs(mGMPThread)); MOZ_ASSERT(mGMPThread); } return mGMPThread; } bool GMPParent::SupportsAPI(const nsCString& aAPI, const nsCString& aTag) { for (uint32_t i = 0; i < mCapabilities.Length(); i++) { if (!mCapabilities[i]->mAPIName.Equals(aAPI)) { continue; } nsTArray& tags = mCapabilities[i]->mAPITags; for (uint32_t j = 0; j < tags.Length(); j++) { if (tags[j].Equals(aTag)) { return true; } } } return false; } bool GMPParent::EnsureProcessLoaded() { if (mState == GMPStateLoaded) { return true; } if (mState == GMPStateClosing || mState == GMPStateUnloading) { return false; } nsresult rv = LoadProcess(); return NS_SUCCEEDED(rv); } #ifdef MOZ_CRASHREPORTER void GMPParent::WriteExtraDataForMinidump(CrashReporter::AnnotationTable& notes) { notes.Put(NS_LITERAL_CSTRING("GMPPlugin"), NS_LITERAL_CSTRING("1")); notes.Put(NS_LITERAL_CSTRING("PluginFilename"), NS_ConvertUTF16toUTF8(mName)); notes.Put(NS_LITERAL_CSTRING("PluginName"), mDisplayName); notes.Put(NS_LITERAL_CSTRING("PluginVersion"), mVersion); } void GMPParent::GetCrashID(nsString& aResult) { CrashReporterParent* cr = static_cast(LoneManagedOrNullAsserts(ManagedPCrashReporterParent())); if (NS_WARN_IF(!cr)) { return; } AnnotationTable notes(4); WriteExtraDataForMinidump(notes); nsCOMPtr dumpFile; TakeMinidump(getter_AddRefs(dumpFile), nullptr); if (!dumpFile) { NS_WARNING("GMP crash without crash report"); aResult = mName; aResult += '-'; AppendUTF8toUTF16(mVersion, aResult); return; } GetIDFromMinidump(dumpFile, aResult); cr->GenerateCrashReportForMinidump(dumpFile, ¬es); } static void GMPNotifyObservers(const uint32_t aPluginID, const nsACString& aPluginName, const nsAString& aPluginDumpID) { nsCOMPtr obs = mozilla::services::GetObserverService(); nsCOMPtr propbag = do_CreateInstance("@mozilla.org/hash-property-bag;1"); if (obs && propbag) { propbag->SetPropertyAsUint32(NS_LITERAL_STRING("pluginID"), aPluginID); propbag->SetPropertyAsACString(NS_LITERAL_STRING("pluginName"), aPluginName); propbag->SetPropertyAsAString(NS_LITERAL_STRING("pluginDumpID"), aPluginDumpID); obs->NotifyObservers(propbag, "gmp-plugin-crash", nullptr); } RefPtr service = gmp::GeckoMediaPluginService::GetGeckoMediaPluginService(); if (service) { service->RunPluginCrashCallbacks(aPluginID, aPluginName); } } #endif void GMPParent::ActorDestroy(ActorDestroyReason aWhy) { LOGD("%s: (%d)", __FUNCTION__, (int)aWhy); #ifdef MOZ_CRASHREPORTER if (AbnormalShutdown == aWhy) { Telemetry::Accumulate(Telemetry::SUBPROCESS_ABNORMAL_ABORT, NS_LITERAL_CSTRING("gmplugin"), 1); nsString dumpID; GetCrashID(dumpID); // NotifyObservers is mainthread-only NS_DispatchToMainThread(WrapRunnableNM(&GMPNotifyObservers, mPluginId, mDisplayName, dumpID), NS_DISPATCH_NORMAL); } #endif // warn us off trying to close again mState = GMPStateClosing; mAbnormalShutdownInProgress = true; CloseActive(false); // Normal Shutdown() will delete the process on unwind. if (AbnormalShutdown == aWhy) { RefPtr self(this); if (mAsyncShutdownRequired) { #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'M', NS_LITERAL_CSTRING("Actor destroyed")); } #endif mService->AsyncShutdownComplete(this); mAsyncShutdownRequired = false; } // Must not call Close() again in DeleteProcess(), as we'll recurse // infinitely if we do. MOZ_ASSERT(mState == GMPStateClosing); DeleteProcess(); // Note: final destruction will be Dispatched to ourself mService->ReAddOnGMPThread(self); } } mozilla::dom::PCrashReporterParent* GMPParent::AllocPCrashReporterParent(const NativeThreadId& aThread) { #ifndef MOZ_CRASHREPORTER MOZ_ASSERT(false, "Should only be sent if crash reporting is enabled."); #endif CrashReporterParent* cr = new CrashReporterParent(); cr->SetChildData(aThread, GeckoProcessType_GMPlugin); return cr; } bool GMPParent::DeallocPCrashReporterParent(PCrashReporterParent* aCrashReporter) { delete aCrashReporter; return true; } PGMPStorageParent* GMPParent::AllocPGMPStorageParent() { GMPStorageParent* p = new GMPStorageParent(mNodeId, this); mStorage.AppendElement(p); // Addrefs, released in DeallocPGMPStorageParent. return p; } bool GMPParent::DeallocPGMPStorageParent(PGMPStorageParent* aActor) { GMPStorageParent* p = static_cast(aActor); p->Shutdown(); mStorage.RemoveElement(p); return true; } bool GMPParent::RecvPGMPStorageConstructor(PGMPStorageParent* aActor) { GMPStorageParent* p = (GMPStorageParent*)aActor; if (NS_WARN_IF(NS_FAILED(p->Init()))) { return false; } return true; } bool GMPParent::RecvPGMPTimerConstructor(PGMPTimerParent* actor) { return true; } PGMPTimerParent* GMPParent::AllocPGMPTimerParent() { GMPTimerParent* p = new GMPTimerParent(GMPThread()); mTimers.AppendElement(p); // Released in DeallocPGMPTimerParent, or on shutdown. return p; } bool GMPParent::DeallocPGMPTimerParent(PGMPTimerParent* aActor) { GMPTimerParent* p = static_cast(aActor); p->Shutdown(); mTimers.RemoveElement(p); return true; } nsresult ParseNextRecord(nsILineInputStream* aLineInputStream, const nsCString& aPrefix, nsCString& aResult, bool& aMoreLines) { nsAutoCString record; nsresult rv = aLineInputStream->ReadLine(record, &aMoreLines); if (NS_FAILED(rv)) { return rv; } if (record.Length() <= aPrefix.Length() || !Substring(record, 0, aPrefix.Length()).Equals(aPrefix)) { return NS_ERROR_FAILURE; } aResult = Substring(record, aPrefix.Length()); aResult.Trim("\b\t\r\n "); return NS_OK; } nsresult GMPParent::ReadGMPMetaData() { MOZ_ASSERT(mDirectory, "Plugin directory cannot be NULL!"); MOZ_ASSERT(!mName.IsEmpty(), "Plugin mName cannot be empty!"); nsCOMPtr infoFile; nsresult rv = mDirectory->Clone(getter_AddRefs(infoFile)); if (NS_FAILED(rv)) { return rv; } infoFile->AppendRelativePath(mName + NS_LITERAL_STRING(".info")); nsCOMPtr inputStream; rv = NS_NewLocalFileInputStream(getter_AddRefs(inputStream), infoFile); if (NS_FAILED(rv)) { return rv; } nsCOMPtr lineInputStream = do_QueryInterface(inputStream, &rv); if (NS_FAILED(rv)) { return rv; } nsCString value; bool moreLines = false; // 'Name:' record nsCString prefix = NS_LITERAL_CSTRING("Name:"); rv = ParseNextRecord(lineInputStream, prefix, value, moreLines); if (NS_FAILED(rv)) { return rv; } if (value.IsEmpty()) { // Not OK for name to be empty. Must have one non-whitespace character. return NS_ERROR_FAILURE; } mDisplayName = value; // 'Description:' record if (!moreLines) { return NS_ERROR_FAILURE; } prefix = NS_LITERAL_CSTRING("Description:"); rv = ParseNextRecord(lineInputStream, prefix, value, moreLines); if (NS_FAILED(rv)) { return rv; } mDescription = value; // 'Version:' record if (!moreLines) { return NS_ERROR_FAILURE; } prefix = NS_LITERAL_CSTRING("Version:"); rv = ParseNextRecord(lineInputStream, prefix, value, moreLines); if (NS_FAILED(rv)) { return rv; } mVersion = value; // 'Capability:' record if (!moreLines) { return NS_ERROR_FAILURE; } prefix = NS_LITERAL_CSTRING("APIs:"); rv = ParseNextRecord(lineInputStream, prefix, value, moreLines); if (NS_FAILED(rv)) { return rv; } nsCCharSeparatedTokenizer apiTokens(value, ','); while (apiTokens.hasMoreTokens()) { nsAutoCString api(apiTokens.nextToken()); api.StripWhitespace(); if (api.IsEmpty()) { continue; } int32_t tagsStart = api.FindChar('['); if (tagsStart == 0) { // Not allowed to be the first character. // API name must be at least one character. continue; } auto cap = new GMPCapability(); if (tagsStart == -1) { // No tags. cap->mAPIName.Assign(api); } else { auto tagsEnd = api.FindChar(']'); if (tagsEnd == -1 || tagsEnd < tagsStart) { // Invalid syntax, skip whole capability. delete cap; continue; } cap->mAPIName.Assign(Substring(api, 0, tagsStart)); if ((tagsEnd - tagsStart) > 1) { const nsDependentCSubstring ts(Substring(api, tagsStart + 1, tagsEnd - tagsStart - 1)); nsCCharSeparatedTokenizer tagTokens(ts, ':'); while (tagTokens.hasMoreTokens()) { const nsDependentCSubstring tag(tagTokens.nextToken()); cap->mAPITags.AppendElement(tag); } } } // We support the current GMPDecryptor version, and the previous. // We Adapt the previous to the current in the GMPContentChild. if (cap->mAPIName.EqualsLiteral(GMP_API_DECRYPTOR_BACKWARDS_COMPAT)) { cap->mAPIName.AssignLiteral(GMP_API_DECRYPTOR); } if (cap->mAPIName.EqualsLiteral(GMP_API_DECRYPTOR)) { mCanDecrypt = true; #if defined(XP_LINUX) && defined(MOZ_GMP_SANDBOX) if (!mozilla::SandboxInfo::Get().CanSandboxMedia()) { printf_stderr("GMPParent::ReadGMPMetaData: Plugin \"%s\" is an EME CDM" " but this system can't sandbox it; not loading.\n", mDisplayName.get()); delete cap; return NS_ERROR_FAILURE; } #endif } mCapabilities.AppendElement(cap); } if (mCapabilities.IsEmpty()) { return NS_ERROR_FAILURE; } return NS_OK; } bool GMPParent::CanBeSharedCrossNodeIds() const { return !mAsyncShutdownInProgress && mNodeId.IsEmpty() && // XXX bug 1159300 hack -- maybe remove after openh264 1.4 // We don't want to use CDM decoders for non-encrypted playback // just yet; especially not for WebRTC. Don't allow CDMs to be used // without a node ID. !mCanDecrypt; } bool GMPParent::CanBeUsedFrom(const nsACString& aNodeId) const { return !mAsyncShutdownInProgress && ((mNodeId.IsEmpty() && State() == GMPStateNotLoaded) || mNodeId == aNodeId); } void GMPParent::SetNodeId(const nsACString& aNodeId) { MOZ_ASSERT(!aNodeId.IsEmpty()); MOZ_ASSERT(CanBeUsedFrom(aNodeId)); mNodeId = aNodeId; } const nsCString& GMPParent::GetDisplayName() const { return mDisplayName; } const nsCString& GMPParent::GetVersion() const { return mVersion; } const uint32_t GMPParent::GetPluginId() const { return mPluginId; } bool GMPParent::RecvAsyncShutdownRequired() { LOGD("%s", __FUNCTION__); if (mAsyncShutdownRequired) { NS_WARNING("Received AsyncShutdownRequired message more than once!"); return true; } mAsyncShutdownRequired = true; mService->AsyncShutdownNeeded(this); return true; } bool GMPParent::RecvAsyncShutdownComplete() { LOGD("%s", __FUNCTION__); MOZ_ASSERT(mAsyncShutdownRequired); #if defined(MOZ_CRASHREPORTER) if (mService) { mService->SetAsyncShutdownPluginState(this, 'L', NS_LITERAL_CSTRING("Received AsyncShutdownComplete")); } #endif AbortAsyncShutdown(); return true; } class RunCreateContentParentCallbacks : public nsRunnable { public: explicit RunCreateContentParentCallbacks(GMPContentParent* aGMPContentParent) : mGMPContentParent(aGMPContentParent) { } void TakeCallbacks(nsTArray>& aCallbacks) { mCallbacks.SwapElements(aCallbacks); } NS_IMETHOD Run() { for (uint32_t i = 0, length = mCallbacks.Length(); i < length; ++i) { mCallbacks[i]->Done(mGMPContentParent); } return NS_OK; } private: RefPtr mGMPContentParent; nsTArray> mCallbacks; }; PGMPContentParent* GMPParent::AllocPGMPContentParent(Transport* aTransport, ProcessId aOtherPid) { MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); MOZ_ASSERT(!mGMPContentParent); mGMPContentParent = new GMPContentParent(this); mGMPContentParent->Open(aTransport, aOtherPid, XRE_GetIOMessageLoop(), ipc::ParentSide); RefPtr runCallbacks = new RunCreateContentParentCallbacks(mGMPContentParent); runCallbacks->TakeCallbacks(mCallbacks); NS_DispatchToCurrentThread(runCallbacks); MOZ_ASSERT(mCallbacks.IsEmpty()); return mGMPContentParent; } bool GMPParent::GetGMPContentParent(UniquePtr&& aCallback) { LOGD("%s %p", __FUNCTION__, this); MOZ_ASSERT(GMPThread() == NS_GetCurrentThread()); if (mGMPContentParent) { aCallback->Done(mGMPContentParent); } else { mCallbacks.AppendElement(Move(aCallback)); // If we don't have a GMPContentParent and we try to get one for the first // time (mCallbacks.Length() == 1) then call PGMPContent::Open. If more // calls to GetGMPContentParent happen before mGMPContentParent has been // set then we should just store them, so that they get called when we set // mGMPContentParent as a result of the PGMPContent::Open call. if (mCallbacks.Length() == 1) { if (!EnsureProcessLoaded() || !PGMPContent::Open(this)) { return false; } // We want to increment this as soon as possible, to avoid that we'd try // to shut down the GMP process while we're still trying to get a // PGMPContentParent actor. ++mGMPContentChildCount; } } return true; } already_AddRefed GMPParent::ForgetGMPContentParent() { MOZ_ASSERT(mCallbacks.IsEmpty()); return Move(mGMPContentParent.forget()); } bool GMPParent::EnsureProcessLoaded(base::ProcessId* aID) { if (!EnsureProcessLoaded()) { return false; } *aID = OtherPid(); return true; } bool GMPParent::Bridge(GMPServiceParent* aGMPServiceParent) { if (NS_FAILED(PGMPContent::Bridge(aGMPServiceParent, this))) { return false; } ++mGMPContentChildCount; return true; } nsString GMPParent::GetPluginBaseName() const { return NS_LITERAL_STRING("gmp-") + mName; } } // namespace gmp } // namespace mozilla #undef LOG #undef LOGD