// Copyright 2015 Google Inc. All rights reserved
//
// 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 STMT_H_
#define STMT_H_

#include <string>
#include <vector>

#include "loc.h"
#include "string_piece.h"
#include "symtab.h"

using namespace std;

class Evaluator;
class Value;

enum struct AssignOp : char {
  EQ,
  COLON_EQ,
  PLUS_EQ,
  QUESTION_EQ,
};

enum struct AssignDirective {
  NONE = 0,
  OVERRIDE = 1,
  EXPORT = 2,
};

enum struct CondOp {
  IFEQ,
  IFNEQ,
  IFDEF,
  IFNDEF,
};

struct Stmt {
 public:
  virtual ~Stmt();

  Loc loc() const { return loc_; }
  void set_loc(Loc loc) { loc_ = loc; }
  StringPiece orig() const { return orig_; }

  virtual void Eval(Evaluator* ev) const = 0;

  virtual string DebugString() const = 0;

 protected:
  Stmt();

 private:
  Loc loc_;
  StringPiece orig_;
};

/* Parsed "rule statement" before evaluation is kept as
 *    <lhs> <sep> <rhs>
 * where <lhs> and <rhs> as Value instances. <sep> is either command
 * separator (';') or an assignment ('=' or '=$=').
 * Until we evaluate <lhs>, we don't know whether it is a rule or
 * a rule-specific variable assignment.
 */
struct RuleStmt : public Stmt {
  Value* lhs;
  enum { SEP_NULL, SEP_SEMICOLON, SEP_EQ, SEP_FINALEQ } sep;
  Value* rhs;

  virtual ~RuleStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

struct AssignStmt : public Stmt {
  Value* lhs;
  Value* rhs;
  StringPiece orig_rhs;
  AssignOp op;
  AssignDirective directive;
  bool is_final;

  AssignStmt() : is_final(false) {}
  virtual ~AssignStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;

  Symbol GetLhsSymbol(Evaluator* ev) const;

 private:
  mutable Symbol lhs_sym_cache_;
};

struct CommandStmt : public Stmt {
  Value* expr;
  StringPiece orig;

  virtual ~CommandStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

struct IfStmt : public Stmt {
  CondOp op;
  Value* lhs;
  Value* rhs;
  vector<Stmt*> true_stmts;
  vector<Stmt*> false_stmts;

  virtual ~IfStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

struct IncludeStmt : public Stmt {
  Value* expr;
  bool should_exist;

  virtual ~IncludeStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

struct ExportStmt : public Stmt {
  Value* expr;
  bool is_export;

  virtual ~ExportStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

struct ParseErrorStmt : public Stmt {
  string msg;

  virtual ~ParseErrorStmt();

  virtual void Eval(Evaluator* ev) const;

  virtual string DebugString() const;
};

#endif  // STMT_H_
