/* * Copyright 2019 The WebRTC Project Authors. All rights reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "rtc_base/operations_chain.h" #include "rtc_base/checks.h" namespace rtc { OperationsChain::CallbackHandle::CallbackHandle( scoped_refptr operations_chain) : operations_chain_(std::move(operations_chain)) {} OperationsChain::CallbackHandle::~CallbackHandle() { RTC_DCHECK(has_run_); } void OperationsChain::CallbackHandle::OnOperationComplete() { RTC_DCHECK(!has_run_); #ifdef RTC_DCHECK_IS_ON has_run_ = true; #endif // RTC_DCHECK_IS_ON operations_chain_->OnOperationComplete(); // We have no reason to keep the |operations_chain_| alive through reference // counting anymore. operations_chain_ = nullptr; } // static scoped_refptr OperationsChain::Create() { return new OperationsChain(); } OperationsChain::OperationsChain() : RefCountedObject() { RTC_DCHECK_RUN_ON(&sequence_checker_); } OperationsChain::~OperationsChain() { // Operations keep the chain alive through reference counting so this should // not be possible. The fact that the chain is empty makes it safe to // destroy the OperationsChain on any sequence. RTC_DCHECK(chained_operations_.empty()); } std::function OperationsChain::CreateOperationsChainCallback() { return [handle = rtc::scoped_refptr( new CallbackHandle(this))]() { handle->OnOperationComplete(); }; } void OperationsChain::OnOperationComplete() { RTC_DCHECK_RUN_ON(&sequence_checker_); // The front element is the operation that just completed, remove it. RTC_DCHECK(!chained_operations_.empty()); chained_operations_.pop(); // If there are any other operations chained, execute the next one. if (!chained_operations_.empty()) { chained_operations_.front()->Run(); } } } // namespace rtc