// Copyright 2021 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_wire/client/Adapter.h"

#include "common/Log.h"
#include "dawn_wire/client/Client.h"

namespace dawn_wire::client {

    Adapter::~Adapter() {
        mRequestDeviceRequests.CloseAll([](RequestDeviceData* request) {
            request->callback(WGPURequestDeviceStatus_Unknown, nullptr,
                              "Adapter destroyed before callback", request->userdata);
        });
    }

    void Adapter::CancelCallbacksForDisconnect() {
        mRequestDeviceRequests.CloseAll([](RequestDeviceData* request) {
            request->callback(WGPURequestDeviceStatus_Unknown, nullptr, "GPU connection lost",
                              request->userdata);
        });
    }

    bool Adapter::GetLimits(WGPUSupportedLimits* limits) const {
        return mLimitsAndFeatures.GetLimits(limits);
    }

    bool Adapter::HasFeature(WGPUFeatureName feature) const {
        return mLimitsAndFeatures.HasFeature(feature);
    }

    uint32_t Adapter::EnumerateFeatures(WGPUFeatureName* features) const {
        return mLimitsAndFeatures.EnumerateFeatures(features);
    }

    void Adapter::SetLimits(const WGPUSupportedLimits* limits) {
        return mLimitsAndFeatures.SetLimits(limits);
    }

    void Adapter::SetFeatures(const WGPUFeatureName* features, uint32_t featuresCount) {
        return mLimitsAndFeatures.SetFeatures(features, featuresCount);
    }

    void Adapter::SetProperties(const WGPUAdapterProperties* properties) {
        mProperties = *properties;
        mProperties.nextInChain = nullptr;
    }

    void Adapter::GetProperties(WGPUAdapterProperties* properties) const {
        *properties = mProperties;
    }

    void Adapter::RequestDevice(const WGPUDeviceDescriptor* descriptor,
                                WGPURequestDeviceCallback callback,
                                void* userdata) {
        if (client->IsDisconnected()) {
            callback(WGPURequestDeviceStatus_Error, nullptr, "GPU connection lost", userdata);
            return;
        }

        auto* allocation = client->DeviceAllocator().New(client);
        uint64_t serial = mRequestDeviceRequests.Add({callback, allocation->object->id, userdata});

        AdapterRequestDeviceCmd cmd;
        cmd.adapterId = this->id;
        cmd.requestSerial = serial;
        cmd.deviceObjectHandle = ObjectHandle(allocation->object->id, allocation->generation);
        cmd.descriptor = descriptor;

        client->SerializeCommand(cmd);
    }

    bool Client::DoAdapterRequestDeviceCallback(Adapter* adapter,
                                                uint64_t requestSerial,
                                                WGPURequestDeviceStatus status,
                                                const char* message,
                                                const WGPUSupportedLimits* limits,
                                                uint32_t featuresCount,
                                                const WGPUFeatureName* features) {
        // May have been deleted or recreated so this isn't an error.
        if (adapter == nullptr) {
            return true;
        }
        return adapter->OnRequestDeviceCallback(requestSerial, status, message, limits,
                                                featuresCount, features);
    }

    bool Adapter::OnRequestDeviceCallback(uint64_t requestSerial,
                                          WGPURequestDeviceStatus status,
                                          const char* message,
                                          const WGPUSupportedLimits* limits,
                                          uint32_t featuresCount,
                                          const WGPUFeatureName* features) {
        RequestDeviceData request;
        if (!mRequestDeviceRequests.Acquire(requestSerial, &request)) {
            return false;
        }

        Device* device = client->DeviceAllocator().GetObject(request.deviceObjectId);

        // If the return status is a failure we should give a null device to the callback and
        // free the allocation.
        if (status != WGPURequestDeviceStatus_Success) {
            client->DeviceAllocator().Free(device);
            request.callback(status, nullptr, message, request.userdata);
            return true;
        }

        device->SetLimits(limits);
        device->SetFeatures(features, featuresCount);

        request.callback(status, ToAPI(device), message, request.userdata);
        return true;
    }

    WGPUDevice Adapter::CreateDevice(const WGPUDeviceDescriptor*) {
        dawn::ErrorLog() << "adapter.CreateDevice not supported with dawn_wire.";
        return nullptr;
    }

}  // namespace dawn_wire::client
