Skip to content

[DenseMap] Fix constness issues with lookup_or #139247

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 2 commits into from
May 28, 2025

Conversation

artagnon
Copy link
Contributor

@artagnon artagnon commented May 9, 2025

Also demonstrate its use in ScalarEvolution.

@llvmbot
Copy link
Member

llvmbot commented May 9, 2025

@llvm/pr-subscribers-llvm-analysis

Author: Ramkumar Ramachandra (artagnon)

Changes

Also demonstrate its use in ScalarEvolution.


Full diff: https://github.com/llvm/llvm-project/pull/139247.diff

3 Files Affected:

  • (modified) llvm/include/llvm/ADT/DenseMap.h (+7-2)
  • (modified) llvm/lib/Analysis/ScalarEvolution.cpp (+11-21)
  • (modified) llvm/unittests/ADT/DenseMapTest.cpp (+3-1)
diff --git a/llvm/include/llvm/ADT/DenseMap.h b/llvm/include/llvm/ADT/DenseMap.h
index 4df50e03de94b..acc81f069a5aa 100644
--- a/llvm/include/llvm/ADT/DenseMap.h
+++ b/llvm/include/llvm/ADT/DenseMap.h
@@ -220,8 +220,13 @@ class DenseMapBase : public DebugEpochBase {
   // Return the entry with the specified key, or \p Default. This variant is
   // useful, because `lookup` cannot be used with non-default-constructible
   // values.
-  ValueT lookup_or(const_arg_type_t<KeyT> Val,
-                   const_arg_type_t<ValueT> Default) const {
+  ValueT lookup_or(const_arg_type_t<KeyT> Val, const ValueT &Default) const {
+    if (const BucketT *Bucket = doFind(Val))
+      return Bucket->getSecond();
+    return Default;
+  }
+
+  ValueT lookup_or(const_arg_type_t<KeyT> Val, ValueT &&Default) const {
     if (const BucketT *Bucket = doFind(Val))
       return Bucket->getSecond();
     return Default;
diff --git a/llvm/lib/Analysis/ScalarEvolution.cpp b/llvm/lib/Analysis/ScalarEvolution.cpp
index 3f9614254ae7a..0b0748b7ffe78 100644
--- a/llvm/lib/Analysis/ScalarEvolution.cpp
+++ b/llvm/lib/Analysis/ScalarEvolution.cpp
@@ -9571,15 +9571,14 @@ getConstantEvolvingPHIOperands(Instruction *UseInst, const Loop *L,
     if (!OpInst || !canConstantEvolve(OpInst, L)) return nullptr;
 
     PHINode *P = dyn_cast<PHINode>(OpInst);
-    if (!P)
+    if (!P) {
       // If this operand is already visited, reuse the prior result.
       // We may have P != PHI if this is the deepest point at which the
       // inconsistent paths meet.
-      P = PHIMap.lookup(OpInst);
-    if (!P) {
       // Recurse and memoize the results, whether a phi is found or not.
       // This recursive call invalidates pointers into PHIMap.
-      P = getConstantEvolvingPHIOperands(OpInst, L, PHIMap, Depth + 1);
+      P = PHIMap.lookup_or(
+          OpInst, getConstantEvolvingPHIOperands(OpInst, L, PHIMap, Depth + 1));
       PHIMap[OpInst] = P;
     }
     if (!P)
@@ -15860,10 +15859,7 @@ const SCEV *ScalarEvolution::LoopGuards::rewrite(const SCEV *Expr) const {
     const SCEV *visitAddRecExpr(const SCEVAddRecExpr *Expr) { return Expr; }
 
     const SCEV *visitUnknown(const SCEVUnknown *Expr) {
-      auto I = Map.find(Expr);
-      if (I == Map.end())
-        return Expr;
-      return I->second;
+      return Map.lookup_or(Expr, Expr);
     }
 
     const SCEV *visitZeroExtendExpr(const SCEVZeroExtendExpr *Expr) {
@@ -15891,25 +15887,19 @@ const SCEV *ScalarEvolution::LoopGuards::rewrite(const SCEV *Expr) const {
     }
 
     const SCEV *visitSignExtendExpr(const SCEVSignExtendExpr *Expr) {
-      auto I = Map.find(Expr);
-      if (I == Map.end())
-        return SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitSignExtendExpr(
-            Expr);
-      return I->second;
+      return Map.lookup_or(
+          Expr,
+          SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitSignExtendExpr(Expr));
     }
 
     const SCEV *visitUMinExpr(const SCEVUMinExpr *Expr) {
-      auto I = Map.find(Expr);
-      if (I == Map.end())
-        return SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitUMinExpr(Expr);
-      return I->second;
+      return Map.lookup_or(
+          Expr, SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitUMinExpr(Expr));
     }
 
     const SCEV *visitSMinExpr(const SCEVSMinExpr *Expr) {
-      auto I = Map.find(Expr);
-      if (I == Map.end())
-        return SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitSMinExpr(Expr);
-      return I->second;
+      return Map.lookup_or(
+          Expr, SCEVRewriteVisitor<SCEVLoopGuardRewriter>::visitSMinExpr(Expr));
     }
 
     const SCEV *visitAddExpr(const SCEVAddExpr *Expr) {
diff --git a/llvm/unittests/ADT/DenseMapTest.cpp b/llvm/unittests/ADT/DenseMapTest.cpp
index d002a3c15cf52..cf90b10495a14 100644
--- a/llvm/unittests/ADT/DenseMapTest.cpp
+++ b/llvm/unittests/ADT/DenseMapTest.cpp
@@ -668,7 +668,9 @@ TEST(DenseMapCustomTest, LookupOr) {
 
   EXPECT_EQ(M.lookup_or(0, 4u), 3u);
   EXPECT_EQ(M.lookup_or(1, 4u), 0u);
-  EXPECT_EQ(M.lookup_or(2, 4u), 4u);
+
+  NonDefaultConstructible DefaultV = M.lookup_or(2, 4u);
+  EXPECT_EQ(DefaultV, 4u);
 }
 
 // Key traits that allows lookup with either an unsigned or char* key;

@llvmbot
Copy link
Member

llvmbot commented May 9, 2025

@llvm/pr-subscribers-llvm-adt

Author: Ramkumar Ramachandra (artagnon)

Changes

Also demonstrate its use in ScalarEvolution.


Full diff: https://github.com/llvm/llvm-project/pull/139247.diff

3 Files Affected:

  • (modified) llvm/include/llvm/ADT/DenseMap.h (+7-2)
  • (modified) llvm/lib/Analysis/ScalarEvolution.cpp (+11-21)
  • (modified) llvm/unittests/ADT/DenseMapTest.cpp (+3-1)
<!DOCTYPE html>
<html>
  <head>
    <meta http-equiv="Content-type" content="text/html; charset=utf-8">
    <meta http-equiv="Content-Security-Policy" content="default-src 'none'; base-uri 'self'; connect-src 'self'; form-action 'self'; img-src 'self' data:; script-src 'self'; style-src 'unsafe-inline'">
    <meta content="origin" name="referrer">
    <title>Rate limit &middot; GitHub</title>
    <meta name="viewport" content="width=device-width">
    <style type="text/css" media="screen">
      body {
        background-color: #f6f8fa;
        color: #24292e;
        font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Helvetica,Arial,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol;
        font-size: 14px;
        line-height: 1.5;
        margin: 0;
      }

      .container { margin: 50px auto; max-width: 600px; text-align: center; padding: 0 24px; }

      a { color: #0366d6; text-decoration: none; }
      a:hover { text-decoration: underline; }

      h1 { line-height: 60px; font-size: 48px; font-weight: 300; margin: 0px; text-shadow: 0 1px 0 #fff; }
      p { color: rgba(0, 0, 0, 0.5); margin: 20px 0 40px; }

      ul { list-style: none; margin: 25px 0; padding: 0; }
      li { display: table-cell; font-weight: bold; width: 1%; }

      .logo { display: inline-block; margin-top: 35px; }
      .logo-img-2x { display: none; }
      @media
      only screen and (-webkit-min-device-pixel-ratio: 2),
      only screen and (   min--moz-device-pixel-ratio: 2),
      only screen and (     -o-min-device-pixel-ratio: 2/1),
      only screen and (        min-device-pixel-ratio: 2),
      only screen and (                min-resolution: 192dpi),
      only screen and (                min-resolution: 2dppx) {
        .logo-img-1x { display: none; }
        .logo-img-2x { display: inline-block; }
      }

      #suggestions {
        margin-top: 35px;
        color: #ccc;
      }
      #suggestions a {
        color: #666666;
        font-weight: 200;
        font-size: 14px;
        margin: 0 10px;
      }

    </style>
  </head>
  <body>

    <div class="container">

      <h1>Whoa there!</h1>
      <p>You have exceeded a secondary rate limit.<br><br>
        Please wait a few minutes before you try again;<br>
        in some cases this may take up to an hour.
      </p>
      <div id="suggestions">
        <a href="https://support.github.com/contact">Contact Support</a> &mdash;
        <a href="https://githubstatus.com">GitHub Status</a> &mdash;
        <a href="https://twitter.com/githubstatus">@githubstatus</a>
      </div>

      <a href="/" class="logo logo-img-1x">
        <img width="32" height="32" title="" alt="" src="">
      </a>

      <a href="/" class="logo logo-img-2x">
        <img width="32" height="32" title="" alt="" src="">
      </a>
    </div>
  </body>
</html>

@artagnon artagnon force-pushed the densemap-lookupor-fix branch from d5ca963 to 9c22c18 Compare May 9, 2025 12:36
@artagnon
Copy link
Contributor Author

Gentle ping. The const_arg_t<ValueT> is incorrect when the value in the map is non-cost. This patch isn't a blocker for anything yet, but it would be nice to get this fixed.

@artagnon
Copy link
Contributor Author

Gentle ping.

@kazutakahirata
Copy link
Contributor

@artagnon Do we need the const ValueT &Default variant? I'm looking at std::optional<T>::value_or as a reference, and it only has the rvalue reference variant.

@artagnon artagnon force-pushed the densemap-lookupor-fix branch from 9c22c18 to b9cd220 Compare May 27, 2025 20:28
@artagnon
Copy link
Contributor Author

@artagnon Do we need the const ValueT &Default variant? I'm looking at std::optional<T>::value_or as a reference, and it only has the rvalue reference variant.

Good catch! I think that, by the same reasoning, several of the const-lvalue-reference variants can also be removed in a follow-up.

@kazutakahirata
Copy link
Contributor

@artagnon Is there any way you can come with a test case that would fail without your change to DenseMap.h? I just tried your changes without DenseMap.h, and both ScalarEvolution.cpp and DenseMapTest.cpp were happy. This means that if somebody reverted your DenseMap.h change in the future, nothing would break.

By the way, what is the core issue here? const_arg_type_t<ValueT> prevents move semantics? Or am I missing something?

artagnon and others added 2 commits May 28, 2025 11:02
@artagnon artagnon force-pushed the densemap-lookupor-fix branch from b9cd220 to 9f49a22 Compare May 28, 2025 09:27
@artagnon
Copy link
Contributor Author

@artagnon Is there any way you can come with a test case that would fail without your change to DenseMap.h? I just tried your changes without DenseMap.h, and both ScalarEvolution.cpp and DenseMapTest.cpp were happy. This means that if somebody reverted your DenseMap.h change in the future, nothing would break.

By the way, what is the core issue here? const_arg_type_t<ValueT> prevents move semantics? Or am I missing something?

Sorry about the thinko in the test case: should be fixed now. The included test case now fails with:

/Users/artagnon/src/llvm/llvm/include/llvm/ADT/DenseMap.h:227:12: error: cannot initialize return object of type 'unsigned int *' with an lvalue of type 'const_arg_type_t<unsigned int *>' (aka 'const unsigned int *')
  227 |     return Default;
      |            ^~~~~~~
/Users/artagnon/src/llvm/llvm/unittests/ADT/DenseMapTest.cpp:677:21: note: in instantiation of member function 'llvm::DenseMapBase<llvm::DenseMap<int, unsigned int *>, int, unsigned int *, llvm::DenseMapInfo<int>, llvm::detail::DenseMapPair<int, unsigned int *>>::lookup_or' requested here
  677 |   unsigned *Ret = M.lookup_or(0, &Default);
      |                     ^

without the DenseMap change. The issue isn't move semantics, but rather a constness.

@kazutakahirata
Copy link
Contributor

Sorry about the thinko in the test case: should be fixed now. The included test case now fails with:

/Users/artagnon/src/llvm/llvm/include/llvm/ADT/DenseMap.h:227:12: error: cannot initialize return object of type 'unsigned int *' with an lvalue of type 'const_arg_type_t<unsigned int *>' (aka 'const unsigned int *')
  227 |     return Default;
      |            ^~~~~~~
/Users/artagnon/src/llvm/llvm/unittests/ADT/DenseMapTest.cpp:677:21: note: in instantiation of member function 'llvm::DenseMapBase<llvm::DenseMap<int, unsigned int *>, int, unsigned int *, llvm::DenseMapInfo<int>, llvm::detail::DenseMapPair<int, unsigned int *>>::lookup_or' requested here
  677 |   unsigned *Ret = M.lookup_or(0, &Default);
      |                     ^

without the DenseMap change. The issue isn't move semantics, but rather a constness.

I've verified the same failure.

So, the issue is about unnecessarily putting const on the value type. Thanks for the update and explanation!

Copy link
Contributor

@kazutakahirata kazutakahirata left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM. Thanks!

@artagnon artagnon merged commit 4bf67cd into llvm:main May 28, 2025
11 checks passed
@artagnon artagnon deleted the densemap-lookupor-fix branch May 28, 2025 18:32
@aaronpuchert
Copy link
Member

Could this be related to the build failure in https://lab.llvm.org/buildbot/#/builders/63/builds/6559?

FAILED: lib/Analysis/CMakeFiles/LLVMAnalysis.dir/ScalarEvolution.cpp.obj 
C:\PROGRA~2\MICROS~3\2019\PROFES~1\VC\Tools\MSVC\1428~1.293\bin\Hostx64\x64\cl.exe  /nologo /TP -DGTEST_HAS_RTTI=0 -DUNICODE -D_CRT_NONSTDC_NO_DEPRECATE -D_CRT_NONSTDC_NO_WARNINGS -D_CRT_SECURE_NO_DEPRECATE -D_CRT_SECURE_NO_WARNINGS -D_HAS_EXCEPTIONS=0 -D_SCL_SECURE_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS -D_UNICODE -D__STDC_CONSTANT_MACROS -D__STDC_FORMAT_MACROS -D__STDC_LIMIT_MACROS -IC:\b\slave\clang-x64-windows-msvc\build\stage1\lib\Analysis -IC:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\lib\Analysis -IC:\b\slave\clang-x64-windows-msvc\build\stage1\include -IC:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\include /DWIN32 /D_WINDOWS   /Zc:inline /Zc:preprocessor /Zc:__cplusplus /Oi /bigobj /permissive- /W4 -wd4141 -wd4146 -wd4244 -wd4267 -wd4291 -wd4351 -wd4456 -wd4457 -wd4458 -wd4459 -wd4503 -wd4624 -wd4722 -wd4100 -wd4127 -wd4512 -wd4505 -wd4610 -wd4510 -wd4702 -wd4245 -wd4706 -wd4310 -wd4701 -wd4703 -wd4389 -wd4611 -wd4805 -wd4204 -wd4577 -wd4091 -wd4592 -wd4319 -wd4709 -wd5105 -wd4324 -wd4251 -wd4275 -w14062 -we4238 /Gw /O2 /Ob2 /DNDEBUG -std:c++17 -MD  /EHs-c- /GR- /showIncludes /Folib\Analysis\CMakeFiles\LLVMAnalysis.dir\ScalarEvolution.cpp.obj /Fdlib\Analysis\CMakeFiles\LLVMAnalysis.dir\LLVMAnalysis.pdb /FS -c C:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\lib\Analysis\ScalarEvolution.cpp
C:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\lib\Analysis\ScalarEvolution.cpp(15907): error C2664: 'ValueT llvm::DenseMapBase<llvm::DenseMap<const llvm::SCEV *,ValueT,llvm::DenseMapInfo<KeyT,void>,llvm::detail::DenseMapPair<KeyT,ValueT>>,KeyT,ValueT,KeyInfoT,BucketT>::lookup_or(const llvm::SCEV *,ValueT &&) const': cannot convert argument 2 from 'const llvm::SCEVUnknown *' to 'ValueT &&'
        with
        [
            ValueT=const llvm::SCEV *,
            KeyT=const llvm::SCEV *,
            KeyInfoT=llvm::DenseMapInfo<const llvm::SCEV *,void>,
            BucketT=llvm::detail::DenseMapPair<const llvm::SCEV *,const llvm::SCEV *>
        ]
        and
        [
            ValueT=const llvm::SCEV *
        ]
C:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\lib\Analysis\ScalarEvolution.cpp(15907): note: You cannot bind an lvalue to an rvalue reference
C:\b\slave\clang-x64-windows-msvc\llvm-project\llvm\include\llvm/ADT/DenseMap.h(223): note: see declaration of 'llvm::DenseMapBase<llvm::DenseMap<const llvm::SCEV *,const llvm::SCEV *,llvm::DenseMapInfo<KeyT,void>,llvm::detail::DenseMapPair<KeyT,ValueT>>,KeyT,ValueT,KeyInfoT,BucketT>::lookup_or'
        with
        [
            KeyT=const llvm::SCEV *,
            ValueT=const llvm::SCEV *,
            KeyInfoT=llvm::DenseMapInfo<const llvm::SCEV *,void>,
            BucketT=llvm::detail::DenseMapPair<const llvm::SCEV *,const llvm::SCEV *>
        ]

@aaronpuchert
Copy link
Member

Side note: I think that MSVC is incorrect and the reference initialization should be allowed according to [dcl.init.ref]p5.4.2, because const SCEV* and const SCEVUnknown* are not reference-related.

@artagnon
Copy link
Contributor Author

artagnon commented May 31, 2025

Side note: I think that MSVC is incorrect and the reference initialization should be allowed according to [dcl.init.ref]p5.4.2, because const SCEV* and const SCEVUnknown* are not reference-related.

Yes, there seems to be a bug in MSVC, and I think creating a const ValueT & variant would fix it? See #142268.

sivan-shani pushed a commit to sivan-shani/llvm-project that referenced this pull request Jun 3, 2025
Also demonstrate its use in ScalarEvolution.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants