// 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/ast/binary_expression.h"

#include "src/tint/program_builder.h"

TINT_INSTANTIATE_TYPEINFO(tint::ast::BinaryExpression);

namespace tint::ast {

BinaryExpression::BinaryExpression(ProgramID pid,
                                   const Source& src,
                                   BinaryOp o,
                                   const Expression* l,
                                   const Expression* r)
    : Base(pid, src), op(o), lhs(l), rhs(r) {
    TINT_ASSERT(AST, lhs);
    TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(AST, lhs, program_id);
    TINT_ASSERT(AST, rhs);
    TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(AST, rhs, program_id);
    TINT_ASSERT(AST, op != BinaryOp::kNone);
}

BinaryExpression::BinaryExpression(BinaryExpression&&) = default;

BinaryExpression::~BinaryExpression() = default;

const BinaryExpression* BinaryExpression::Clone(CloneContext* ctx) const {
    // Clone arguments outside of create() call to have deterministic ordering
    auto src = ctx->Clone(source);
    auto* l = ctx->Clone(lhs);
    auto* r = ctx->Clone(rhs);
    return ctx->dst->create<BinaryExpression>(src, op, l, r);
}

}  // namespace tint::ast
