// Copyright (c) 2020-now by the Zeek Project. See LICENSE for details. #pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include namespace hilti::declaration { /** AST node for a struct/union field declaration. */ class Field : public Declaration { public: auto callingConvention() const { return _cc; } auto attributes() const { return child(1); } auto inlineFunction() const { return child(2); } /** Returns an operator corresponding to a call to the member function that the declaration corresponds to, if any. */ auto operator_() const { return _operator; } QualifiedType* type() const { if ( const auto& func = inlineFunction() ) return func->type(); else return child(0); } bool isResolved(node::CycleDetector* cd = nullptr) const { if ( auto func = inlineFunction() ) return func->type()->isResolved(cd); if ( auto type = child(0); type->type()->isA() ) return true; else return type->isResolved(cd); } hilti::Expression* default_() const { if ( auto a = attributes()->find(hilti::attribute::kind::Default) ) return *a->valueAsExpression(); else return {}; } auto isAnonymous() const { return attributes()->find(hilti::attribute::kind::Anonymous) != nullptr; } auto isInternal() const { return attributes()->find(hilti::attribute::kind::Internal) != nullptr; } auto isOptional() const { return attributes()->find(hilti::attribute::kind::Optional) != nullptr; } auto isStatic() const { return attributes()->find(hilti::attribute::kind::Static) != nullptr; } auto isNoEmit() const { return attributes()->find(hilti::attribute::kind::NoEmit) != nullptr; } auto linkedTypeIndex() const { return _linked_type_index; } void setAttributes(ASTContext* ctx, AttributeSet* attrs) { setChild(ctx, 1, attrs); } void setOperator(const Operator* op) { _operator = op; } void setType(ASTContext* ctx, QualifiedType* t) { setChild(ctx, 0, t); } void setLinkedTypeIndex(ast::TypeIndex idx) { assert(idx); _linked_type_index = idx; } std::string_view displayName() const final { return "struct field"; } node::Properties properties() const final; static auto create(ASTContext* ctx, ID id, QualifiedType* type, AttributeSet* attrs, Meta meta = {}) { if ( ! attrs ) attrs = AttributeSet::create(ctx); if ( attrs->has(hilti::attribute::kind::Static) ) // make it assignable type = type->recreateAsLhs(ctx); return ctx->make(ctx, {type, attrs, nullptr}, std::move(id), std::nullopt, std::move(meta)); } static auto create(ASTContext* ctx, ID id, ::hilti::function::CallingConvention cc, type::Function* ftype, AttributeSet* attrs, Meta meta = {}) { if ( ! attrs ) attrs = AttributeSet::create(ctx); return ctx->make(ctx, {QualifiedType::create(ctx, ftype, Constness::Const), attrs, nullptr}, std::move(id), cc, std::move(meta)); } static auto create(ASTContext* ctx, ID id, hilti::Function* inline_func, AttributeSet* attrs, Meta meta = {}) { if ( ! attrs ) attrs = AttributeSet::create(ctx); return ctx->make(ctx, {nullptr, attrs, inline_func}, std::move(id), std::nullopt, std::move(meta)); } protected: Field(ASTContext* ctx, Nodes children, ID id, std::optional<::hilti::function::CallingConvention> cc, Meta meta) : Declaration(ctx, NodeTags, std::move(children), std::move(id), declaration::Linkage::Struct, std::move(meta)) {} std::string _dump() const override; HILTI_NODE_1(declaration::Field, Declaration, final); private: std::optional<::hilti::function::CallingConvention> _cc; const Operator* _operator = nullptr; ast::TypeIndex _linked_type_index; }; } // namespace hilti::declaration