// Copyright 2019 The Dawn Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "dawn_native/ErrorScope.h" #include "common/Assert.h" namespace dawn_native { ErrorScope::ErrorScope() = default; ErrorScope::ErrorScope(dawn::ErrorFilter errorFilter, ErrorScope* parent) : RefCounted(), mErrorFilter(errorFilter), mParent(parent) { ASSERT(mParent.Get() != nullptr); } ErrorScope::~ErrorScope() { if (mCallback == nullptr || IsRoot()) { return; } mCallback(static_cast(mErrorType), mErrorMessage.c_str(), mUserdata); } void ErrorScope::SetCallback(dawn::ErrorCallback callback, void* userdata) { mCallback = callback; mUserdata = userdata; } ErrorScope* ErrorScope::GetParent() { return mParent.Get(); } bool ErrorScope::IsRoot() const { return mParent.Get() == nullptr; } void ErrorScope::HandleError(dawn::ErrorType type, const char* message) { HandleErrorImpl(this, type, message); } // static void ErrorScope::HandleErrorImpl(ErrorScope* scope, dawn::ErrorType type, const char* message) { ErrorScope* currentScope = scope; for (; !currentScope->IsRoot(); currentScope = currentScope->GetParent()) { ASSERT(currentScope != nullptr); bool consumed = false; switch (type) { case dawn::ErrorType::Validation: if (currentScope->mErrorFilter != dawn::ErrorFilter::Validation) { // Error filter does not match. Move on to the next scope. continue; } consumed = true; break; case dawn::ErrorType::OutOfMemory: if (currentScope->mErrorFilter != dawn::ErrorFilter::OutOfMemory) { // Error filter does not match. Move on to the next scope. continue; } consumed = true; break; // Unknown and DeviceLost are fatal. All error scopes capture them. // |consumed| is false because these should bubble to all scopes. case dawn::ErrorType::Unknown: case dawn::ErrorType::DeviceLost: consumed = false; break; case dawn::ErrorType::NoError: default: UNREACHABLE(); return; } // Record the error if the scope doesn't have one yet. if (currentScope->mErrorType == dawn::ErrorType::NoError) { currentScope->mErrorType = type; currentScope->mErrorMessage = message; } if (consumed) { return; } } // The root error scope captures all uncaptured errors. ASSERT(currentScope->IsRoot()); if (currentScope->mCallback) { currentScope->mCallback(static_cast(type), message, currentScope->mUserdata); } } void ErrorScope::Destroy() { if (!IsRoot()) { mErrorType = dawn::ErrorType::Unknown; mErrorMessage = "Error scope destroyed"; } } } // namespace dawn_native