diff --git a/mlir/lib/Bindings/Python/IRCore.cpp b/mlir/lib/Bindings/Python/IRCore.cpp --- a/mlir/lib/Bindings/Python/IRCore.cpp +++ b/mlir/lib/Bindings/Python/IRCore.cpp @@ -2102,6 +2102,10 @@ }) .def("__eq__", [](PyOperationBase &self, py::object other) { return false; }) + .def("__hash__", + [](PyOperationBase &self) { + return static_cast(llvm::hash_value(&self.getOperation())); + }) .def_property_readonly("attributes", [](PyOperationBase &self) { return PyOpAttributeMap( @@ -2458,7 +2462,10 @@ .def("__eq__", [](PyAttribute &self, PyAttribute &other) { return self == other; }) .def("__eq__", [](PyAttribute &self, py::object &other) { return false; }) - .def("__hash__", [](PyAttribute &self) { return (size_t)self.get().ptr; }) + .def("__hash__", + [](PyAttribute &self) { + return static_cast(llvm::hash_value(self.get().ptr)); + }) .def( "dump", [](PyAttribute &self) { mlirAttributeDump(self); }, kDumpDocstring) @@ -2552,7 +2559,10 @@ "Context that owns the Type") .def("__eq__", [](PyType &self, PyType &other) { return self == other; }) .def("__eq__", [](PyType &self, py::object &other) { return false; }) - .def("__hash__", [](PyType &self) { return (size_t)self.get().ptr; }) + .def("__hash__", + [](PyType &self) { + return static_cast(llvm::hash_value(self.get().ptr)); + }) .def( "dump", [](PyType &self) { mlirTypeDump(self); }, kDumpDocstring) .def( @@ -2603,6 +2613,10 @@ return self.get().ptr == other.get().ptr; }) .def("__eq__", [](PyValue &self, py::object other) { return false; }) + .def("__hash__", + [](PyValue &self) { + return static_cast(llvm::hash_value(self.get().ptr)); + }) .def( "__str__", [](PyValue &self) { diff --git a/mlir/test/python/ir/attributes.py b/mlir/test/python/ir/attributes.py --- a/mlir/test/python/ir/attributes.py +++ b/mlir/test/python/ir/attributes.py @@ -66,10 +66,6 @@ a3 = Attribute.parse('"attr1"') # CHECK: hash(a1) == hash(a3): True print("hash(a1) == hash(a3):", a1.__hash__() == a3.__hash__()) - # In general, hashes don't have to be unique. In this case, however, the - # hash is just the underlying pointer so it will be. - # CHECK: hash(a1) == hash(a2): False - print("hash(a1) == hash(a2):", a1.__hash__() == a2.__hash__()) s = set() s.add(a1) diff --git a/mlir/test/python/ir/builtin_types.py b/mlir/test/python/ir/builtin_types.py --- a/mlir/test/python/ir/builtin_types.py +++ b/mlir/test/python/ir/builtin_types.py @@ -67,10 +67,6 @@ # CHECK: hash(t1) == hash(t3): True print("hash(t1) == hash(t3):", t1.__hash__() == t3.__hash__()) - # In general, hashes don't have to be unique. In this case, however, the - # hash is just the underlying pointer so it will be. - # CHECK: hash(t1) == hash(t2): False - print("hash(t1) == hash(t2):", t1.__hash__() == t2.__hash__()) s = set() s.add(t1) diff --git a/mlir/test/python/ir/operation.py b/mlir/test/python/ir/operation.py --- a/mlir/test/python/ir/operation.py +++ b/mlir/test/python/ir/operation.py @@ -740,3 +740,13 @@ op = Operation.create("custom.op", loc=loc) assert op.location == loc assert op.operation.location == loc + + +# CHECK-LABEL: TEST: testOperationHash +@run +def testOperationHash(): + ctx = Context() + ctx.allow_unregistered_dialects = True + with ctx, Location.unknown(): + op = Operation.create("custom.op1") + assert hash(op) == hash(op.operation) diff --git a/mlir/test/python/ir/value.py b/mlir/test/python/ir/value.py --- a/mlir/test/python/ir/value.py +++ b/mlir/test/python/ir/value.py @@ -55,3 +55,22 @@ op = func.regions[0].blocks[0].operations[0] assert not BlockArgument.isinstance(op.results[0]) assert OpResult.isinstance(op.results[0]) + + +# CHECK-LABEL: TEST: testValueHash +@run +def testValueHash(): + ctx = Context() + ctx.allow_unregistered_dialects = True + module = Module.parse( + r""" + func @foo(%arg0: f32) -> f32 { + %0 = "some_dialect.some_op"(%arg0) : (f32) -> f32 + return %0 : f32 + }""", ctx) + + [func] = module.body.operations + block = func.entry_block + op, ret = block.operations + assert hash(block.arguments[0]) == hash(op.operands[0]) + assert hash(op.result) == hash(ret.operands[0])