#ifndef OT_LIBERTY_LUT_HPP_
#define OT_LIBERTY_LUT_HPP_

#include <ot/headerdef.hpp>
#include <ot/static/logger.hpp>
#include <ot/utility/utility.hpp>
#include <ot/exception.hpp>

namespace ot {

// Enum: LutVar
enum class LutVar {
  TOTAL_OUTPUT_NET_CAPACITANCE = 0,
  INPUT_NET_TRANSITION,
  CONSTRAINED_PIN_TRANSITION,
  RELATED_PIN_TRANSITION,
  INPUT_TRANSITION_TIME,
};

// Variable mapping
inline const std::unordered_map<std::string_view, LutVar> lut_vars {
  {"total_output_net_capacitance", LutVar::TOTAL_OUTPUT_NET_CAPACITANCE},
  {"input_net_transition",         LutVar::INPUT_NET_TRANSITION},
  {"constrained_pin_transition",   LutVar::CONSTRAINED_PIN_TRANSITION},
  {"related_pin_transition",       LutVar::RELATED_PIN_TRANSITION},
  {"input_transition_timing",      LutVar::INPUT_TRANSITION_TIME}
};

// Function: set_use_iccad2014_models
// if set, the simplified 2x2 LUT model will be used.
void set_use_iccad2014_models(bool use);

// Function: is_time_lut_var
bool is_time_lut_var(LutVar);
bool is_capacitance_lut_var(LutVar);

// Function: to_string
std::string to_string(LutVar);

// Struct: LutTemplate
struct LutTemplate {

  std::string name;

  std::optional<LutVar> variable1;
  std::optional<LutVar> variable2;

  std::vector<float> indices1;
  std::vector<float> indices2;
};

std::ostream& operator << (std::ostream& os, const LutTemplate&);

// ------------------------------------------------------------------------------------------------

// Struct: Lut
struct Lut {
  
  std::string name;

  std::vector<float> indices1;
  std::vector<float> indices2;
  std::vector<float> table;

  const LutTemplate* lut_template {nullptr};
  
  float operator() (float, float) const;

  bool is_scalar() const;
  bool empty() const;
  
  void scale_time(float);
  void scale_capacitance(float);
};

std::ostream& operator << (std::ostream& os, const Lut&);

};  // end of namespace ot ------------------------------------------------------------------------

#endif
