Skip to content
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

[mlir][bugfix] Fix erroneous condition in getEffectsOnResource #133638

Merged
merged 1 commit into from
Apr 2, 2025

Conversation

banach-space
Copy link
Contributor

@banach-space banach-space commented Mar 30, 2025

This patch corrects an invalid condition in getEffectsOnResource used
to identify relevant "resources":

return it.getResource() != resource;

The current implementation assumes that only one instance of each
resource will exist, so comparing raw pointers is both safe and
sufficient. This assumption stems from constructs like:

static DerivedResource *get() {
  static DerivedResource instance;
  return &instance;
}

i.e., resource instances returned via static singleton methods.

However, as discussed in

this assumption breaks in practice — notably on macOS (Apple Silicon)
when built with:

  • -DBUILD_SHARED_LIBS=On.

In such cases, multiple instances of the same logical resource may exist
across shared library boundaries, leading to incorrect behavior and
causing failures in tests like:

  • test/Dialect/Transform/check-use-after-free.mlir

This patch replaces the pointer comparison with a comparison based on
resource identity:

return it.getResource()->getResourceID() != resource->getResourceID();

This approach aligns better with the intent of getEffectsOnResource,
which is to:

/// Collect all of the effect instances that operate on the provided
/// resource (...)

Fixes #129216

This patch corrects an invalid condition in `getEffectsOnResource` used
to identify relevant "resources":

```cpp
return it.getResource() != resource;
```

The current implementation assumes that only one instance of each
resource will exist, so comparing raw pointers is both safe and
sufficient. This assumption stems from constructs like:

```cpp
static DerivedResource *get() {
  static DerivedResource instance;
  return &instance;
}
```
i.e., resource instances returned via static singleton methods.

However, as discussed in
 * llvm#129216,
this assumption breaks in practice — notably on macOS (Apple Silicon)
when built with:

* `-DBUILD_SHARED_LIBS=On`.

In such cases, multiple instances of the same logical resource may exist
across shared library boundaries, leading to incorrect behavior and
causing failures in tests like:

* test/Dialect/Transform/check-use-after-free.mlir

This patch replaces the pointer comparison with a comparison based on
resource identity:

```cpp
return it.getResource()->getResourceID() != resource->getResourceID();
```
This approach aligns better with the intent of `getEffectsOnResource`,
which is to:

```cpp
/// Collect all of the effect instances that operate on the provided
/// resource (...)
```

Fixes llvm#129216
@llvmbot llvmbot added the mlir label Mar 30, 2025
@llvmbot
Copy link
Member

llvmbot commented Mar 30, 2025

@llvm/pr-subscribers-mlir

Author: Andrzej Warzyński (banach-space)

Changes

This patch corrects an invalid condition in getEffectsOnResource used
to identify relevant "resources":

return it.getResource() != resource;

The current implementation assumes that only one instance of each
resource will exist, so comparing raw pointers is both safe and
sufficient. This assumption stems from constructs like:

static DerivedResource *get() {
  static DerivedResource instance;
  return &instance;
}

i.e., resource instances returned via static singleton methods.

However, as discussed in

In such cases, multiple instances of the same logical resource may exist
across shared library boundaries, leading to incorrect behavior and
causing failures in tests like:

  • test/Dialect/Transform/check-use-after-free.mlir

This patch replaces the pointer comparison with a comparison based on
resource identity:

return it.getResource()->getResourceID() != resource->getResourceID();

This approach aligns better with the intent of getEffectsOnResource,
which is to:

/// Collect all of the effect instances that operate on the provided
/// resource (...)

Fixes #129216


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

1 Files Affected:

  • (modified) mlir/include/mlir/Interfaces/SideEffectInterfaceBase.td (+1-1)
diff --git a/mlir/include/mlir/Interfaces/SideEffectInterfaceBase.td b/mlir/include/mlir/Interfaces/SideEffectInterfaceBase.td
index 45a9ffa94363e..043829c24fda8 100644
--- a/mlir/include/mlir/Interfaces/SideEffectInterfaceBase.td
+++ b/mlir/include/mlir/Interfaces/SideEffectInterfaceBase.td
@@ -140,7 +140,7 @@ class EffectOpInterfaceBase<string name, string baseEffect>
               }] # baseEffect # [{>> & effects) {
       getEffects(effects);
       ::llvm::erase_if(effects, [&](auto &it) {
-        return it.getResource() != resource;
+        return it.getResource()->getResourceID() != resource->getResourceID();
       });
     }
   }];

Copy link
Member

@ftynse ftynse left a comment

Choose a reason for hiding this comment

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

Thanks! LGTM in principle, but I'd like to better understand why resource ID is not subject to having multiple definitions in different shared libraries?

@banach-space
Copy link
Contributor Author

Thanks for taking a look, Alex! 🙏

I'd like to better understand why the resource ID is not subject to having multiple definitions in different shared libraries?

Sure! If I understand correctly, this comes down to where the ID is "registered". That happens in ImplicitTypeIDRegistry, which is decorated with LLVM_ALWAYS_EXPORT and which ensures that there can be only one definition of the registry.

To add more context, let me share what makes ImplicitTypeIDRegistry relevant. Looking at the backtrace for Resource::getResourceID(), it calls:

  1. TypeId::get<DerivedResource>(), which calls ...
  2. resolveTypeID(), which in turn calls ...
  3. registerImplicitTypeID().

I hope that clarifies things and that I didn't miss anything here.

@banach-space
Copy link
Contributor Author

Since there are no new comments, I assume this is good to land.

@banach-space banach-space merged commit 2bee246 into llvm:main Apr 2, 2025
13 checks passed
@banach-space banach-space deleted the andrzej/fix/resource branch April 2, 2025 20:27
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[mlir][td] check-use-after-free.mlir failing (AppleSillcon + shared libs)
3 participants