#pragma once #include #include namespace ppc::lang { struct namespace_name_t : public std::vector { using base = std::vector; int compare(const namespace_name_t &other) const; bool operator==(const namespace_name_t &other) const { return compare(other) == 0; } bool operator!=(const namespace_name_t &other) const { return compare(other) != 0; } bool operator<(const namespace_name_t &other) const { return compare(other) < 0; } bool operator<=(const namespace_name_t &other) const { return compare(other) <= 0; } bool operator>(const namespace_name_t &other) const { return compare(other) > 0; } bool operator>=(const namespace_name_t &other) const { return compare(other) >= 0; } operator std::string() const { return to_string(); } std::string to_string() const; namespace_name_t() { } namespace_name_t(std::initializer_list segments): base(segments.begin(), segments.end()) { } }; } template <> struct std::hash { std::size_t operator()(const ppc::lang::namespace_name_t& k) const { size_t res = 0; for (auto &el : k) { res ^= std::hash()(el); } return res; } }; #include "utils/message.hh" #include "utils/location.hh" namespace ppc::lang { template struct located_t : T { location_t location; located_t(location_t loc, const T &val): T(val), location(loc) { } located_t(const T &val): T(val), location(location_t::NONE) { } located_t() { } }; template struct slocated_t { T value; location_t location; bool operator ==(const slocated_t &other) { return value == other.value && location == other.location; } bool operator !=(const slocated_t &other) { return !(*this == other); } slocated_t(location_t loc, const T &val): value(val), location(loc) { } slocated_t(const T &val): value(val), location(location_t::NONE) { } slocated_t() { } }; struct loc_namespace_name_t : public std::vector> { using base = std::vector>; int compare(const loc_namespace_name_t &other) const; bool operator==(const loc_namespace_name_t &other) const { return compare(other) == 0; } bool operator!=(const loc_namespace_name_t &other) const { return compare(other) != 0; } bool operator<(const loc_namespace_name_t &other) const { return compare(other) < 0; } bool operator<=(const loc_namespace_name_t &other) const { return compare(other) <= 0; } bool operator>(const loc_namespace_name_t &other) const { return compare(other) > 0; } bool operator>=(const loc_namespace_name_t &other) const { return compare(other) >= 0; } namespace_name_t strip_location() const; operator std::string() const { return to_string(); } std::string to_string() const; loc_namespace_name_t() { } loc_namespace_name_t(std::initializer_list> segments): base(segments.begin(), segments.end()) { } }; bool is_identifier_valid(messages::msg_stack_t &msg_stack, ppc::location_t location, const std::string &name); inline bool is_identifier_valid(const std::string &name) { messages::msg_stack_t ms; return is_identifier_valid(ms, { }, name); } }