blob: a1e52a3c5ce00e7db6f3439bbbd0702676b500b4 [file] [log] [blame]
// 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/common/Log.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) {
if (!ValidateCallbackMode(event->GetCallbackMode())) {
// TODO: crbug.com/42241407 - Update to use instance logging callback.
dawn::ErrorLog() << "Invalid callback mode: " << event->GetCallbackMode();
return {kNullFutureID, false};
}
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