// Copyright (C) 2020-2025 Free Software Foundation, Inc.

// This file is part of GCC.

// GCC is free software; you can redistribute it and/or modify it under
// the terms of the GNU General Public License as published by the Free
// Software Foundation; either version 3, or (at your option) any later
// version.

// GCC is distributed in the hope that it will be useful, but WITHOUT ANY
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
// FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
// for more details.

// You should have received a copy of the GNU General Public License
// along with GCC; see the file COPYING3.  If not see
// <http://www.gnu.org/licenses/>.

#ifndef RUST_FEATURE_GATE_H
#define RUST_FEATURE_GATE_H

#include "rust-ast-visitor.h"
#include "rust-ast-full.h"
#include "rust-feature.h"

namespace Rust {

class FeatureGate : public AST::DefaultASTVisitor
{
public:
  FeatureGate () {}

  using AST::DefaultASTVisitor::visit;

  void check (AST::Crate &crate);
  void visit (AST::Crate &crate) override;

  void visit (AST::Token &tok) override {}
  void visit (AST::DelimTokenTree &delim_tok_tree) override {}
  void visit (AST::AttrInputMetaItemContainer &input) override {}
  void visit (AST::IdentifierExpr &ident_expr) override {}
  void visit (AST::Lifetime &lifetime) override {}
  void visit (AST::LifetimeParam &lifetime_param) override;
  void visit (AST::ConstGenericParam &const_param) override;
  void visit (AST::PathInExpression &path) override {}
  void visit (AST::TypePathSegment &segment) override {}
  void visit (AST::TypePathSegmentGeneric &segment) override {}
  void visit (AST::TypePathSegmentFunction &segment) override {}
  void visit (AST::TypePath &path) override {}
  void visit (AST::QualifiedPathInExpression &path) override {}
  void visit (AST::QualifiedPathInType &path) override {}
  void visit (AST::LiteralExpr &expr) override {}
  void visit (AST::AttrInputLiteral &attr_input) override {}
  void visit (AST::AttrInputMacro &attr_input) override {}
  void visit (AST::MetaItemLitExpr &meta_item) override {}
  void visit (AST::MetaItemPathLit &meta_item) override {}
  void visit (AST::BorrowExpr &expr) override;
  void visit (AST::DereferenceExpr &expr) override {}
  void visit (AST::ErrorPropagationExpr &expr) override {}
  void visit (AST::NegationExpr &expr) override {}
  void visit (AST::ArithmeticOrLogicalExpr &expr) override {}
  void visit (AST::ComparisonExpr &expr) override {}
  void visit (AST::LazyBooleanExpr &expr) override {}
  void visit (AST::TypeCastExpr &expr) override {}
  void visit (AST::AssignmentExpr &expr) override {}
  void visit (AST::CompoundAssignmentExpr &expr) override {}
  void visit (AST::GroupedExpr &expr) override {}
  void visit (AST::ArrayElemsValues &elems) override {}
  void visit (AST::ArrayElemsCopied &elems) override {}
  void visit (AST::ArrayExpr &expr) override {}
  void visit (AST::ArrayIndexExpr &expr) override {}
  void visit (AST::TupleExpr &expr) override {}
  void visit (AST::TupleIndexExpr &expr) override {}
  void visit (AST::StructExprStruct &expr) override {}
  void visit (AST::StructExprFieldIdentifier &field) override {}
  void visit (AST::StructExprFieldIdentifierValue &field) override {}
  void visit (AST::StructExprFieldIndexValue &field) override {}
  void visit (AST::StructExprStructFields &expr) override {}
  void visit (AST::StructExprStructBase &expr) override {}
  void visit (AST::CallExpr &expr) override {}
  void visit (AST::MethodCallExpr &expr) override {}
  void visit (AST::FieldAccessExpr &expr) override {}
  void visit (AST::ClosureExprInner &expr) override {}
  void visit (AST::ClosureExprInnerTyped &expr) override {}
  void visit (AST::ContinueExpr &expr) override {}
  void visit (AST::BreakExpr &expr) override {}
  void visit (AST::RangeFromToExpr &expr) override {}
  void visit (AST::RangeFromExpr &expr) override {}
  void visit (AST::RangeToExpr &expr) override {}
  void visit (AST::RangeFullExpr &expr) override {}
  void visit (AST::RangeFromToInclExpr &expr) override {}
  void visit (AST::RangeToInclExpr &expr) override {}
  void visit (AST::ReturnExpr &expr) override {}
  void visit (AST::BoxExpr &expr) override;
  void visit (AST::UnsafeBlockExpr &expr) override {}
  void visit (AST::LoopExpr &expr) override {}
  void visit (AST::WhileLoopExpr &expr) override {}
  void visit (AST::WhileLetLoopExpr &expr) override {}
  void visit (AST::ForLoopExpr &expr) override {}
  void visit (AST::IfExpr &expr) override {}
  void visit (AST::IfExprConseqElse &expr) override {}
  void visit (AST::IfLetExprConseqElse &expr) override {}
  void visit (AST::AwaitExpr &expr) override {}
  void visit (AST::AsyncBlockExpr &expr) override {}
  void visit (AST::TypeParam &param) override;
  void visit (AST::LifetimeWhereClauseItem &item) override {}
  void visit (AST::TypeBoundWhereClauseItem &item) override {}
  void visit (AST::Module &module) override {}
  void visit (AST::ExternCrate &crate) override {}
  void visit (AST::UseTreeGlob &use_tree) override {}
  void visit (AST::UseTreeList &use_tree) override {}
  void visit (AST::UseTreeRebind &use_tree) override {}
  void visit (AST::UseDeclaration &use_decl) override {}
  void visit (AST::Function &function) override;
  void visit (AST::TypeAlias &type_alias) override {}
  void visit (AST::StructStruct &struct_item) override {}
  void visit (AST::TupleStruct &tuple_struct) override {}
  void visit (AST::EnumItem &item) override {}
  void visit (AST::EnumItemTuple &item) override {}
  void visit (AST::EnumItemStruct &item) override {}
  void visit (AST::EnumItemDiscriminant &item) override {}
  void visit (AST::Enum &enum_item) override {}
  void visit (AST::Union &union_item) override {}
  void visit (AST::ConstantItem &const_item) override {}
  void visit (AST::StaticItem &static_item) override {}
  void visit (AST::TraitItemConst &item) override {}
  void visit (AST::TraitItemType &item) override {}
  void visit (AST::TraitImpl &impl) override;
  void visit (AST::Trait &trait) override {}
  void visit (AST::ExternalTypeItem &item) override;
  void visit (AST::ExternalStaticItem &item) override {}
  void visit (AST::ExternBlock &block) override;
  void visit (AST::MacroMatchFragment &match) override {}
  void visit (AST::MacroMatchRepetition &match) override {}
  void visit (AST::MacroMatcher &matcher) override {}
  void visit (AST::MacroRulesDefinition &rules_def) override;
  void visit (AST::MacroInvocation &macro_invoc) override {}
  void visit (AST::MetaItemPath &meta_item) override {}
  void visit (AST::MetaItemSeq &meta_item) override {}
  void visit (AST::MetaWord &meta_item) override {}
  void visit (AST::MetaNameValueStr &meta_item) override {}
  void visit (AST::MetaListPaths &meta_item) override {}
  void visit (AST::MetaListNameValueStr &meta_item) override {}
  void visit (AST::LiteralPattern &pattern) override {}
  void visit (AST::IdentifierPattern &pattern) override {}
  void visit (AST::WildcardPattern &pattern) override {}
  void visit (AST::RestPattern &pattern) override {}
  void visit (AST::RangePatternBoundLiteral &bound) override {}
  void visit (AST::RangePatternBoundPath &bound) override {}
  void visit (AST::RangePatternBoundQualPath &bound) override {}
  void visit (AST::RangePattern &pattern) override;
  void visit (AST::ReferencePattern &pattern) override {}
  void visit (AST::StructPatternFieldTuplePat &field) override {}
  void visit (AST::StructPatternFieldIdentPat &field) override {}
  void visit (AST::StructPatternFieldIdent &field) override {}
  void visit (AST::StructPattern &pattern) override {}
  void visit (AST::TupleStructItemsNoRange &tuple_items) override {}
  void visit (AST::TupleStructItemsRange &tuple_items) override {}
  void visit (AST::TupleStructPattern &pattern) override {}
  void visit (AST::TuplePatternItemsMultiple &tuple_items) override {}
  void visit (AST::TuplePatternItemsRanged &tuple_items) override {}
  void visit (AST::TuplePattern &pattern) override {}
  void visit (AST::GroupedPattern &pattern) override {}
  void visit (AST::SlicePattern &pattern) override {}
  void visit (AST::AltPattern &pattern) override {}
  void visit (AST::EmptyStmt &stmt) override {}
  void visit (AST::ExprStmt &stmt) override {}
  void visit (AST::TraitBound &bound) override {}
  void visit (AST::ImplTraitType &type) override {}
  void visit (AST::TraitObjectType &type) override {}
  void visit (AST::ParenthesisedType &type) override {}
  void visit (AST::ImplTraitTypeOneBound &type) override {}
  void visit (AST::TraitObjectTypeOneBound &type) override {}
  void visit (AST::TupleType &type) override {}
  void visit (AST::NeverType &type) override {}
  void visit (AST::RawPointerType &type) override {}
  void visit (AST::ReferenceType &type) override {}
  void visit (AST::ArrayType &type) override {}
  void visit (AST::SliceType &type) override {}
  void visit (AST::InferredType &type) override {}
  void visit (AST::BareFunctionType &type) override {}
  void visit (AST::FunctionParam &param) override {}
  void visit (AST::VariadicParam &param) override {}
  void visit (AST::SelfParam &param) override {}

private:
  void gate (Feature::Name name, location_t loc, const std::string &error_msg);
  void check_rustc_attri (const std::vector<AST::Attribute> &attributes);
  void
  check_may_dangle_attribute (const std::vector<AST::Attribute> &attributes);
  std::set<Feature::Name> valid_features;
};
} // namespace Rust
#endif