// Copyright 2025 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 "dawn/native/webgpu/PipelineLayoutWGPU.h"

#include <array>
#include <vector>

#include "dawn/common/Constants.h"
#include "dawn/common/StringViewUtils.h"
#include "dawn/native/webgpu/BindGroupLayoutWGPU.h"
#include "dawn/native/webgpu/CaptureContext.h"
#include "dawn/native/webgpu/DeviceWGPU.h"

namespace dawn::native::webgpu {

// static
Ref<PipelineLayout> PipelineLayout::Create(
    Device* device,
    const UnpackedPtr<PipelineLayoutDescriptor>& descriptor) {
    return AcquireRef(new PipelineLayout(device, descriptor));
}

PipelineLayout::PipelineLayout(Device* device,
                               const UnpackedPtr<PipelineLayoutDescriptor>& descriptor)
    : PipelineLayoutBase(device, descriptor),
      RecordableObject(schema::ObjectType::PipelineLayout),
      ObjectWGPU(device->wgpu.pipelineLayoutRelease) {
    WGPUPipelineLayoutDescriptor desc;
    desc.nextInChain = nullptr;
    desc.label = ToOutputStringView(GetLabel());

    std::array<WGPUBindGroupLayout, kMaxBindGroups> bindGroupLayouts;
    bindGroupLayouts.fill(nullptr);
    size_t bindGroupLayoutCount = 0;
    for (BindGroupIndex group : GetBindGroupLayoutsMask()) {
        // Empty or null bind group layout aren't iterated.
        auto idx = static_cast<size_t>(group);
        bindGroupLayoutCount = idx + 1;
        bindGroupLayouts[idx] = ToBackend(GetBindGroupLayout(group))->GetInnerHandle();
    }

    desc.bindGroupLayouts = bindGroupLayouts.data();
    desc.bindGroupLayoutCount = bindGroupLayoutCount;
    desc.immediateSize = GetImmediateDataRangeByteSize();

    mInnerHandle = device->wgpu.deviceCreatePipelineLayout(device->GetInnerHandle(), &desc);
    DAWN_ASSERT(mInnerHandle);
}

void PipelineLayout::SetLabelImpl() {
    ToBackend(GetDevice())->CaptureSetLabel(this, GetLabel());
}

MaybeError PipelineLayout::AddReferenced(CaptureContext& captureContext) {
    for (BindGroupIndex groupIndex : GetBindGroupLayoutsMask()) {
        auto frontendLayout = GetFrontendBindGroupLayout(groupIndex);
        DAWN_TRY(
            captureContext.AddResource(ToBackend(frontendLayout->GetInternalBindGroupLayout())));
    }
    return {};
}

MaybeError PipelineLayout::CaptureCreationParameters(CaptureContext& captureContext) {
    std::vector<schema::ObjectId> bindGroupLayoutIds;

    for (BindGroupIndex groupIndex : GetBindGroupLayoutsMask()) {
        auto frontendLayout = GetFrontendBindGroupLayout(groupIndex);
        bindGroupLayoutIds.push_back(
            captureContext.GetId(frontendLayout->GetInternalBindGroupLayout()));
    }

    schema::PipelineLayout data{{
        .bindGroupLayoutIds = bindGroupLayoutIds,
        .immediateSize = GetImmediateDataRangeByteSize(),
    }};
    Serialize(captureContext, data);
    return {};
}

}  // namespace dawn::native::webgpu
