/* ----------------------------------------------------------------------------- Copyright 2019-2021 Kevin P. Barry 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. ----------------------------------------------------------------------------- */ // Author: Kevin P. Barry [ta0kira@gmail.com] #ifndef CATEGORY_SOURCE_HPP_ #define CATEGORY_SOURCE_HPP_ #include // For occasional debugging output in generated code. #include #include #include #include "types.hpp" #include "function.hpp" #define BUILTIN_FAIL(e) { \ FAIL() << TypeValue::Call((e), Function_Formatted_formatted, \ ParamTuple(), ArgTuple()).Only()->AsString(); \ __builtin_unreachable(); \ } #define RAW_FAIL(e) { \ FAIL() << e; \ __builtin_unreachable(); \ } extern const S& Var_empty; S Box_Bool(bool value); S Box_String(const PrimString& value); S Box_Char(PrimChar value); S Box_Int(PrimInt value); S Box_Float(PrimFloat value); S Merge_Intersect(L> params); S Merge_Union(L> params); const S& GetMerged_Any(); const S& GetMerged_All(); extern const S& Var_empty; class TypeCategory { public: inline ReturnTuple Call(const CategoryFunction& label, const ParamTuple& params, const ValueTuple& args) { return FAIL_WHEN_NULL(Dispatch(label, params, FAIL_WHEN_NULL(args))); } virtual std::string CategoryName() const = 0; ALWAYS_PERMANENT(TypeCategory) virtual ~TypeCategory() = default; protected: TypeCategory() = default; virtual ReturnTuple Dispatch(const CategoryFunction& label, const ParamTuple& params, const ValueTuple& args); }; class TypeInstance { public: inline static ReturnTuple Call(const S& target, const TypeFunction& label, ParamTuple params, const ValueTuple& args) { if (target == nullptr) { FAIL() << "Function called on null value"; } return FAIL_WHEN_NULL(target->Dispatch(target, label, params, FAIL_WHEN_NULL(args))); } virtual std::string CategoryName() const = 0; virtual void BuildTypeName(std::ostream& output) const = 0; static std::string TypeName(const S& type) { TRACE_FUNCTION("typename") std::ostringstream output; type->BuildTypeName(output); return output.str(); } static S Reduce(const S& from, const S& to, S target) { TRACE_FUNCTION("reduce") return CanConvert(from, to)? target : Var_empty; } virtual bool TypeArgsForParent( const TypeCategory& category, std::vector>& args) const { return false; } ALWAYS_PERMANENT(TypeInstance) virtual ~TypeInstance() = default; protected: TypeInstance() = default; virtual ReturnTuple Dispatch(const S& self, const TypeFunction& label, const ParamTuple& params, const ValueTuple& args); virtual bool CanConvertFrom(const S& from) const { return false; } static bool CanConvert(const S& from, const S& to); template static void TypeNameFrom(std::ostream& output, const TypeCategory& category, const Ts&... params) { std::vector params2{params.get()...}; output << category.CategoryName(); if (!params2.empty()) { output << "<"; bool first = true; for (const auto param : params2) { if (!first) output << ","; first = false; param->BuildTypeName(output); } output << ">"; } } enum class MergeType { SINGLE, UNION, INTERSECT, }; private: virtual MergeType InstanceMergeType() const { return MergeType::SINGLE; } virtual std::vector> MergedTypes() const { FAIL() << "Category " << CategoryName() << " is not a merged type"; __builtin_unreachable(); } }; class TypeValue { public: inline static ReturnTuple Call(const S& target, const ValueFunction& label, const ParamTuple& params, const ValueTuple& args) { if (target == nullptr) { FAIL() << "Function called on null value"; } return FAIL_WHEN_NULL(target->Dispatch(target, label, params, FAIL_WHEN_NULL(args))); } static bool Present(S target); static S Require(S target); static S Strong(W target); virtual bool AsBool() const; virtual const PrimString& AsString() const; virtual PrimChar AsChar() const; virtual PrimCharBuffer& AsCharBuffer(); virtual PrimInt AsInt() const; virtual PrimFloat AsFloat() const; ALWAYS_PERMANENT(TypeValue) virtual ~TypeValue() = default; protected: TypeValue() = default; // NOTE: For some reason, making this private causes a segfault. virtual std::string CategoryName() const = 0; virtual bool Present() const; virtual ReturnTuple Dispatch(const S& self, const ValueFunction& label, const ParamTuple& params, const ValueTuple& args); }; class AnonymousOrder : public TypeValue { protected: // Passing in the function labels allows linking without depending on Order // when this class isn't used anywhere. AnonymousOrder(const S cont, const ValueFunction& func_next, const ValueFunction& func_get); std::string CategoryName() const final; ReturnTuple Dispatch(const S& self, const ValueFunction& label, const ParamTuple& params, const ValueTuple& args) final; virtual ~AnonymousOrder() = default; private: virtual S Call_next(const S& self) = 0; virtual S Call_get(const S& self) = 0; const S container; const ValueFunction& function_next; const ValueFunction& function_get; }; #endif // CATEGORY_SOURCE_HPP_