On Wed, Jul 1, 2026 at 1:33 AM Nathan Myers <[email protected]> wrote:

> As shipped, the heterogeneous-key members equal_range of rbtree
> containers map, multimap, set, and multiset walk the entire range
> calling the predicate, in violation of the requirement for
> logarithmic complexity. This patch revises them to match the
> behavior of the non-heterogeneous multimap and multiset members,
> and provides tests to verify it.
>
> libstdc++-v3/Changelog:
>         PR libstdc++/118851
>         * include/bits/stl_tree.h (_M_equal_range_tr): Rewrite to match
>         non-heterogeneous equal_range implementation.
>         * testsuite/23_containers/map/operations/hetero/equal_range.cc:
>         New test.
>         *
> testsuite/23_containers/multimap/operations/hetero/equal_range.cc:
>         Same.
>         *
> testsuite/23_containers/multiset/operations/hetero/equal_range.cc:
>         Same.
>         * testsuite/23_containers/set/operations/hetero/equal_range.cc:
> Same.
> ---

LGTM only small nitpicks comments.
You didn't mention running tests.



>  libstdc++-v3/include/bits/stl_tree.h          |  23 +++-
>  .../map/operations/hetero/equal_range.cc      | 114 ++++++++++++++++++
>  .../multimap/operations/hetero/equal_range.cc | 114 ++++++++++++++++++
>  .../multiset/operations/hetero/equal_range.cc | 114 ++++++++++++++++++
>  .../set/operations/hetero/equal_range.cc      | 114 ++++++++++++++++++
>  5 files changed, 473 insertions(+), 6 deletions(-)
>  create mode 100644
> libstdc++-v3/testsuite/23_containers/map/operations/hetero/equal_range.cc
>  create mode 100644
> libstdc++-v3/testsuite/23_containers/multimap/operations/hetero/equal_range.cc
>  create mode 100644
> libstdc++-v3/testsuite/23_containers/multiset/operations/hetero/equal_range.cc
>  create mode 100644
> libstdc++-v3/testsuite/23_containers/set/operations/hetero/equal_range.cc
>
> diff --git a/libstdc++-v3/include/bits/stl_tree.h
> b/libstdc++-v3/include/bits/stl_tree.h
> index a7781b72114..14dc4ca1416 100644
> --- a/libstdc++-v3/include/bits/stl_tree.h
> +++ b/libstdc++-v3/include/bits/stl_tree.h
> @@ -2026,12 +2026,23 @@ namespace __rb_tree
>         pair<const_iterator, const_iterator>
>         _M_equal_range_tr(const _Kt& __k) const
>         {
> -         const_iterator __low(_M_lower_bound_tr(__k));
> -         auto __high = __low;
> -         auto& __cmp = _M_impl._M_key_compare;
> -         while (__high != end() && !__cmp(__k, _S_key(__high._M_node)))
> -           ++__high;
> -         return { __low, __high };
> +         auto __x = _M_begin(), __y = _M_end();
> +         while (__x)
> +           {
> +             if (_M_key_compare(_S_key(__x), __k))
> +               __x = _S_right(__x);
> +             else if (_M_key_compare(__k, _S_key(__x)))
> +               __y = __x, __x = _S_left(__x);
> +             else
> +               {
> +                 auto __xu(__x), __yu(__y);
>
Why not integrate the two line below and have:
                     auto __xu =   _S_right(__x), __yu = __y;

> +                 __y = __x, __x = _S_left(__x);
> +                 __xu = _S_right(__xu);
> +                 return { const_iterator(_M_lower_bound_tr(__x, __y,
> __k)),
> +                        const_iterator(_M_upper_bound_tr(__xu, __yu,
> __k)) };
> +               }
> +           }
> +         return { const_iterator(__y), const_iterator(__y) };
>         }
>  #endif // __glibcxx_generic_associative_lookup
>
> diff --git
> a/libstdc++-v3/testsuite/23_containers/map/operations/hetero/equal_range.cc
> b/libstdc++-v3/testsuite/23_containers/map/operations/hetero/equal_range.cc
> new file mode 100644
> index 00000000000..d11a052b3f7
> --- /dev/null
> +++
> b/libstdc++-v3/testsuite/23_containers/map/operations/hetero/equal_range.cc
> @@ -0,0 +1,114 @@
> +// { dg-do run { target c++14 } }
> +
> +#include <map>
> +#include <string>
> +#include <utility>
> +#include <functional>
> +#include <testsuite_hooks.h>
> +
> +struct Y;
> +struct Z;
> +
> +struct X {
> +  std::string s;
> +  X(std::string str) : s(str) {}
> +  X(Y&&);
> +  X(const Y&);
> +  X(Z&&);
> +  X(const Z&);
> +  friend bool operator<(X const& a, X const& b) { return a.s < b.s; }
> +  friend bool operator==(X const& a, X const& b) { return a.s == b.s; }
> +};
> +
> +struct Y {
> +  std::string s;
> +  Y() = default;
> +  Y(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +  Y(const Y& y) = default;
> +  Y& operator=(Y&& y) { s = std::move(y.s); y.s.clear(); return *this; }
> +  Y& operator=(const Y& y) = default;
> +  explicit Y(std::string s) : s(s) {}
> +  Y(int n) : s(std::string(n, 'a')) {}
> +  Y(const Y& a, const Y& b) : s(a.s + "1" + b.s) { }
> +  Y(const Y& a, Y&& b)      : s(a.s + "2" + b.s) { b.s.clear(); }
> +  Y(Y&& a, const Y& b)      : s(a.s + "3" + b.s) { a.s.clear(); }
> +  Y(Y&& a, Y&& b) : s(a.s + "4" + b.s) { a.s.clear(), b.s.clear(); }
> +  friend bool operator<(Y const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(Y const& a, X const& b) { return a.s < b.s; }
> +};
> +
> +X::X(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +X::X(const Y& y) : s(y.s) {}
> +
> +using cmp = std::less<void>;
> +
> +struct Z {
> +  std::string s;
> +  mutable int compares = 0;
> +
> +  Z() = default;
> +  Z(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +  Z(const Z& z) = default;
> +  Z& operator=(Z&& z) { s = std::move(z.s); z.s.clear(); return *this; }
> +  Z& operator=(const Z& z) = default;
> +  Z(std::string s) : s(s) {}
> +  Z(int n) : s(std::string(n, 'a')) {}
> +  friend bool operator<(Z const& a, Z const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Z const& b)
> +    { return ++b.compares, a.s.substr(0, b.s.size()) < b.s; }
> +  friend bool operator<(Z const& a, X const& b)
> +    { return ++a.compares, a.s < b.s.substr(0, a.s.size()); }
> +};
> +
> +X::X(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +X::X(const Z& z) : s(z.s) {}
> +
> +// A heterogeneous key type like Z here that compares equal
> +// if it matches just the first part of the key is allowed.
> +
> +template <typename T>
> +T populate(T a)
> +{
> +  const std::string vs[] = { "dec", "ded", "dee", "def", "deg", "deh",
> "dei" };
> +  for (auto const& v : vs)

+    a[Y{v}] = Y{v};
>
Why not to use emplace here?

> +  return a;
> +}
> +
> +void test_equal_range()
> +{
> +  std::map<X, Y, cmp> amap{cmp{}};
> +  amap.insert({{Y{"abc"}, 1}, {Y{"def"}, 2}, {Y{"ghi"}, 3}});
>
Do we need to use Y here? As far as can see, this does
std::string -> Y -> X for the key. Couldn't we pass the string-literal
directly. And if we change the value type to int, we could remove
Completely fill out the test file.



> +
> +  {
> +    auto a = populate(amap);
> +    VERIFY(a.size() == 9);
> +    Z z{"de"};
> +    auto it = a.equal_range(z);
> +    VERIFY(it.first != a.end());
> +    VERIFY(it.second != a.end());
> +    VERIFY(it.first->first == std::string("dec"));
> +    VERIFY(it.second->first == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +  {
> +    auto a = populate(amap);
> +    VERIFY(a.size() == 9);
> +    Z z{"de"};
> +    auto const& ca = a;
> +    auto it = ca.equal_range(z);
> +    VERIFY(it.first  != ca.end());
> +    VERIFY(it.second != ca.end());
> +    VERIFY(it.first->first  == std::string("dec"));
> +    VERIFY(it.second->first == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +}
> +
> +int main()
> +{
> +  test_equal_range();
> +}
> diff --git
> a/libstdc++-v3/testsuite/23_containers/multimap/operations/hetero/equal_range.cc
> b/libstdc++-v3/testsuite/23_containers/multimap/operations/hetero/equal_range.cc
> new file mode 100644
> index 00000000000..e8558f480e9
> --- /dev/null
> +++
> b/libstdc++-v3/testsuite/23_containers/multimap/operations/hetero/equal_range.cc
> @@ -0,0 +1,114 @@
> +// { dg-do run { target c++14 } }
> +
> +#include <map>
> +#include <string>
> +#include <utility>
> +#include <functional>
> +#include <testsuite_hooks.h>
> +
> +struct Y;
> +struct Z;
> +
> +struct X {
> +  std::string s;
> +  X(std::string str) : s(str) {}
> +  X(Y&&);
> +  X(const Y&);
> +  X(Z&&);
> +  X(const Z&);
> +  friend bool operator<(X const& a, X const& b) { return a.s < b.s; }
> +  friend bool operator==(X const& a, X const& b) { return a.s == b.s; }
> +};
> +
> +struct Y {
> +  std::string s;
> +  Y() = default;
> +  Y(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +  Y(const Y& y) = default;
> +  Y& operator=(Y&& y) { s = std::move(y.s); y.s.clear(); return *this; }
> +  Y& operator=(const Y& y) = default;
> +  Y(std::string sv) : s(sv) {}
> +  Y(int n) : s(std::string(n, 'a')) {}
> +  Y(const Y& a, const Y& b) : s(a.s + "1" + b.s) { }
> +  Y(const Y& a, Y&& b)      : s(a.s + "2" + b.s) { b.s.clear(); }
> +  Y(Y&& a, const Y& b)      : s(a.s + "3" + b.s) { a.s.clear(); }
> +  Y(Y&& a, Y&& b)           : s(a.s + "4" + b.s) { a.s.clear(),
> b.s.clear(); }
> +  friend bool operator<(Y const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(Y const& a, X const& b) { return a.s < b.s; }
> +};
> +
> +X::X(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +X::X(const Y& y) : s(y.s) {}
> +
> +using cmp = std::less<void>;
> +
> +struct Z {
> +  std::string s;
> +  mutable int compares = 0;
> +
> +  Z() = default;
> +  Z(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +  Z(const Z& z) = default;
> +  Z& operator=(Z&& z) { s = std::move(z.s); z.s.clear(); return *this; }
> +  Z& operator=(const Z& z) = default;
> +  Z(std::string sv) : s(sv) {}
> +  Z(int n) : s(std::string(n, 'a')) {}
> +  friend bool operator<(Z const& a, Z const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Z const& b)
> +    { return ++b.compares, a.s.substr(0, b.s.size()) < b.s; }
> +  friend bool operator<(Z const& a, X const& b)
> +    { return ++a.compares, a.s < b.s.substr(0, a.s.size()); }
> +};
> +
> +X::X(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +X::X(const Z& z) : s(z.s) {}
> +
> +// A heterogeneous key type like Z here that compares equal
> +// if it matches just the first part of the key is allowed.
> +
> +template <typename T>
> +T populate(T a)
> +{
> +  const std::string vs[] = { "dec", "ded", "dee", "def", "deg", "deh",
> "dei" };
> +  for (auto const& v : vs)
> +    a.insert(std::pair<const Y, Y>{Y{v}, Y{v}});
> +  return a;
> +}
> +
> +void test_equal_range()
> +{
> +  std::multimap<X, Y, cmp> amap{cmp{}};
> +  amap.insert({{Y{"abc"}, 1}, {Y{"def"}, 2}, {Y{"ghi"}, 3}});
>
// Same comment about Y here here.

> +
> +  {
> +    auto a = populate(amap);
> +    VERIFY(a.size() == 10);
> +    Z z{"de"};
> +    auto it = a.equal_range(z);
> +    VERIFY(it.first != a.end());
> +    VERIFY(it.second != a.end());
> +    VERIFY(it.first->first == std::string("dec"));
> +    VERIFY(it.second->first == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +  {
> +    auto a = populate(amap);
> +    VERIFY(a.size() == 10);
> +    Z z{"de"};
> +    auto const& ca = a;
> +    auto it = ca.equal_range(z);
> +    VERIFY(it.first  != ca.end());
> +    VERIFY(it.second != ca.end());
> +    VERIFY(it.first->first  == std::string("dec"));
> +    VERIFY(it.second->first == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +}
> +
> +int main()
> +{
> +  test_equal_range();
> +}
> diff --git
> a/libstdc++-v3/testsuite/23_containers/multiset/operations/hetero/equal_range.cc
> b/libstdc++-v3/testsuite/23_containers/multiset/operations/hetero/equal_range.cc
> new file mode 100644
> index 00000000000..052c218c097
> --- /dev/null
> +++
> b/libstdc++-v3/testsuite/23_containers/multiset/operations/hetero/equal_range.cc
> @@ -0,0 +1,114 @@
> +// { dg-do run { target c++14 } }
> +
> +#include <set>
> +#include <string>
> +#include <utility>
> +#include <functional>
> +#include <testsuite_hooks.h>
> +
> +struct Y;
> +struct Z;
> +
> +struct X {
> +  std::string s;
> +  X(std::string str) : s(str) {}
> +  X(Y&&);
> +  X(const Y&);
> +  X(Z&&);
> +  X(const Z&);
> +  friend bool operator<(X const& a, X const& b) { return a.s < b.s; }
> +  friend bool operator==(X const& a, X const& b) { return a.s == b.s; }
> +};
> +
> +struct Y {
> +  std::string s;
> +  Y() = default;
> +  Y(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +  Y(const Y& y) = default;
> +  Y& operator=(Y&& y) { s = std::move(y.s); y.s.clear(); return *this; }
> +  Y& operator=(const Y& y) = default;
> +  Y(std::string sv) : s(sv) {}
> +  Y(int n) : s(std::string(n, 'a')) {}
> +  Y(const Y& a, const Y& b) : s(a.s + "1" + b.s) { }
> +  Y(const Y& a, Y&& b)      : s(a.s + "2" + b.s) { b.s.clear(); }
> +  Y(Y&& a, const Y& b)      : s(a.s + "3" + b.s) { a.s.clear(); }
> +  Y(Y&& a, Y&& b)           : s(a.s + "4" + b.s) { a.s.clear(),
> b.s.clear(); }
> +  friend bool operator<(Y const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(Y const& a, X const& b) { return a.s < b.s; }
> +};
> +
> +X::X(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +X::X(const Y& y) : s(y.s) {}
> +
> +using cmp = std::less<void>;
> +
> +struct Z {
> +  std::string s;
> +  mutable int compares = 0;
> +
> +  Z() = default;
> +  Z(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +  Z(const Z& z) = default;
> +  Z& operator=(Z&& z) { s = std::move(z.s); z.s.clear(); return *this; }
> +  Z& operator=(const Z& z) = default;
> +  Z(std::string sv) : s(sv) {}
> +  Z(int n) : s(std::string(n, 'a')) {}
> +  friend bool operator<(Z const& a, Z const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Z const& b)
> +    { return ++b.compares, a.s.substr(0, b.s.size()) < b.s; }
> +  friend bool operator<(Z const& a, X const& b)
> +    { return ++a.compares, a.s < b.s.substr(0, a.s.size()); }
> +};
> +
> +X::X(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +X::X(const Z& z) : s(z.s) {}
> +
> +// A heterogeneous key type like Z here that compares equal
> +// if it matches just the first part of the key is allowed.
> +
> +template <typename T>
> +T populate(T a)
> +{
> +  const std::string vs[] = { "dec", "ded", "dee", "def", "deg", "deh",
> "dei" };
> +  for (auto const& v : vs)
> +    a.insert(Y{v});
> +  return a;
> +}
> +
> +void test_equal_range()
> +{
> +  std::multiset<X, cmp> aset{cmp{}};
> +  aset.insert({Y{"abc"}, Y{"def"}, Y{"ghi"}});
> +
> +  {
> +    auto a = populate(aset);
> +    VERIFY(a.size() == 10);
> +    Z z{"de"};
> +    auto it = a.equal_range(z);
> +    VERIFY(it.first != a.end());
> +    VERIFY(it.second != a.end());
> +    VERIFY(*it.first == std::string("dec"));
> +    VERIFY(*it.second == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +  {
> +    auto a = populate(aset);
> +    VERIFY(a.size() == 10);
> +    Z z{"de"};
> +    auto const& ca = a;
> +    auto it = ca.equal_range(z);
> +    VERIFY(it.first  != ca.end());
> +    VERIFY(it.second != ca.end());
> +    VERIFY(*it.first  == std::string("dec"));
> +    VERIFY(*it.second == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +}
> +
> +int main()
> +{
> +  test_equal_range();
> +}
> diff --git
> a/libstdc++-v3/testsuite/23_containers/set/operations/hetero/equal_range.cc
> b/libstdc++-v3/testsuite/23_containers/set/operations/hetero/equal_range.cc
> new file mode 100644
> index 00000000000..1ba59140fdc
> --- /dev/null
> +++
> b/libstdc++-v3/testsuite/23_containers/set/operations/hetero/equal_range.cc
> @@ -0,0 +1,114 @@
> +// { dg-do run { target c++14 } }
> +
> +#include <set>
> +#include <string>
> +#include <utility>
> +#include <functional>
> +#include <testsuite_hooks.h>
> +
> +struct Y;
> +struct Z;
> +
> +struct X {
> +  std::string s;
> +  X(std::string str) : s(str) {}
> +  X(Y&&);
> +  X(const Y&);
> +  X(Z&&);
> +  X(const Z&);
> +  friend bool operator<(X const& a, X const& b) { return a.s < b.s; }
> +  friend bool operator==(X const& a, X const& b) { return a.s == b.s; }
> +};
> +
> +struct Y {
> +  std::string s;
> +  Y() = default;
> +  Y(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +  Y(const Y& y) = default;
> +  Y& operator=(Y&& y) { s = std::move(y.s); y.s.clear(); return *this; }
> +  Y& operator=(const Y& y) = default;
> +  Y(std::string sv) : s(sv) {}
> +  Y(int n) : s(std::string(n, 'a')) {}
> +  Y(const Y& a, const Y& b) : s(a.s + "1" + b.s) { }
> +  Y(const Y& a, Y&& b)      : s(a.s + "2" + b.s) { b.s.clear(); }
> +  Y(Y&& a, const Y& b)      : s(a.s + "3" + b.s) { a.s.clear(); }
> +  Y(Y&& a, Y&& b)           : s(a.s + "4" + b.s) { a.s.clear(),
> b.s.clear(); }
> +  friend bool operator<(Y const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Y const& b) { return a.s < b.s; }
> +  friend bool operator<(Y const& a, X const& b) { return a.s < b.s; }
> +};
> +
> +X::X(Y&& y) : s(std::move(y.s)) { y.s.clear(); }
> +X::X(const Y& y) : s(y.s) {}
> +
> +using cmp = std::less<void>;
> +
> +struct Z {
> +  std::string s;
> +  mutable int compares = 0;
> +
> +  Z() = default;
> +  Z(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +  Z(const Z& z) = default;
> +  Z& operator=(Z&& z) { s = std::move(z.s); z.s.clear(); return *this; }
> +  Z& operator=(const Z& z) = default;
> +  Z(std::string sv) : s(sv) {}
> +  Z(int n) : s(std::string(n, 'a')) {}
> +  friend bool operator<(Z const& a, Z const& b) { return a.s < b.s; }
> +  friend bool operator<(X const& a, Z const& b)
> +    { return ++b.compares, a.s.substr(0, b.s.size()) < b.s; }
> +  friend bool operator<(Z const& a, X const& b)
> +    { return ++a.compares, a.s < b.s.substr(0, a.s.size()); }
> +};
> +
> +X::X(Z&& z) : s(std::move(z.s)) { z.s.clear(); }
> +X::X(const Z& z) : s(z.s) {}
> +
> +// A heterogeneous key type like Z here that compares equal
> +// if it matches just the first part of the key is allowed.
> +
> +template <typename T>
> +T populate(T a)
> +{
> +  const std::string vs[] = { "dec", "ded", "dee", "def", "deg", "deh",
> "dei" };
> +  for (auto const& v : vs)
> +    a.insert(Y{v});
> +  return a;
> +}
> +
> +void test_equal_range()
> +{
> +  std::set<X, cmp> aset{cmp{}};
> +  aset.insert({Y{"abc"}, Y{"def"}, Y{"ghi"}});
>
Same comments regarding Y to rest of the teest.

> +
> +  {
> +    auto a = populate(aset);
> +    VERIFY(a.size() == 9);
> +    Z z{"de"};
> +    auto it = a.equal_range(z);
> +    VERIFY(it.first != a.end());
> +    VERIFY(it.second != a.end());
> +    VERIFY(*it.first == std::string("dec"));
> +    VERIFY(*it.second == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +  {
> +    auto a = populate(aset);
> +    VERIFY(a.size() == 9);
> +    Z z{"de"};
> +    auto const& ca = a;
> +    auto it = ca.equal_range(z);
> +    VERIFY(it.first  != ca.end());
> +    VERIFY(it.second != ca.end());
> +    VERIFY(*it.first  == std::string("dec"));
> +    VERIFY(*it.second == std::string("ghi"));
> +
> +    VERIFY(z.compares == 7);
> +  }
> +}
> +
> +int main()
> +{
> +  test_equal_range();
> +}
> --
> 2.54.0
>
>

Reply via email to