/* * Copyright (C) 2013, 2015 Apple Inc. All Rights Reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #import "config.h" #import "RemoteConnectionToTarget.h" #if ENABLE(REMOTE_INSPECTOR) #import "EventLoop.h" #import "RemoteAutomationTarget.h" #import "RemoteInspectionTarget.h" #import #import #import #if PLATFORM(IOS) #import #endif namespace Inspector { static StaticLock rwiQueueMutex; static CFRunLoopSourceRef rwiRunLoopSource; static RemoteTargetQueue* rwiQueue; static void RemoteTargetHandleRunSourceGlobal(void*) { ASSERT(CFRunLoopGetCurrent() == CFRunLoopGetMain()); ASSERT(rwiRunLoopSource); ASSERT(rwiQueue); RemoteTargetQueue queueCopy; { std::lock_guard lock(rwiQueueMutex); queueCopy = *rwiQueue; rwiQueue->clear(); } for (const auto& block : queueCopy) block(); } static void RemoteTargetQueueTaskOnGlobalQueue(void (^task)()) { ASSERT(rwiRunLoopSource); ASSERT(rwiQueue); { std::lock_guard lock(rwiQueueMutex); rwiQueue->append(task); } CFRunLoopSourceSignal(rwiRunLoopSource); CFRunLoopWakeUp(CFRunLoopGetMain()); } static void RemoteTargetInitializeGlobalQueue() { static dispatch_once_t pred; dispatch_once(&pred, ^{ rwiQueue = new RemoteTargetQueue; CFRunLoopSourceContext runLoopSourceContext = {0, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, RemoteTargetHandleRunSourceGlobal}; rwiRunLoopSource = CFRunLoopSourceCreate(kCFAllocatorDefault, 1, &runLoopSourceContext); // Add to the default run loop mode for default handling, and the JSContext remote inspector run loop mode when paused. CFRunLoopAddSource(CFRunLoopGetMain(), rwiRunLoopSource, kCFRunLoopDefaultMode); CFRunLoopAddSource(CFRunLoopGetMain(), rwiRunLoopSource, EventLoop::remoteInspectorRunLoopMode()); }); } static void RemoteTargetHandleRunSourceWithInfo(void* info) { RemoteConnectionToTarget *connectionToTarget = static_cast(info); RemoteTargetQueue queueCopy; { std::lock_guard lock(connectionToTarget->queueMutex()); queueCopy = connectionToTarget->queue(); connectionToTarget->clearQueue(); } for (const auto& block : queueCopy) block(); } RemoteConnectionToTarget::RemoteConnectionToTarget(RemoteControllableTarget* target, NSString *connectionIdentifier, NSString *destination) : m_target(target) , m_connectionIdentifier(connectionIdentifier) , m_destination(destination) { setupRunLoop(); } RemoteConnectionToTarget::~RemoteConnectionToTarget() { teardownRunLoop(); } Optional RemoteConnectionToTarget::targetIdentifier() const { return m_target ? Optional(m_target->targetIdentifier()) : Nullopt; } NSString *RemoteConnectionToTarget::connectionIdentifier() const { return [[m_connectionIdentifier copy] autorelease]; } NSString *RemoteConnectionToTarget::destination() const { return [[m_destination copy] autorelease]; } void RemoteConnectionToTarget::dispatchAsyncOnTarget(void (^block)()) { if (m_runLoop) { queueTaskOnPrivateRunLoop(block); return; } #if PLATFORM(IOS) if (WebCoreWebThreadIsEnabled && WebCoreWebThreadIsEnabled()) { WebCoreWebThreadRun(block); return; } #endif RemoteTargetQueueTaskOnGlobalQueue(block); } bool RemoteConnectionToTarget::setup(bool isAutomaticInspection, bool automaticallyPause) { std::lock_guard lock(m_targetMutex); if (!m_target) return false; ref(); dispatchAsyncOnTarget(^{ { std::lock_guard lock(m_targetMutex); if (!m_target || !m_target->remoteControlAllowed()) { RemoteInspector::singleton().setupFailed(targetIdentifier().valueOr(0)); m_target = nullptr; } else if (is(m_target)) { auto castedTarget = downcast(m_target); castedTarget->connect(this, isAutomaticInspection); m_connected = true; if (automaticallyPause) castedTarget->pause(); } else if (is(m_target)) { auto castedTarget = downcast(m_target); castedTarget->connect(this); m_connected = true; } } deref(); }); return true; } void RemoteConnectionToTarget::targetClosed() { std::lock_guard lock(m_targetMutex); m_target = nullptr; } void RemoteConnectionToTarget::close() { ref(); dispatchAsyncOnTarget(^{ { std::lock_guard lock(m_targetMutex); if (m_target) { if (m_connected) m_target->disconnect(this); m_target = nullptr; } } deref(); }); } void RemoteConnectionToTarget::sendMessageToTarget(NSString *message) { ref(); dispatchAsyncOnTarget(^{ { RemoteControllableTarget* target = nullptr; { std::lock_guard lock(m_targetMutex); if (!m_target) return; target = m_target; } target->dispatchMessageFromRemote(message); } deref(); }); } bool RemoteConnectionToTarget::sendMessageToFrontend(const String& message) { if (!m_target) return false; RemoteInspector::singleton().sendMessageToRemote(targetIdentifier().value(), message); return true; } void RemoteConnectionToTarget::setupRunLoop() { CFRunLoopRef targetRunLoop = m_target->targetRunLoop(); if (!targetRunLoop) { RemoteTargetInitializeGlobalQueue(); return; } m_runLoop = targetRunLoop; CFRunLoopSourceContext runLoopSourceContext = {0, this, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, RemoteTargetHandleRunSourceWithInfo}; m_runLoopSource = adoptCF(CFRunLoopSourceCreate(kCFAllocatorDefault, 1, &runLoopSourceContext)); CFRunLoopAddSource(m_runLoop.get(), m_runLoopSource.get(), kCFRunLoopDefaultMode); CFRunLoopAddSource(m_runLoop.get(), m_runLoopSource.get(), EventLoop::remoteInspectorRunLoopMode()); } void RemoteConnectionToTarget::teardownRunLoop() { if (!m_runLoop) return; CFRunLoopRemoveSource(m_runLoop.get(), m_runLoopSource.get(), kCFRunLoopDefaultMode); CFRunLoopRemoveSource(m_runLoop.get(), m_runLoopSource.get(), EventLoop::remoteInspectorRunLoopMode()); m_runLoop = nullptr; m_runLoopSource = nullptr; } void RemoteConnectionToTarget::queueTaskOnPrivateRunLoop(void (^block)()) { ASSERT(m_runLoop); { std::lock_guard lock(m_queueMutex); m_queue.append(block); } CFRunLoopSourceSignal(m_runLoopSource.get()); CFRunLoopWakeUp(m_runLoop.get()); } } // namespace Inspector #endif // ENABLE(REMOTE_INSPECTOR)