| // Copyright 2023 The Dawn & Tint Authors |
| // |
| // 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. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 THE COPYRIGHT HOLDER 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. |
| |
| #include <map> |
| #include <optional> |
| #include <utility> |
| #include <vector> |
| |
| #include "dawn/wire/client/EventManager.h" |
| |
| #include "dawn/wire/client/Client.h" |
| |
| namespace dawn::wire::client { |
| |
| // TrackedEvent |
| |
| TrackedEvent::TrackedEvent(WGPUCallbackMode mode) : mMode(mode) {} |
| |
| TrackedEvent::~TrackedEvent() { |
| DAWN_ASSERT(mEventState == EventState::Complete); |
| } |
| |
| WGPUCallbackMode TrackedEvent::GetCallbackMode() const { |
| return mMode; |
| } |
| |
| bool TrackedEvent::IsReady() const { |
| return mEventState == EventState::Ready; |
| } |
| |
| void TrackedEvent::SetReady() { |
| DAWN_ASSERT(mEventState != EventState::Complete); |
| mEventState = EventState::Ready; |
| } |
| |
| void TrackedEvent::Complete(FutureID futureID, EventCompletionType type) { |
| DAWN_ASSERT(mEventState != EventState::Complete); |
| CompleteImpl(futureID, type); |
| mEventState = EventState::Complete; |
| } |
| |
| // EventManager |
| |
| EventManager::~EventManager() { |
| TransitionTo(State::ClientDropped); |
| } |
| |
| std::pair<FutureID, bool> EventManager::TrackEvent(std::unique_ptr<TrackedEvent> event) { |
| FutureID futureID = mNextFutureID++; |
| |
| switch (mState) { |
| case State::InstanceDropped: { |
| if (event->GetCallbackMode() != WGPUCallbackMode_AllowSpontaneous) { |
| event->Complete(futureID, EventCompletionType::Shutdown); |
| return {futureID, false}; |
| } |
| break; |
| } |
| case State::ClientDropped: { |
| event->Complete(futureID, EventCompletionType::Shutdown); |
| return {futureID, false}; |
| } |
| case State::Nominal: |
| break; |
| } |
| |
| mTrackedEvents.Use([&](auto trackedEvents) { |
| auto [it, inserted] = trackedEvents->emplace(futureID, std::move(event)); |
| DAWN_ASSERT(inserted); |
| }); |
| |
| return {futureID, true}; |
| } |
| |
| void EventManager::TransitionTo(EventManager::State state) { |
| // If the client is disconnected, this becomes a no-op. |
| if (mState == State::ClientDropped) { |
| return; |
| } |
| |
| // Only forward state transitions are allowed. |
| DAWN_ASSERT(state > mState); |
| mState = state; |
| |
| while (true) { |
| std::map<FutureID, std::unique_ptr<TrackedEvent>> events; |
| switch (state) { |
| case State::InstanceDropped: { |
| mTrackedEvents.Use([&](auto trackedEvents) { |
| for (auto it = trackedEvents->begin(); it != trackedEvents->end();) { |
| auto& event = it->second; |
| if (event->GetCallbackMode() != WGPUCallbackMode_AllowSpontaneous) { |
| events.emplace(it->first, std::move(event)); |
| it = trackedEvents->erase(it); |
| } else { |
| ++it; |
| } |
| } |
| }); |
| break; |
| } |
| case State::ClientDropped: { |
| mTrackedEvents.Use([&](auto trackedEvents) { events = std::move(*trackedEvents); }); |
| break; |
| } |
| case State::Nominal: |
| // We always start in the nominal state so we should never be transitioning to it. |
| DAWN_UNREACHABLE(); |
| } |
| if (events.empty()) { |
| break; |
| } |
| for (auto& [futureID, event] : events) { |
| event->Complete(futureID, EventCompletionType::Shutdown); |
| } |
| } |
| } |
| |
| void EventManager::ProcessPollEvents() { |
| std::vector<std::pair<FutureID, std::unique_ptr<TrackedEvent>>> eventsToCompleteNow; |
| mTrackedEvents.Use([&](auto trackedEvents) { |
| for (auto it = trackedEvents->begin(); it != trackedEvents->end();) { |
| auto& event = it->second; |
| WGPUCallbackMode callbackMode = event->GetCallbackMode(); |
| bool shouldRemove = (callbackMode == WGPUCallbackMode_AllowProcessEvents || |
| callbackMode == WGPUCallbackMode_AllowSpontaneous) && |
| event->IsReady(); |
| if (!shouldRemove) { |
| ++it; |
| continue; |
| } |
| eventsToCompleteNow.emplace_back(it->first, std::move(event)); |
| it = trackedEvents->erase(it); |
| } |
| }); |
| |
| // Since events were initially stored and iterated from an ordered map, they must be ordered. |
| for (auto& [futureID, event] : eventsToCompleteNow) { |
| event->Complete(futureID, EventCompletionType::Ready); |
| } |
| } |
| |
| WGPUWaitStatus EventManager::WaitAny(size_t count, WGPUFutureWaitInfo* infos, uint64_t timeoutNS) { |
| // Validate for feature support. |
| if (timeoutNS > 0) { |
| // Wire doesn't support timedWaitEnable (for now). (There's no UnsupportedCount or |
| // UnsupportedMixedSources validation here, because those only apply to timed waits.) |
| // |
| // TODO(crbug.com/dawn/1987): CreateInstance needs to validate timedWaitEnable was false. |
| return WGPUWaitStatus_UnsupportedTimeout; |
| } |
| |
| if (count == 0) { |
| return WGPUWaitStatus_Success; |
| } |
| |
| // Since the user can specify the FutureIDs in any order, we need to use another ordered map |
| // here to ensure that the result is ordered for JS event ordering. |
| std::map<FutureID, std::unique_ptr<TrackedEvent>> eventsToCompleteNow; |
| bool anyCompleted = false; |
| const FutureID firstInvalidFutureID = mNextFutureID; |
| mTrackedEvents.Use([&](auto trackedEvents) { |
| for (size_t i = 0; i < count; ++i) { |
| FutureID futureID = infos[i].future.id; |
| DAWN_ASSERT(futureID < firstInvalidFutureID); |
| |
| auto it = trackedEvents->find(futureID); |
| if (it == trackedEvents->end()) { |
| infos[i].completed = true; |
| anyCompleted = true; |
| continue; |
| } |
| |
| auto& event = it->second; |
| // Early update .completed, in prep to complete the callback if ready. |
| infos[i].completed = event->IsReady(); |
| if (event->IsReady()) { |
| anyCompleted = true; |
| eventsToCompleteNow.emplace(it->first, std::move(event)); |
| trackedEvents->erase(it); |
| } |
| } |
| }); |
| |
| for (auto& [futureID, event] : eventsToCompleteNow) { |
| // .completed has already been set to true (before the callback, per API contract). |
| event->Complete(futureID, EventCompletionType::Ready); |
| } |
| |
| return anyCompleted ? WGPUWaitStatus_Success : WGPUWaitStatus_TimedOut; |
| } |
| |
| } // namespace dawn::wire::client |