/* +----------------------------------------------------------------------+ | HipHop for PHP | +----------------------------------------------------------------------+ | Copyright (c) 2010-2013 Facebook, Inc. (http://www.facebook.com) | | Copyright (c) 1997-2010 The PHP Group | +----------------------------------------------------------------------+ | This source file is subject to version 3.01 of the PHP license, | | that is bundled with this package in the file LICENSE, and is | | available through the world-wide-web at the following url: | | http://www.php.net/license/3_01.txt | | If you did not receive a copy of the PHP license and are unable to | | obtain it through the world-wide-web, please send a note to | | license@php.net so we can mail you a copy immediately. | +----------------------------------------------------------------------+ */ #include "hphp/runtime/ext/ext_asio.h" #include "hphp/runtime/ext/ext_closure.h" #include "hphp/runtime/ext/asio/asio_context.h" #include "hphp/runtime/ext/asio/asio_session.h" #include "hphp/system/systemlib.h" namespace HPHP { /////////////////////////////////////////////////////////////////////////////// namespace { StaticString s_genArray(""); void putException(Object& exception_field, ObjectData* new_exception) { assert(new_exception); assert(new_exception->instanceof(SystemLib::s_ExceptionClass)); if (exception_field.isNull()) { exception_field = new_exception; } } } c_GenArrayWaitHandle::c_GenArrayWaitHandle(Class* cb) : c_BlockableWaitHandle(cb), m_exception() { } c_GenArrayWaitHandle::~c_GenArrayWaitHandle() { } void c_GenArrayWaitHandle::t___construct() { Object e(SystemLib::AllocInvalidOperationExceptionObject( "Use GenArrayWaitHandle::create() instead of constructor")); throw e; } void c_GenArrayWaitHandle::ti_setoncreatecallback(CVarRef callback) { if (!callback.isNull() && !callback.instanceof(c_Closure::s_cls)) { Object e(SystemLib::AllocInvalidArgumentExceptionObject( "Unable to set GenArrayWaitHandle::onCreate: on_create_cb not a closure")); throw e; } AsioSession::Get()->setOnGenArrayCreateCallback(callback.getObjectDataOrNull()); } Object c_GenArrayWaitHandle::ti_create(CArrRef dependencies) { Array deps = dependencies->copy(); for (ssize_t iter_pos = deps->iter_begin(); iter_pos != ArrayData::invalid_index; iter_pos = deps->iter_advance(iter_pos)) { TypedValue* current = deps->nvGetValueRef(iter_pos); if (UNLIKELY(current->m_type == KindOfRef)) { tvUnbox(current); } if (!c_WaitHandle::fromTypedValue(current) && !IS_NULL_TYPE(current->m_type)) { Object e(SystemLib::AllocInvalidArgumentExceptionObject( "Expected dependencies to be an array of WaitHandle instances")); throw e; } } Object exception; for (ssize_t iter_pos = deps->iter_begin(); iter_pos != ArrayData::invalid_index; iter_pos = deps->iter_advance(iter_pos)) { TypedValue* current = deps->nvGetValueRef(iter_pos); if (IS_NULL_TYPE(current->m_type)) { // {uninit,null} yields null tvWriteNull(current); continue; } assert(current->m_type == KindOfObject); assert(dynamic_cast(current->m_data.pobj)); auto child = static_cast(current->m_data.pobj); if (child->isSucceeded()) { tvSetIgnoreRef(*child->getResult(), *current); } else if (child->isFailed()) { putException(exception, child->getException()); } else { assert(dynamic_cast(child)); auto child_wh = static_cast(child); p_GenArrayWaitHandle my_wh = NEWOBJ(c_GenArrayWaitHandle)(); my_wh->initialize(exception, deps, iter_pos, child_wh); AsioSession* session = AsioSession::Get(); if (UNLIKELY(session->hasOnGenArrayCreateCallback())) { session->onGenArrayCreate(my_wh.get(), dependencies); } return my_wh; } } if (exception.isNull()) { TypedValue tv; tv.m_type = KindOfArray; tv.m_data.parr = deps.get(); return c_StaticResultWaitHandle::Create(&tv); } else { return c_StaticExceptionWaitHandle::Create(exception.get()); } } void c_GenArrayWaitHandle::initialize(CObjRef exception, CArrRef deps, ssize_t iter_pos, c_WaitableWaitHandle* child) { m_exception = exception; m_deps = deps; m_iterPos = iter_pos; try { blockOn(child); } catch (const Object& cycle_exception) { putException(m_exception, cycle_exception.get()); m_iterPos = m_deps->iter_advance(m_iterPos); onUnblocked(); } } void c_GenArrayWaitHandle::onUnblocked() { for (; m_iterPos != ArrayData::invalid_index; m_iterPos = m_deps->iter_advance(m_iterPos)) { TypedValue* current = m_deps->nvGetValueRef(m_iterPos); if (IS_NULL_TYPE(current->m_type)) { // {uninit,null} yields null tvWriteNull(current); continue; } assert(current->m_type == KindOfObject); assert(dynamic_cast(current->m_data.pobj)); auto child = static_cast(current->m_data.pobj); if (child->isSucceeded()) { tvSetIgnoreRef(*child->getResult(), *current); } else if (child->isFailed()) { putException(m_exception, child->getException()); } else { assert(dynamic_cast(child)); auto child_wh = static_cast(child); try { blockOn(child_wh); return; } catch (const Object& cycle_exception) { putException(m_exception, cycle_exception.get()); } } } if (m_exception.isNull()) { TypedValue result; result.m_type = KindOfArray; result.m_data.parr = m_deps.get(); setResult(&result); m_deps = nullptr; } else { setException(m_exception.get()); m_exception = nullptr; m_deps = nullptr; } } String c_GenArrayWaitHandle::getName() { return s_genArray; } c_WaitableWaitHandle* c_GenArrayWaitHandle::getChild() { assert(getState() == STATE_BLOCKED); return static_cast( m_deps->nvGetValueRef(m_iterPos)->m_data.pobj); } void c_GenArrayWaitHandle::enterContext(context_idx_t ctx_idx) { assert(AsioSession::Get()->getContext(ctx_idx)); // stop before corrupting unioned data if (isFinished()) { return; } // already in the more specific context? if (LIKELY(getContextIdx() >= ctx_idx)) { return; } assert(getState() == STATE_BLOCKED); // recursively import current child { assert(m_iterPos != ArrayData::invalid_index); TypedValue* current = m_deps->nvGetValueRef(m_iterPos); assert(current->m_type == KindOfObject); assert(dynamic_cast(current->m_data.pobj)); auto child_wh = static_cast(current->m_data.pobj); child_wh->enterContext(ctx_idx); } // import ourselves setContextIdx(ctx_idx); // try to import other children try { for (ssize_t iter_pos = m_deps->iter_advance(m_iterPos); iter_pos != ArrayData::invalid_index; iter_pos = m_deps->iter_advance(iter_pos)) { TypedValue* current = m_deps->nvGetValueRef(iter_pos); if (IS_NULL_TYPE(current->m_type)) { continue; } assert(current->m_type == KindOfObject); assert(dynamic_cast(current->m_data.pobj)); auto child = static_cast(current->m_data.pobj); if (child->isFinished()) { continue; } assert(dynamic_cast(child)); auto child_wh = static_cast(child); child_wh->enterContext(ctx_idx); } } catch (const Object& cycle_exception) { // exception will be eventually processed by onUnblocked() } } /////////////////////////////////////////////////////////////////////////////// }