diff --git a/regression-tests/pure2-bugfix-for-deducible-parameters.cpp2 b/regression-tests/pure2-bugfix-for-deducible-parameters.cpp2 new file mode 100644 index 0000000000..cd371175b6 --- /dev/null +++ b/regression-tests/pure2-bugfix-for-deducible-parameters.cpp2 @@ -0,0 +1,34 @@ +// Dependent, non-deducible parameters +// are wrapped like non-dependent parameters. +init: (out x: std::integral_constant) = { x = (); } +init: (out x: std::integral_constant, _: T) = { x = (); } +id: (x: std::integral_constant) -> forward _ = x; +id: (x: std::integral_constant, y: T) = { [[assert: x& == y&]] } + +main: () = { + zero: type == std::integral_constant; + + z: zero; + init(out z); + [[assert: id(z)& == z&]] + + // Deducible parameters. + _ = :v = 0; + _ = : (x: std::vector) = {}(:std::vector = ()); + _ = : (x: std::vector>) = {}(:std::vector> = ()); + // Uncomment once `typename` is supported for template arguments. + // _ = : (x: std::pair, y: U) = {}(:std::pair = (0, 0), z); + _ = : (x: std::array, y: U) = {}(:std::array = (), z); + init(out z, z); + id(z, z); + + // Test that these are emitted unwrapped in case they are deducible. + (copy f := : (x: std::vector>) = {}) + static_assert(!std::is_invocable_v>); + (copy f := : (x: std::vector>) = {}) + static_assert(std::is_invocable_v>>); +} + +v: type = { + operator=: (out this, x: T) = { } +} diff --git a/regression-tests/pure2-bugfix-for-dependent-types.cpp2 b/regression-tests/pure2-bugfix-for-dependent-types.cpp2 new file mode 100644 index 0000000000..c2d8512c3a --- /dev/null +++ b/regression-tests/pure2-bugfix-for-dependent-types.cpp2 @@ -0,0 +1,92 @@ +identity: type == T; + +f: (x: T::value_type) -> T::value_type = { + [[assert: x is T::value_type]] + y: T::value_type; + y = x; + z: type == T::value_type; + return (:T::value_type = x); + + // Dependent *template-id*s. + _ = :identity::value_type = (); // First identifier. + _ = :std::optional::value_type = (); // Non-first identifier. + _ = :std::array::value_type = (); + _ = :std::array::value_type = (); + + // Emitted `template`. + ptr: type == * T; // Needed, pending #502. + nptr: type == * i32; // Needed, pending #502. + _ = :std::pointer_traits::rebind = (); // Type-only context. + _ = :std::pointer_traits::rebind = (); // Non-dependent. + _ = :std::pointer_traits::rebind = (); // Dependent on the nested template. + _ = :std::pointer_traits::rebind = (); // Dependent on the outer template. + // Uncomment once `typename` is supported for template arguments. + // _ = :identity::rebind> = (); // Non type-only context. + + // Aliases. + w: type == T; + _ = :w::value_type = x; + v: type == w; + _ = :v::value_type = x; + a: type == T::type; + _ = :a::value_type = x; + + { + // Test that there's no prefixed `typename` to.... + _ = std::integral_constant(); // `T::value`. + _ = :std::type_identity_t = (); // `std::type_identity_t`. + + // Test that non-dependent names aren't emitted with `typename`. + a: type == std::integral_constant; + b: type == a; + c: type == b; + _ = :b::value_type = x; + _ = :c::value_type = x; + } +} + +t: @struct type = { + u: @struct type = { + x: T::value_type = (); + this: T::type = (); + } + x: T::value_type = 0; +} + +main: () = { + zero: type == std::integral_constant; + _ = f(0); + + // clang-format off + _ = : ::t = (); // clang-format on + + // Emitted `template` (noop, taken care of by the UFCS macro). + _ = :(f) = { _ = f.operator() (); }(: () = {}); + + // Nesting is irrelevant. + _ = : () = { _ = :T::value_type = (); }; + _ = :() = { _ = : () = { _ = :T::value_type = (); }; }; + _ = :() = { _ = :() = { _ = : () = { _ = :T::value_type = (); }; }; }; + _ = :() = { _ = :() = { _ = :() = { _ = : () = { _ = :T::value_type = (); }; }; }; }; + _ = :() = { _ = :() = { _ = : () = { _ = :() = { _ = :T::value_type = (); }; }; }; }; + _ = :() = { _ = : () = { _ = :() = { _ = :() = { _ = :T::value_type = (); }; }; }; }; + _ = : () = { _ = :() = { _ = :() = { _ = :() = { _ = :T::value_type = (); }; }; }; }; + _ = : () = { _ = :() = { _ = :() = { _ = :(x: T::value_type) = {}; }; }; }; + _ = : () = { _ = :() = { _ = :(x: T::value_type) = { _ = :() = {}; }; }; }; + _ = : () = { _ = :(x: T::value_type) = { _ = :() = { _ = :() = {}; }; }; }; + _ = : (x: T::value_type) = { _ = :() = { _ = :() = { _ = :() = {}; }; }; }; + + // Lookup. + { + alias: type == std::integral_constant; + _ = :alias::value_type = 0; // Non-dependent. + } + _ = : (_: T) = { + alias: type == std::integral_constant; + _ = :alias::value_type = 0; // Dependent. + { + alias: type == std::integral_constant; + _ = :alias::value_type = 0; // Non-dependent. + } + }(0); +} diff --git a/regression-tests/test-results/clang-18/clang-version.output b/regression-tests/test-results/clang-18/clang-version.output new file mode 100644 index 0000000000..b9df074ec2 --- /dev/null +++ b/regression-tests/test-results/clang-18/clang-version.output @@ -0,0 +1,4 @@ +clang version 18.0.0 (https://github.com/llvm/llvm-project.git 3723ede3cf5324827f8fbbe7f484c2ee4d7a7204) +Target: x86_64-pc-linux-gnu +Thread model: posix +InstalledDir: /home/johel/root/clang-main/bin diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-deducible-parameters.cpp.execution b/regression-tests/test-results/clang-18/pure2-bugfix-for-deducible-parameters.cpp.execution new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-deducible-parameters.cpp.output b/regression-tests/test-results/clang-18/pure2-bugfix-for-deducible-parameters.cpp.output new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-dependent-types.cpp.execution b/regression-tests/test-results/clang-18/pure2-bugfix-for-dependent-types.cpp.execution new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-dependent-types.cpp.output b/regression-tests/test-results/clang-18/pure2-bugfix-for-dependent-types.cpp.output new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-non-local-function-expression.cpp.execution b/regression-tests/test-results/clang-18/pure2-bugfix-for-non-local-function-expression.cpp.execution new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-bugfix-for-non-local-function-expression.cpp.output b/regression-tests/test-results/clang-18/pure2-bugfix-for-non-local-function-expression.cpp.output new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-print.cpp.output b/regression-tests/test-results/clang-18/pure2-print.cpp.output new file mode 100644 index 0000000000..e69de29bb2 diff --git a/regression-tests/test-results/clang-18/pure2-union.cpp.execution b/regression-tests/test-results/clang-18/pure2-union.cpp.execution new file mode 100644 index 0000000000..0643cc9a89 --- /dev/null +++ b/regression-tests/test-results/clang-18/pure2-union.cpp.execution @@ -0,0 +1,3 @@ +sizeof(x) is 25 +(not a name) +xyz diff --git a/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp b/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp new file mode 100644 index 0000000000..1e79482870 --- /dev/null +++ b/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp @@ -0,0 +1,88 @@ + +#define CPP2_USE_MODULES Yes + +//=== Cpp2 type declarations ==================================================== + + +#include "cpp2util.h" + + +#line 32 "pure2-bugfix-for-deducible-parameters.cpp2" +template class v; + + +//=== Cpp2 type definitions and function declarations =========================== + +// Dependent, non-deducible parameters +// are wrapped like non-dependent parameters. +#line 3 "pure2-bugfix-for-deducible-parameters.cpp2" +template auto init(cpp2::out> x) -> void; +template auto init(cpp2::out> x, [[maybe_unused]] T const& param2) -> void; +template [[nodiscard]] auto id(cpp2::in> x) -> auto&&; +template auto id(cpp2::in> x, T const& y) -> void; + +auto main() -> int; + + +#line 32 "pure2-bugfix-for-deducible-parameters.cpp2" +template class v { + public: explicit v(T const& x); +#line 33 "pure2-bugfix-for-deducible-parameters.cpp2" + public: auto operator=(T const& x) -> v& ; + + public: v(v const&) = delete; /* No 'that' constructor, suppress copy */ + public: auto operator=(v const&) -> void = delete; +#line 34 "pure2-bugfix-for-deducible-parameters.cpp2" +}; + + +//=== Cpp2 function definitions ================================================= + + +#line 3 "pure2-bugfix-for-deducible-parameters.cpp2" +template auto init(cpp2::out> x) -> void{x.construct(); } +template auto init(cpp2::out> x, [[maybe_unused]] T const& param2) -> void{x.construct(); } +template [[nodiscard]] auto id(cpp2::in> x) -> auto&& { return x; } +template auto id(cpp2::in> x, T const& y) -> void{cpp2::Default.expects(&x == &y, ""); } + +auto main() -> int{ + using zero = std::integral_constant; + + cpp2::deferred_init z; + init(cpp2::out(&z)); + cpp2::Default.expects(&id(z.value()) == &z.value(), ""); + + // Deducible parameters. + static_cast(v{0}); + static_cast([](std::vector const& x) -> void{}(std::vector{})); + static_cast([](std::vector> const& x) -> void{}(std::vector>{})); + // Uncomment once `typename` is supported for template arguments. + // _ = : (x: std::pair, y: U) = {}(:std::pair = (0, 0), z); + static_cast([](std::array const& x, U const& y) -> void{}(std::array{}, z.value())); + init(cpp2::out(&z.value()), z.value()); + id(z.value(), std::move(z.value())); +{ +auto f = [](std::vector> const& x) -> void{}; + + // Test that these are emitted unwrapped in case they are deducible. + +#line 27 "pure2-bugfix-for-deducible-parameters.cpp2" + static_assert(!(std::is_invocable_v>)); +} +{ +auto f = [](std::vector> const& x) -> void{}; + +#line 29 "pure2-bugfix-for-deducible-parameters.cpp2" + static_assert(std::is_invocable_v>>); +} +#line 30 "pure2-bugfix-for-deducible-parameters.cpp2" +} + +#line 33 "pure2-bugfix-for-deducible-parameters.cpp2" + template v::v(T const& x){} +#line 33 "pure2-bugfix-for-deducible-parameters.cpp2" + template auto v::operator=(T const& x) -> v& { + return *this; +#line 33 "pure2-bugfix-for-deducible-parameters.cpp2" + } + diff --git a/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp2.output b/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp2.output new file mode 100644 index 0000000000..7e84d47be1 --- /dev/null +++ b/regression-tests/test-results/pure2-bugfix-for-deducible-parameters.cpp2.output @@ -0,0 +1,2 @@ +pure2-bugfix-for-deducible-parameters.cpp2... ok (all Cpp2, passes safety checks) + diff --git a/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp b/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp new file mode 100644 index 0000000000..cd1fc2d102 --- /dev/null +++ b/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp @@ -0,0 +1,123 @@ + +#define CPP2_USE_MODULES Yes + +//=== Cpp2 type declarations ==================================================== + + +#include "cpp2util.h" + + +#line 48 "pure2-bugfix-for-dependent-types.cpp2" +template class t; + + +//=== Cpp2 type definitions and function declarations =========================== + +template using identity = T; + +template [[nodiscard]] auto f(cpp2::in x) -> T::value_type; + + +#line 48 "pure2-bugfix-for-dependent-types.cpp2" +template class t { +struct u_x_as_base { T::value_type x; }; + +#line 49 "pure2-bugfix-for-dependent-types.cpp2" + public: class u: public u_x_as_base, public T::type { + +#line 52 "pure2-bugfix-for-dependent-types.cpp2" + }; + public: T::value_type x {0}; +}; + +auto main() -> int; + + +//=== Cpp2 function definitions ================================================= + + +#line 3 "pure2-bugfix-for-dependent-types.cpp2" +template [[nodiscard]] auto f(cpp2::in x) -> T::value_type{ + cpp2::Default.expects(cpp2::is(x), ""); + cpp2::deferred_init y; + y.construct(x); + using z = T::value_type; + return { typename T::value_type{x} }; + + // Dependent *template-id*s. + static_cast(typename identity::value_type{});// First identifier. + static_cast(typename std::optional::value_type{});// Non-first identifier. + static_cast(typename std::array::value_type{}); + static_cast(typename std::array::value_type{}); + + // Emitted `template`. + using ptr = T*; // Needed, pending #502. + using nptr = cpp2::i32*; // Needed, pending #502. + static_cast(typename std::pointer_traits::template rebind{});// Type-only context. + static_cast(std::pointer_traits::rebind{});// Non-dependent. + static_cast(std::pointer_traits::rebind{});// Dependent on the nested template. + static_cast(typename std::pointer_traits::template rebind{});// Dependent on the outer template. + // Uncomment once `typename` is supported for template arguments. + // _ = :identity::rebind> = (); // Non type-only context. + + // Aliases. + using w = T; + static_cast(typename w::value_type{x}); + using v = w; + static_cast(typename v::value_type{x}); + using a = T::type; + static_cast(typename a::value_type{x}); + + { + // Test that there's no prefixed `typename` to.... + static_cast(std::integral_constant());// `T::value`. + static_cast(std::type_identity_t{});// `std::type_identity_t`. + + // Test that non-dependent names aren't emitted with `typename`. + using a = std::integral_constant; + using b = a; + using c = b; + static_cast(b::value_type{x}); + static_cast(c::value_type{x}); + } +} + +#line 56 "pure2-bugfix-for-dependent-types.cpp2" +auto main() -> int{ + using zero = std::integral_constant; + static_cast(f(0)); + + // clang-format off + static_cast(::t{});// clang-format on + + // Emitted `template` (noop, taken care of by the UFCS macro). + static_cast([](auto const& f) -> void{static_cast(CPP2_UFCS_TEMPLATE_0(operator(), (), f)); }([]() -> void{})); + + // Nesting is irrelevant. + static_cast([]() -> void{static_cast(typename T::value_type{}); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast(typename T::value_type{}); }); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([](cpp2::in x) -> void{}); }); }); }); + static_cast([]() -> void{static_cast([]() -> void{static_cast([](cpp2::in x) -> void{static_cast([]() -> void{}); }); }); }); + static_cast([]() -> void{static_cast([](cpp2::in x) -> void{static_cast([]() -> void{static_cast([]() -> void{}); }); }); }); + static_cast([](cpp2::in x) -> void{static_cast([]() -> void{static_cast([]() -> void{static_cast([]() -> void{}); }); }); }); + + // Lookup. + { + using alias = std::integral_constant; + static_cast(alias::value_type{0});// Non-dependent. + } + static_cast([]([[maybe_unused]] T const& param1) -> void{ + using alias = std::integral_constant; + static_cast(typename alias::value_type{0});// Dependent. + { + using alias = std::integral_constant; + static_cast(typename alias::value_type{0});// Non-dependent. + } + }(0)); +} + diff --git a/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp2.output b/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp2.output new file mode 100644 index 0000000000..6896db60e6 --- /dev/null +++ b/regression-tests/test-results/pure2-bugfix-for-dependent-types.cpp2.output @@ -0,0 +1,2 @@ +pure2-bugfix-for-dependent-types.cpp2... ok (all Cpp2, passes safety checks) + diff --git a/source/cppfront.cpp b/source/cppfront.cpp index 5f5dd1f9ee..76d1e02760 100644 --- a/source/cppfront.cpp +++ b/source/cppfront.cpp @@ -1617,7 +1617,8 @@ class cppfront unqualified_id_node const& n, bool in_synthesized_multi_return = false, bool is_local_name = true, - bool is_qualified = false + bool is_qualified = false, + bool is_dependent = false ) -> void { @@ -1666,6 +1667,10 @@ class cppfront printer.print_cpp2("CPP2_FORWARD(", {n.position().lineno, n.position().colno - 8}); } + if (is_dependent && n.open_angle != source_position{}) { + printer.print_cpp2("template ", n.position()); + } + assert(n.identifier); emit(*n.identifier, is_qualified); // inform the identifier if we know this is qualified @@ -1728,9 +1733,268 @@ class cppfront } + auto has_template_parameter_named(declaration_node const& decl, token const& id) + -> bool + { + if (auto& params = decl.template_parameters) { + return std::any_of(params->parameters.begin(), params->parameters.end(), [&](auto& param) { + assert(param->has_name()); + return *param->name() == id; + }); + } + return false; + } + + auto is_template_parameter(token const& lookup_id) + -> bool + { + // If any parent declaration + return std::any_of(current_declarations.begin() + 1, current_declarations.end(), [&](auto& decl) { + return has_template_parameter_named(*decl, lookup_id); + }); + } + + std::vector looking_up = {}; + + auto is_dependent_alias(token const& lookup_id) + -> bool + { + // Prevent recursion. + if (contains(looking_up, &lookup_id)) { + return false; + } + looking_up.push_back(&lookup_id); + auto guard = finally([&]{ looking_up.pop_back(); }); + + // If any parent declaration + return std::any_of(current_declarations.rbegin(), current_declarations.rend() - 1, [&](declaration_node const* decl) { + // that can have aliases + if ((decl->is_function() + || decl->is_type() + || decl->is_namespace()) + && decl->initializer + && decl->initializer->is_compound()) + { + auto& stmts = decl->initializer->get_if()->statements; + // among its statements + return std::any_of(stmts.rbegin(), stmts.rend(), [&](decltype(stmts.front())& stmt) { + if (auto decl = stmt->get_if(); + decl + && decl->is_alias()) { + auto& alias = get(decl->type); + // has a type alias declaration of equal name + if (alias->is_type_alias() + && decl->identifier + && *decl->identifier->identifier == lookup_id) { + auto& type_id = get(alias->initializer); + // and its value is a dependent _type-id_. + return is_dependent(*type_id); + } + } + return false; + }); + } + return false; + }); + } + + auto is_dependent(const type_id_node& n) + -> bool + { + if (auto qual = get_if(&n.id)) { + return is_dependent(**qual); + } + else if (auto unqual = get_if(&n.id)) { + return is_dependent(**unqual); + } + return false; + } + + auto is_dependent(const id_expression_node& n) + -> bool + { + if (auto qual = get_if(&n.id)) { + return is_dependent(**qual); + } + else if (auto unqual = get_if(&n.id)) { + return is_dependent(**unqual); + } + return false; + } + + struct is_dependent_expression_visitor { + cppfront* self; + + auto operator()(expression_node const& expr) const + -> bool + { + return (*this)(*expr.expr); + } + + template + auto operator()(binary_expression_node const& expr) const + -> bool + { + return (*this)(*expr.expr) || std::any_of(expr.terms.begin(), expr.terms.end(), *this); + } + + template + auto operator()(BinaryExpressionTerm const& term) const + -> bool + requires requires { term.op; term.expr; } + { + return (*this)(*term.expr); + } + + auto operator()(is_as_expression_node const& expr) const + -> bool + { + return (*this)(*expr.expr) || std::any_of(expr.ops.begin(), expr.ops.end(), *this); + } + + auto operator()(is_as_expression_node::term const& expr) const + -> bool + { + if (expr.expr) { + return (*this)(*expr.expr); + } + return self->is_dependent(*expr.type); + } + + auto operator()(prefix_expression_node const& expr) const + -> bool + { + return (*this)(*expr.expr); + } + + auto operator()(postfix_expression_node const& expr) const + -> bool + { + return (*this)(*expr.expr) || std::any_of(expr.ops.begin(), expr.ops.end(), *this); + } + + auto operator()(postfix_expression_node::term const& expr) const + -> bool + { + if (expr.id_expr) { + return (*this)(*expr.id_expr); + } + return (*this)(*expr.expr_list); + } + + auto operator()(primary_expression_node const& expr) const + -> bool + { + return std::visit([&](T const& expr) { + if constexpr (std::is_same_v + || std::is_same_v + || std::is_same_v>) { + return false; + } else { + return (*this)(*expr); + } + }, expr.expr); + } + + auto operator()(expression_list_node const& expr) const + -> bool + { + return std::any_of(expr.expressions.begin(), expr.expressions.end(), *this); + } + + auto operator()(expression_list_node::term const& term) const + -> bool + { + return (*this)(*term.expr); + } + + auto operator()(id_expression_node const& expr) const + -> bool + { + return std::visit([&](T const& expr) { + if constexpr (std::is_same_v) { + return false; + } else { + return self->is_dependent(*expr); + } + }, expr.id); + } + + auto operator()(declaration_node const& n) const + -> bool + { + auto type_id = std::get_if(&n.type); + if ( + !type_id + || (*type_id)->is_wildcard() + ) + { + return false; // (*this)(*term.initializer); + } + return self->is_dependent(**type_id); + } + + auto operator()(inspect_expression_node const& expr) const + -> bool + { + return self->is_dependent(*expr.result_type); + } + }; + + auto is_dependent( + const unqualified_id_node& n, + bool /*is_qualified*/ = false, + bool is_first = true) + -> bool + { + if (is_first && n.open_angle == source_position{}) { + assert(n.identifier); + return is_template_parameter(*n.identifier) + || is_dependent_alias(*n.identifier); + } + // If it's a _template-id_ + if (n.open_angle != source_position{}) { + assert(n.identifier); + // and any of its template arguments is either + return std::any_of(n.template_args.begin(), n.template_args.end(), [&](template_argument const& arg) { + if (auto expr = get_if(&arg.arg)) { + // a dependent _expression_ + return is_dependent_expression_visitor{this}(**expr); + } + else if (auto type_id = get_if(&arg.arg)) { + // or a dependent _type-id_. + return is_dependent(**type_id); + } + return false; + }); + } + return false; + } + + auto is_dependent(const qualified_id_node& n, bool exclude_last = false) + -> bool + { + if (is_dependent(*n.ids[0].id, true, true)) { + return true; + } + return std::any_of( + n.ids.begin() + 1, + n.ids.end() - int{exclude_last && n.ids.size() > 1}, + [&](auto& id) { + return is_dependent(*id.id, true, false); + } + ); + } + + auto should_be_prefixed_with_typename(const qualified_id_node& n) + -> bool + { + return is_dependent(n, true); + } + //----------------------------------------------------------------------- // - auto emit(qualified_id_node const& n) + auto emit(qualified_id_node const& n, bool requires_typename_keyword = false) -> void { if (!sema.check(n)) { @@ -1754,12 +2018,20 @@ class cppfront auto ident = std::string{}; printer.emit_to_string(&ident); + if (requires_typename_keyword && should_be_prefixed_with_typename(n)) { + printer.print_cpp2("typename ", n.position()); + } + + auto is_dependent = false; for (auto const& id : n.ids) { if (id.scope_op) { emit(*id.scope_op); } - emit(*id.id, false, true, true); // inform the unqualified-id that it's qualified + emit(*id.id, false, true, true, is_dependent); // inform the unqualified-id that it's qualified + if (!is_dependent && this->is_dependent(*id.id)) { + is_dependent = true; + } } printer.emit_to_string(); @@ -1771,6 +2043,7 @@ class cppfront // auto emit( type_id_node const& n, + bool is_type_only_context = false, source_position pos = {} ) -> void @@ -1784,7 +2057,7 @@ class cppfront } else { try_emit(n.id, false, false); - try_emit(n.id); + try_emit(n.id, !is_type_only_context && n.type_only_context); try_emit(n.id); } @@ -3837,6 +4110,85 @@ class cppfront } + auto is_deducible_template_parameter(token const& lookup_id) + -> bool + { + assert( current_declarations.back() ); + return has_template_parameter_named(*current_declarations.back(), lookup_id) + || ( + current_declarations.back()->parent_is_type() + && current_declarations.back()->has_name("operator=") + && has_template_parameter_named(*current_declarations.back()->get_parent(), lookup_id) + ); + } + + auto is_deducible(const type_id_node& n) + -> bool + { + if (auto qual = get_if(&n.id)) { + return is_deducible(**qual); + } + else if (auto unqual = get_if(&n.id)) { + return is_deducible(**unqual); + } + return false; + } + + auto is_deducible(const id_expression_node& n) + -> bool + { + if (auto qual = get_if(&n.id)) { + return is_deducible(**qual); + } + else if (auto unqual = get_if(&n.id)) { + return is_deducible(**unqual); + } + return false; + } + + auto is_deducible( + const unqualified_id_node& n, + bool /*is_qualified*/ = false, + bool is_first = true) + -> bool + { + // If it's not a _template-id_ + if (is_first && n.open_angle == source_position{}) { + assert(n.identifier); + // and it names a deducible template parameter. + return is_deducible_template_parameter(*n.identifier); + } + // If it's a _template-id_ + if (n.open_angle != source_position{}) { + assert(n.identifier); + // and any of its template arguments is either + return std::any_of(n.template_args.begin(), n.template_args.end(), [&](template_argument const& arg) { + if (auto expr = get_if(&arg.arg); + expr && (*expr)->is_id_expression()) { + // a deducible _id-expression_ + auto& pid = (*expr)->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr->expr; + if (auto id = get_if(&pid)) { + return is_deducible(**id); + } + } + else if (auto type_id = get_if(&arg.arg)) { + // or a deducible _type-id_. + return is_deducible(**type_id); + } + return false; + }); + } + return false; + } + + auto is_deducible(const qualified_id_node& n) + -> bool + { + assert(n.ids.size() > 0); + // Its non-terminal components are non-dependent and + // its terminal component is deducible. + return !is_dependent(n, true) && is_deducible(*n.ids.back().id); + } //----------------------------------------------------------------------- // auto emit( @@ -3944,7 +4296,7 @@ class cppfront auto const& type_id = *std::get(n.declaration->type); if (is_template_parameter) { - emit( type_id ); + emit( type_id, true ); printer.print_cpp2(" ", type_id.position()); assert (n.declaration->identifier); emit(*n.declaration->identifier); @@ -3954,52 +4306,7 @@ class cppfront //----------------------------------------------------------------------- // Else handle ordinary parameters - auto param_type = print_to_string(type_id); - - // If there are template parameters on this function or its enclosing - // type, see if this parameter's name is an unqualified-id with a - // template parameter name, or mentions a template parameter as a - // template argument - auto has_template_parameter_type_named = []( - declaration_node const& decl, - std::string_view name - ) - -> bool - { - if (decl.template_parameters) { - for (auto& tparam : decl.template_parameters->parameters) - { - assert( - tparam - && tparam->name() - ); - // For now just do a quick string match - auto tparam_name = tparam->name()->to_string(); - if ( - tparam->declaration->is_type() - && ( - name == tparam_name - || name.find("<"+tparam_name) != std::string_view::npos - || name.find(","+tparam_name) != std::string_view::npos - ) - ) - { - return true; - } - } - } - return false; - }; - - assert( current_declarations.back() ); - auto is_dependent_parameter_type = - has_template_parameter_type_named( *current_declarations.back(), param_type ) - || ( - current_declarations.back()->parent_is_type() - && current_declarations.back()->has_name("operator=") - && has_template_parameter_type_named( *current_declarations.back()->get_parent(), param_type) - ) - ; + auto is_deducible_parameter_type = is_deducible(type_id); assert( n.declaration->identifier ); auto identifier = print_to_string( *n.declaration->identifier ); @@ -4015,7 +4322,7 @@ class cppfront !is_returns && !n.declaration->is_variadic && !type_id.is_wildcard() - && !is_dependent_parameter_type + && !is_deducible_parameter_type && !type_id.is_pointer_qualified() ) { @@ -4037,13 +4344,13 @@ class cppfront } else if ( type_id.is_wildcard() - || is_dependent_parameter_type + || is_deducible_parameter_type || n.declaration->is_variadic ) { auto name = std::string{"auto"}; - if (is_dependent_parameter_type) { - name = param_type; + if (is_deducible_parameter_type) { + name = print_to_string(type_id); } switch (n.pass) { break;case passing_style::in : printer.print_cpp2( name+" const&", n.position() ); @@ -4094,7 +4401,7 @@ class cppfront if ( !is_returns && !type_id.is_wildcard() - && !is_dependent_parameter_type + && !is_deducible_parameter_type && !type_id.is_pointer_qualified() && !n.declaration->is_variadic ) @@ -4377,7 +4684,7 @@ class cppfront } } else { - emit(*r.type); + emit(*r.type, true); } } @@ -5043,7 +5350,7 @@ class cppfront "using " + print_to_string(*n.identifier) + " = " - + print_to_string( *std::get(a->initializer) ) + + print_to_string( *std::get(a->initializer), true ) + ";\n", n.position() ); @@ -5218,7 +5525,7 @@ class cppfront + "_" + decl->name()->to_string() + "_as_base { " - + print_to_string( *decl->get_object_type() ) + + print_to_string( *decl->get_object_type(), true ) + " " + decl->name()->to_string() + "; };" @@ -5384,7 +5691,7 @@ class cppfront if (decl->has_name("this")) { if (printer.get_phase() == printer.phase1_type_defs_func_decls) { printer.print_cpp2( - separator + " public " + print_to_string(*decl->get_object_type()), + separator + " public " + print_to_string(*decl->get_object_type(), true), compound_stmt->position() ); separator = ","; @@ -5982,7 +6289,7 @@ class cppfront // Emit "auto" for deduced types (of course) if (type->is_wildcard()) { assert(n.initializer); - emit( *type, n.position() ); + emit( *type, false, n.position() ); } // Otherwise, emit the type else { @@ -6000,7 +6307,7 @@ class cppfront } } printer.preempt_position_push(n.position()); - emit( *type ); + emit( *type, n.parent_is_type() ); printer.preempt_position_pop(); // one pointer is enough for now, pointer-to-function fun can be later if ( diff --git a/source/parse.h b/source/parse.h index de5680b3df..068f9a52b9 100644 --- a/source/parse.h +++ b/source/parse.h @@ -1194,6 +1194,8 @@ struct type_id_node token const* > id; + bool type_only_context = false; + auto is_wildcard() const -> bool { @@ -5871,10 +5873,11 @@ class parser //G 'const' //G '*' //G - auto type_id() + auto type_id(bool type_only_context = true) -> std::unique_ptr { auto n = std::make_unique(); + n->type_only_context = type_only_context; while ( (curr().type() == lexeme::Keyword && curr() == "const") @@ -6072,7 +6075,7 @@ class parser } // Else try parsing it as a type id - else if (auto i = type_id()) { + else if (auto i = type_id(false)) { term.arg = std::move(i); }