// Copyright 2020 The Tint 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 "src/tint/lang/spirv/writer/writer.h"

#include <memory>
#include <utility>

#include "src/tint/lang/core/ir/transform/binding_remapper.h"
#include "src/tint/lang/spirv/writer/ast_printer/ast_printer.h"
#include "src/tint/lang/spirv/writer/printer/printer.h"
#include "src/tint/lang/spirv/writer/raise/raise.h"
#include "src/tint/lang/wgsl/reader/program_to_ir/program_to_ir.h"

// Included by 'ast_printer.h', included again here for './tools/run gen' track the dependency.
#include "spirv/unified1/spirv.h"

namespace tint::spirv::writer {

Output::Output() = default;
Output::~Output() = default;
Output::Output(const Output&) = default;

Result<Output, std::string> Generate(const Program* program, const Options& options) {
    if (!program->IsValid()) {
        return std::string("input program is not valid");
    }

    bool zero_initialize_workgroup_memory =
        !options.disable_workgroup_init && options.use_zero_initialize_workgroup_memory_extension;

    Output output;

    if (options.use_tint_ir) {
        // Convert the AST program to an IR module.
        auto converted = wgsl::reader::ProgramToIR(program);
        if (!converted) {
            return "IR converter: " + converted.Failure();
        }

        auto ir = converted.Move();

        // Apply transforms as required by writer options.
        auto remapper = core::ir::transform::BindingRemapper(&ir, options.binding_remapper_options);
        if (!remapper) {
            return remapper.Failure();
        }

        // Raise the IR to the SPIR-V dialect.
        auto raised = raise::Raise(&ir, options);
        if (!raised) {
            return std::move(raised.Failure());
        }

        // Generate the SPIR-V code.
        auto impl = std::make_unique<Printer>(&ir, zero_initialize_workgroup_memory);
        auto spirv = impl->Generate();
        if (!spirv) {
            return std::move(spirv.Failure());
        }
        output.spirv = std::move(spirv.Get());
    } else {
        // Sanitize the program.
        auto sanitized_result = Sanitize(program, options);
        if (!sanitized_result.program.IsValid()) {
            return sanitized_result.program.Diagnostics().str();
        }

        // Generate the SPIR-V code.
        auto impl = std::make_unique<ASTPrinter>(
            &sanitized_result.program, zero_initialize_workgroup_memory,
            options.experimental_require_subgroup_uniform_control_flow);
        if (!impl->Generate()) {
            return impl->Diagnostics().str();
        }
        output.spirv = std::move(impl->Result());
    }

    return output;
}

}  // namespace tint::spirv::writer
