// Copyright 2022 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.

#ifndef SRC_TINT_RESOLVER_CONST_EVAL_H_
#define SRC_TINT_RESOLVER_CONST_EVAL_H_

#include <stddef.h>
#include <string>

#include "src/tint/utils/result.h"
#include "src/tint/utils/vector.h"

// Forward declarations
namespace tint {
class ProgramBuilder;
class Source;
}  // namespace tint
namespace tint::ast {
class LiteralExpression;
}  // namespace tint::ast
namespace tint::sem {
class Constant;
class Expression;
class StructMember;
class Type;
}  // namespace tint::sem

namespace tint::resolver {

/// ConstEval performs shader creation-time (constant expression) expression evaluation.
/// Methods are called from the resolver, either directly or via member-function pointers indexed by
/// the IntrinsicTable. All child-expression nodes are guaranteed to have been already resolved
/// before calling a method to evaluate an expression's value.
class ConstEval {
  public:
    /// The result type of a method that may raise a diagnostic error and the caller should abort
    /// resolving. Can be one of three distinct values:
    /// * A non-null sem::Constant pointer. Returned when a expression resolves to a creation time
    ///   value.
    /// * A null sem::Constant pointer. Returned when a expression cannot resolve to a creation time
    ///   value, but is otherwise legal.
    /// * `utils::Failure`. Returned when there was a resolver error. In this situation the method
    ///   will have already reported a diagnostic error message, and the caller should abort
    ///   resolving.
    using ConstantResult = utils::Result<const sem::Constant*>;

    /// Typedef for a constant evaluation function
    using Function = ConstantResult (ConstEval::*)(const sem::Type* result_ty,
                                                   utils::VectorRef<const sem::Expression*>);

    /// Constructor
    /// @param b the program builder
    explicit ConstEval(ProgramBuilder& b);

    ////////////////////////////////////////////////////////////////////////////////////////////////
    // Constant value evaluation methods, to be called directly from Resolver
    ////////////////////////////////////////////////////////////////////////////////////////////////

    /// @param ty the target type - must be an array or constructor
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult ArrayOrStructCtor(const sem::Type* ty,
                                     utils::VectorRef<const sem::Expression*> args);

    /// @param ty the target type
    /// @param expr the input expression
    /// @return the bit-cast of the given expression to the given type, or null if the value cannot
    ///         be calculated
    ConstantResult Bitcast(const sem::Type* ty, const sem::Expression* expr);

    /// @param obj the object being indexed
    /// @param idx the index expression
    /// @return the result of the index, or null if the value cannot be calculated
    ConstantResult Index(const sem::Expression* obj, const sem::Expression* idx);

    /// @param ty the result type
    /// @param lit the literal AST node
    /// @return the constant value of the literal
    ConstantResult Literal(const sem::Type* ty, const ast::LiteralExpression* lit);

    /// @param obj the object being accessed
    /// @param member the member
    /// @return the result of the member access, or null if the value cannot be calculated
    ConstantResult MemberAccess(const sem::Expression* obj, const sem::StructMember* member);

    /// @param ty the result type
    /// @param vector the vector being swizzled
    /// @param indices the swizzle indices
    /// @return the result of the swizzle, or null if the value cannot be calculated
    ConstantResult Swizzle(const sem::Type* ty,
                           const sem::Expression* vector,
                           utils::VectorRef<uint32_t> indices);

    /// Convert the `value` to `target_type`
    /// @param ty the result type
    /// @param value the value being converted
    /// @param source the source location of the conversion
    /// @return the converted value, or null if the value cannot be calculated
    ConstantResult Convert(const sem::Type* ty, const sem::Constant* value, const Source& source);

    ////////////////////////////////////////////////////////////////////////////////////////////////
    // Constant value evaluation methods, to be indirectly called via the intrinsic table
    ////////////////////////////////////////////////////////////////////////////////////////////////

    /// Type conversion
    /// @param ty the result type
    /// @param args the input arguments
    /// @return the converted value, or null if the value cannot be calculated
    ConstantResult Conv(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Zero value type constructor
    /// @param ty the result type
    /// @param args the input arguments (no arguments provided)
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult Zero(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Identity value type constructor
    /// @param ty the result type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult Identity(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Vector splat constructor
    /// @param ty the vector type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult VecSplat(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Vector constructor using scalars
    /// @param ty the vector type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult VecCtorS(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Vector constructor using a mix of scalars and smaller vectors
    /// @param ty the vector type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult VecCtorM(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Matrix constructor using scalar values
    /// @param ty the matrix type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult MatCtorS(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Matrix constructor using column vectors
    /// @param ty the matrix type
    /// @param args the input arguments
    /// @return the constructed value, or null if the value cannot be calculated
    ConstantResult MatCtorV(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    ////////////////////////////////////////////////////////////////////////////
    // Unary Operators
    ////////////////////////////////////////////////////////////////////////////

    /// Complement operator '~'
    /// @param ty the integer type
    /// @param args the input arguments
    /// @return the result value, or null if the value cannot be calculated
    ConstantResult OpComplement(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// Minus operator '-'
    /// @param ty the expression type
    /// @param args the input arguments
    /// @return the result value, or null if the value cannot be calculated
    ConstantResult OpMinus(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    ////////////////////////////////////////////////////////////////////////////
    // Binary Operators
    ////////////////////////////////////////////////////////////////////////////

    /// Plus operator '+'
    /// @param ty the expression type
    /// @param args the input arguments
    /// @return the result value, or null if the value cannot be calculated
    ConstantResult OpPlus(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    ////////////////////////////////////////////////////////////////////////////
    // Builtins
    ////////////////////////////////////////////////////////////////////////////

    /// atan2 builtin
    /// @param ty the expression type
    /// @param args the input arguments
    /// @return the result value, or null if the value cannot be calculated
    ConstantResult atan2(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

    /// clamp builtin
    /// @param ty the expression type
    /// @param args the input arguments
    /// @return the result value, or null if the value cannot be calculated
    ConstantResult clamp(const sem::Type* ty, utils::VectorRef<const sem::Expression*> args);

  private:
    /// Adds the given error message to the diagnostics
    void AddError(const std::string& msg, const Source& source) const;

    /// Adds the given warning message to the diagnostics
    void AddWarning(const std::string& msg, const Source& source) const;

    ProgramBuilder& builder;
};

}  // namespace tint::resolver

#endif  // SRC_TINT_RESOLVER_CONST_EVAL_H_
