xref: /llvm-project/mlir/lib/IR/OperationSupport.cpp (revision 98de5dfe6a8cbb70f21de545acec4710a77294ed)
1a7215a90SAlex Zinenko //===- OperationSupport.cpp -----------------------------------------------===//
2a7215a90SAlex Zinenko //
330857107SMehdi Amini // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
456222a06SMehdi Amini // See https://llvm.org/LICENSE.txt for license information.
556222a06SMehdi Amini // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6a7215a90SAlex Zinenko //
756222a06SMehdi Amini //===----------------------------------------------------------------------===//
8a7215a90SAlex Zinenko //
9a7215a90SAlex Zinenko // This file contains out-of-line implementations of the support types that
109ffdc930SRiver Riddle // Operation and related classes build on top of.
11a7215a90SAlex Zinenko //
12a7215a90SAlex Zinenko //===----------------------------------------------------------------------===//
13a7215a90SAlex Zinenko 
14a7215a90SAlex Zinenko #include "mlir/IR/OperationSupport.h"
154e103a12SRiver Riddle #include "mlir/IR/BuiltinAttributes.h"
1609f7a55fSRiver Riddle #include "mlir/IR/BuiltinTypes.h"
17108abd2fSRiver Riddle #include "mlir/IR/OpDefinition.h"
18a776ecb6SRiver Riddle #include "llvm/ADT/BitVector.h"
192e36dadbSIvan Butygin #include "llvm/Support/SHA1.h"
204e103a12SRiver Riddle #include <numeric>
21a1fe1f5fSKazu Hirata #include <optional>
22a776ecb6SRiver Riddle 
239ffdc930SRiver Riddle using namespace mlir;
24a7215a90SAlex Zinenko 
259ffdc930SRiver Riddle //===----------------------------------------------------------------------===//
265eae715aSJacques Pienaar // NamedAttrList
275eae715aSJacques Pienaar //===----------------------------------------------------------------------===//
285eae715aSJacques Pienaar 
295eae715aSJacques Pienaar NamedAttrList::NamedAttrList(ArrayRef<NamedAttribute> attributes) {
305eae715aSJacques Pienaar   assign(attributes.begin(), attributes.end());
315eae715aSJacques Pienaar }
325eae715aSJacques Pienaar 
33fc5cf50eSRiver Riddle NamedAttrList::NamedAttrList(DictionaryAttr attributes)
34fc5cf50eSRiver Riddle     : NamedAttrList(attributes ? attributes.getValue()
35fc5cf50eSRiver Riddle                                : ArrayRef<NamedAttribute>()) {
36fc5cf50eSRiver Riddle   dictionarySorted.setPointerAndInt(attributes, true);
37fc5cf50eSRiver Riddle }
38fc5cf50eSRiver Riddle 
3902b6fb21SMehdi Amini NamedAttrList::NamedAttrList(const_iterator inStart, const_iterator inEnd) {
4002b6fb21SMehdi Amini   assign(inStart, inEnd);
415eae715aSJacques Pienaar }
425eae715aSJacques Pienaar 
435eae715aSJacques Pienaar ArrayRef<NamedAttribute> NamedAttrList::getAttrs() const { return attrs; }
445eae715aSJacques Pienaar 
450a81ace0SKazu Hirata std::optional<NamedAttribute> NamedAttrList::findDuplicate() const {
460a81ace0SKazu Hirata   std::optional<NamedAttribute> duplicate =
47c298824fSRahul Joshi       DictionaryAttr::findDuplicate(attrs, isSorted());
48c298824fSRahul Joshi   // DictionaryAttr::findDuplicate will sort the list, so reset the sorted
49c298824fSRahul Joshi   // state.
50c298824fSRahul Joshi   if (!isSorted())
51c298824fSRahul Joshi     dictionarySorted.setPointerAndInt(nullptr, true);
52c298824fSRahul Joshi   return duplicate;
53c298824fSRahul Joshi }
54c298824fSRahul Joshi 
555eae715aSJacques Pienaar DictionaryAttr NamedAttrList::getDictionary(MLIRContext *context) const {
565eae715aSJacques Pienaar   if (!isSorted()) {
575eae715aSJacques Pienaar     DictionaryAttr::sortInPlace(attrs);
585eae715aSJacques Pienaar     dictionarySorted.setPointerAndInt(nullptr, true);
595eae715aSJacques Pienaar   }
605eae715aSJacques Pienaar   if (!dictionarySorted.getPointer())
612f37cdd5SRiver Riddle     dictionarySorted.setPointer(DictionaryAttr::getWithSorted(context, attrs));
62c1fa60b4STres Popp   return llvm::cast<DictionaryAttr>(dictionarySorted.getPointer());
635eae715aSJacques Pienaar }
645eae715aSJacques Pienaar 
655eae715aSJacques Pienaar /// Replaces the attributes with new list of attributes.
6602b6fb21SMehdi Amini void NamedAttrList::assign(const_iterator inStart, const_iterator inEnd) {
6702b6fb21SMehdi Amini   DictionaryAttr::sort(ArrayRef<NamedAttribute>{inStart, inEnd}, attrs);
685eae715aSJacques Pienaar   dictionarySorted.setPointerAndInt(nullptr, true);
695eae715aSJacques Pienaar }
705eae715aSJacques Pienaar 
715eae715aSJacques Pienaar void NamedAttrList::push_back(NamedAttribute newAttribute) {
720c7890c8SRiver Riddle   if (isSorted())
730c7890c8SRiver Riddle     dictionarySorted.setInt(attrs.empty() || attrs.back() < newAttribute);
745eae715aSJacques Pienaar   dictionarySorted.setPointer(nullptr);
755eae715aSJacques Pienaar   attrs.push_back(newAttribute);
765eae715aSJacques Pienaar }
775eae715aSJacques Pienaar 
785eae715aSJacques Pienaar /// Return the specified attribute if present, null otherwise.
795eae715aSJacques Pienaar Attribute NamedAttrList::get(StringRef name) const {
802125eb34SMogball   auto it = findAttr(*this, name);
810c7890c8SRiver Riddle   return it.second ? it.first->getValue() : Attribute();
825eae715aSJacques Pienaar }
83195730a6SRiver Riddle Attribute NamedAttrList::get(StringAttr name) const {
842125eb34SMogball   auto it = findAttr(*this, name);
850c7890c8SRiver Riddle   return it.second ? it.first->getValue() : Attribute();
865eae715aSJacques Pienaar }
875eae715aSJacques Pienaar 
8815ae9964SKazu Hirata /// Return the specified named attribute if present, std::nullopt otherwise.
890a81ace0SKazu Hirata std::optional<NamedAttribute> NamedAttrList::getNamed(StringRef name) const {
902125eb34SMogball   auto it = findAttr(*this, name);
910a81ace0SKazu Hirata   return it.second ? *it.first : std::optional<NamedAttribute>();
925eae715aSJacques Pienaar }
930a81ace0SKazu Hirata std::optional<NamedAttribute> NamedAttrList::getNamed(StringAttr name) const {
942125eb34SMogball   auto it = findAttr(*this, name);
950a81ace0SKazu Hirata   return it.second ? *it.first : std::optional<NamedAttribute>();
965eae715aSJacques Pienaar }
975eae715aSJacques Pienaar 
985eae715aSJacques Pienaar /// If the an attribute exists with the specified name, change it to the new
995eae715aSJacques Pienaar /// value.  Otherwise, add a new attribute with the specified name/value.
100195730a6SRiver Riddle Attribute NamedAttrList::set(StringAttr name, Attribute value) {
1015eae715aSJacques Pienaar   assert(value && "attributes may never be null");
1025eae715aSJacques Pienaar 
1032125eb34SMogball   // Look for an existing attribute with the given name, and set its value
1042125eb34SMogball   // in-place. Return the previous value of the attribute, if there was one.
1052125eb34SMogball   auto it = findAttr(*this, name);
1062125eb34SMogball   if (it.second) {
1072125eb34SMogball     // Update the existing attribute by swapping out the old value for the new
1082125eb34SMogball     // value. Return the old value.
1090c7890c8SRiver Riddle     Attribute oldValue = it.first->getValue();
1100c7890c8SRiver Riddle     if (it.first->getValue() != value) {
1110c7890c8SRiver Riddle       it.first->setValue(value);
1120c7890c8SRiver Riddle 
1132125eb34SMogball       // If the attributes have changed, the dictionary is invalidated.
1145eae715aSJacques Pienaar       dictionarySorted.setPointer(nullptr);
115fc5cf50eSRiver Riddle     }
1160c7890c8SRiver Riddle     return oldValue;
1175eae715aSJacques Pienaar   }
1182125eb34SMogball   // Perform a string lookup to insert the new attribute into its sorted
1192125eb34SMogball   // position.
1202125eb34SMogball   if (isSorted())
1212125eb34SMogball     it = findAttr(*this, name.strref());
1222125eb34SMogball   attrs.insert(it.first, {name, value});
1232125eb34SMogball   // Invalidate the dictionary. Return null as there was no previous value.
1245eae715aSJacques Pienaar   dictionarySorted.setPointer(nullptr);
125fc5cf50eSRiver Riddle   return Attribute();
1265eae715aSJacques Pienaar }
1272125eb34SMogball 
128fc5cf50eSRiver Riddle Attribute NamedAttrList::set(StringRef name, Attribute value) {
1292125eb34SMogball   assert(value && "attributes may never be null");
130195730a6SRiver Riddle   return set(mlir::StringAttr::get(value.getContext(), name), value);
1315eae715aSJacques Pienaar }
1325eae715aSJacques Pienaar 
13303e6f40cSAlex Zinenko Attribute
13403e6f40cSAlex Zinenko NamedAttrList::eraseImpl(SmallVectorImpl<NamedAttribute>::iterator it) {
13503e6f40cSAlex Zinenko   // Erasing does not affect the sorted property.
1360c7890c8SRiver Riddle   Attribute attr = it->getValue();
13703e6f40cSAlex Zinenko   attrs.erase(it);
13803e6f40cSAlex Zinenko   dictionarySorted.setPointer(nullptr);
13903e6f40cSAlex Zinenko   return attr;
14003e6f40cSAlex Zinenko }
14103e6f40cSAlex Zinenko 
142195730a6SRiver Riddle Attribute NamedAttrList::erase(StringAttr name) {
1432125eb34SMogball   auto it = findAttr(*this, name);
1442125eb34SMogball   return it.second ? eraseImpl(it.first) : Attribute();
14503e6f40cSAlex Zinenko }
14603e6f40cSAlex Zinenko 
14703e6f40cSAlex Zinenko Attribute NamedAttrList::erase(StringRef name) {
1482125eb34SMogball   auto it = findAttr(*this, name);
1492125eb34SMogball   return it.second ? eraseImpl(it.first) : Attribute();
15003e6f40cSAlex Zinenko }
15103e6f40cSAlex Zinenko 
1525eae715aSJacques Pienaar NamedAttrList &
1535eae715aSJacques Pienaar NamedAttrList::operator=(const SmallVectorImpl<NamedAttribute> &rhs) {
1545eae715aSJacques Pienaar   assign(rhs.begin(), rhs.end());
1555eae715aSJacques Pienaar   return *this;
1565eae715aSJacques Pienaar }
1575eae715aSJacques Pienaar 
1585eae715aSJacques Pienaar NamedAttrList::operator ArrayRef<NamedAttribute>() const { return attrs; }
1595eae715aSJacques Pienaar 
1605eae715aSJacques Pienaar //===----------------------------------------------------------------------===//
1619ffdc930SRiver Riddle // OperationState
1629ffdc930SRiver Riddle //===----------------------------------------------------------------------===//
163a7215a90SAlex Zinenko 
16448d6cf1cSRiver Riddle OperationState::OperationState(Location location, StringRef name)
1652f59f768SRiver Riddle     : location(location), name(name, location->getContext()) {}
166a7215a90SAlex Zinenko 
16748d6cf1cSRiver Riddle OperationState::OperationState(Location location, OperationName name)
1682f59f768SRiver Riddle     : location(location), name(name) {}
169a7215a90SAlex Zinenko 
17011067d71SRiver Riddle OperationState::OperationState(Location location, OperationName name,
171a6ae6950SRahul Joshi                                ValueRange operands, TypeRange types,
172a7215a90SAlex Zinenko                                ArrayRef<NamedAttribute> attributes,
173a6ae6950SRahul Joshi                                BlockRange successors,
1744dfd1b5fSRiver Riddle                                MutableArrayRef<std::unique_ptr<Region>> regions)
17511067d71SRiver Riddle     : location(location), name(name),
176a7215a90SAlex Zinenko       operands(operands.begin(), operands.end()),
177a7215a90SAlex Zinenko       types(types.begin(), types.end()),
178a7215a90SAlex Zinenko       attributes(attributes.begin(), attributes.end()),
179a7215a90SAlex Zinenko       successors(successors.begin(), successors.end()) {
1802f59f768SRiver Riddle   for (std::unique_ptr<Region> &r : regions)
181a7215a90SAlex Zinenko     this->regions.push_back(std::move(r));
182a7215a90SAlex Zinenko }
18311067d71SRiver Riddle OperationState::OperationState(Location location, StringRef name,
18411067d71SRiver Riddle                                ValueRange operands, TypeRange types,
18511067d71SRiver Riddle                                ArrayRef<NamedAttribute> attributes,
18611067d71SRiver Riddle                                BlockRange successors,
18711067d71SRiver Riddle                                MutableArrayRef<std::unique_ptr<Region>> regions)
18811067d71SRiver Riddle     : OperationState(location, OperationName(name, location.getContext()),
18911067d71SRiver Riddle                      operands, types, attributes, successors, regions) {}
190a7215a90SAlex Zinenko 
1915e118f93SMehdi Amini OperationState::~OperationState() {
1925e118f93SMehdi Amini   if (properties)
1935e118f93SMehdi Amini     propertiesDeleter(properties);
1945e118f93SMehdi Amini }
1955e118f93SMehdi Amini 
1968c2bff1aSMehdi Amini LogicalResult OperationState::setProperties(
197c50617daSMehdi Amini     Operation *op, function_ref<InFlightDiagnostic()> emitError) const {
1985e118f93SMehdi Amini   if (LLVM_UNLIKELY(propertiesAttr)) {
1995e118f93SMehdi Amini     assert(!properties);
200c50617daSMehdi Amini     return op->setPropertiesFromAttribute(propertiesAttr, emitError);
2015e118f93SMehdi Amini   }
2025e118f93SMehdi Amini   if (properties)
2035e118f93SMehdi Amini     propertiesSetter(op->getPropertiesStorage(), properties);
2045e118f93SMehdi Amini   return success();
2055e118f93SMehdi Amini }
2065e118f93SMehdi Amini 
207d6ee6a03SRiver Riddle void OperationState::addOperands(ValueRange newOperands) {
208d6ee6a03SRiver Riddle   operands.append(newOperands.begin(), newOperands.end());
209d6ee6a03SRiver Riddle }
210d6ee6a03SRiver Riddle 
2118893d081SRahul Joshi void OperationState::addSuccessors(BlockRange newSuccessors) {
212cb177712SRiver Riddle   successors.append(newSuccessors.begin(), newSuccessors.end());
213d6ee6a03SRiver Riddle }
214d6ee6a03SRiver Riddle 
215a7215a90SAlex Zinenko Region *OperationState::addRegion() {
216a7215a90SAlex Zinenko   regions.emplace_back(new Region);
217a7215a90SAlex Zinenko   return regions.back().get();
218a7215a90SAlex Zinenko }
219a7215a90SAlex Zinenko 
220a7215a90SAlex Zinenko void OperationState::addRegion(std::unique_ptr<Region> &&region) {
221a7215a90SAlex Zinenko   regions.push_back(std::move(region));
222a7215a90SAlex Zinenko }
223a7215a90SAlex Zinenko 
224eaeadce9SRiver Riddle void OperationState::addRegions(
225eaeadce9SRiver Riddle     MutableArrayRef<std::unique_ptr<Region>> regions) {
226eaeadce9SRiver Riddle   for (std::unique_ptr<Region> &region : regions)
227eaeadce9SRiver Riddle     addRegion(std::move(region));
228eaeadce9SRiver Riddle }
229eaeadce9SRiver Riddle 
2309ffdc930SRiver Riddle //===----------------------------------------------------------------------===//
2319ffdc930SRiver Riddle // OperandStorage
2329ffdc930SRiver Riddle //===----------------------------------------------------------------------===//
2339ffdc930SRiver Riddle 
234a0391134SRiver Riddle detail::OperandStorage::OperandStorage(Operation *owner,
235a0391134SRiver Riddle                                        OpOperand *trailingOperands,
236a0391134SRiver Riddle                                        ValueRange values)
237a0391134SRiver Riddle     : isStorageDynamic(false), operandStorage(trailingOperands) {
238a0391134SRiver Riddle   numOperands = capacity = values.size();
239a0391134SRiver Riddle   for (unsigned i = 0; i < numOperands; ++i)
240a0391134SRiver Riddle     new (&operandStorage[i]) OpOperand(owner, values[i]);
2414dfd1b5fSRiver Riddle }
2424dfd1b5fSRiver Riddle 
2434dfd1b5fSRiver Riddle detail::OperandStorage::~OperandStorage() {
244a0391134SRiver Riddle   for (auto &operand : getOperands())
245a0391134SRiver Riddle     operand.~OpOperand();
246a0391134SRiver Riddle 
247a0391134SRiver Riddle   // If the storage is dynamic, deallocate it.
248a0391134SRiver Riddle   if (isStorageDynamic)
249a0391134SRiver Riddle     free(operandStorage);
2504dfd1b5fSRiver Riddle }
2514dfd1b5fSRiver Riddle 
2529ffdc930SRiver Riddle /// Replace the operands contained in the storage with the ones provided in
2534dfd1b5fSRiver Riddle /// 'values'.
2544dfd1b5fSRiver Riddle void detail::OperandStorage::setOperands(Operation *owner, ValueRange values) {
2554dfd1b5fSRiver Riddle   MutableArrayRef<OpOperand> storageOperands = resize(owner, values.size());
2564dfd1b5fSRiver Riddle   for (unsigned i = 0, e = values.size(); i != e; ++i)
2574dfd1b5fSRiver Riddle     storageOperands[i].set(values[i]);
2584dfd1b5fSRiver Riddle }
2594dfd1b5fSRiver Riddle 
260108abd2fSRiver Riddle /// Replace the operands beginning at 'start' and ending at 'start' + 'length'
261108abd2fSRiver Riddle /// with the ones provided in 'operands'. 'operands' may be smaller or larger
262108abd2fSRiver Riddle /// than the range pointed to by 'start'+'length'.
263108abd2fSRiver Riddle void detail::OperandStorage::setOperands(Operation *owner, unsigned start,
264108abd2fSRiver Riddle                                          unsigned length, ValueRange operands) {
265108abd2fSRiver Riddle   // If the new size is the same, we can update inplace.
266108abd2fSRiver Riddle   unsigned newSize = operands.size();
267108abd2fSRiver Riddle   if (newSize == length) {
268108abd2fSRiver Riddle     MutableArrayRef<OpOperand> storageOperands = getOperands();
269108abd2fSRiver Riddle     for (unsigned i = 0, e = length; i != e; ++i)
270108abd2fSRiver Riddle       storageOperands[start + i].set(operands[i]);
271108abd2fSRiver Riddle     return;
272108abd2fSRiver Riddle   }
273108abd2fSRiver Riddle   // If the new size is greater, remove the extra operands and set the rest
274108abd2fSRiver Riddle   // inplace.
275108abd2fSRiver Riddle   if (newSize < length) {
276108abd2fSRiver Riddle     eraseOperands(start + operands.size(), length - newSize);
277108abd2fSRiver Riddle     setOperands(owner, start, newSize, operands);
278108abd2fSRiver Riddle     return;
279108abd2fSRiver Riddle   }
280108abd2fSRiver Riddle   // Otherwise, the new size is greater so we need to grow the storage.
281108abd2fSRiver Riddle   auto storageOperands = resize(owner, size() + (newSize - length));
282108abd2fSRiver Riddle 
283108abd2fSRiver Riddle   // Shift operands to the right to make space for the new operands.
284108abd2fSRiver Riddle   unsigned rotateSize = storageOperands.size() - (start + length);
285108abd2fSRiver Riddle   auto rbegin = storageOperands.rbegin();
286108abd2fSRiver Riddle   std::rotate(rbegin, std::next(rbegin, newSize - length), rbegin + rotateSize);
287108abd2fSRiver Riddle 
288108abd2fSRiver Riddle   // Update the operands inplace.
289108abd2fSRiver Riddle   for (unsigned i = 0, e = operands.size(); i != e; ++i)
290108abd2fSRiver Riddle     storageOperands[start + i].set(operands[i]);
291108abd2fSRiver Riddle }
292108abd2fSRiver Riddle 
293108abd2fSRiver Riddle /// Erase an operand held by the storage.
294108abd2fSRiver Riddle void detail::OperandStorage::eraseOperands(unsigned start, unsigned length) {
295a0391134SRiver Riddle   MutableArrayRef<OpOperand> operands = getOperands();
296108abd2fSRiver Riddle   assert((start + length) <= operands.size());
297a0391134SRiver Riddle   numOperands -= length;
298108abd2fSRiver Riddle 
299108abd2fSRiver Riddle   // Shift all operands down if the operand to remove is not at the end.
300a0391134SRiver Riddle   if (start != numOperands) {
3015eae715aSJacques Pienaar     auto *indexIt = std::next(operands.begin(), start);
302108abd2fSRiver Riddle     std::rotate(indexIt, std::next(indexIt, length), operands.end());
303108abd2fSRiver Riddle   }
304108abd2fSRiver Riddle   for (unsigned i = 0; i != length; ++i)
305a0391134SRiver Riddle     operands[numOperands + i].~OpOperand();
306108abd2fSRiver Riddle }
307108abd2fSRiver Riddle 
308b7f93c28SJeff Niu void detail::OperandStorage::eraseOperands(const BitVector &eraseIndices) {
309a0391134SRiver Riddle   MutableArrayRef<OpOperand> operands = getOperands();
310a776ecb6SRiver Riddle   assert(eraseIndices.size() == operands.size());
311a776ecb6SRiver Riddle 
312a776ecb6SRiver Riddle   // Check that at least one operand is erased.
313a776ecb6SRiver Riddle   int firstErasedIndice = eraseIndices.find_first();
314a776ecb6SRiver Riddle   if (firstErasedIndice == -1)
315a776ecb6SRiver Riddle     return;
316a776ecb6SRiver Riddle 
317a776ecb6SRiver Riddle   // Shift all of the removed operands to the end, and destroy them.
318a0391134SRiver Riddle   numOperands = firstErasedIndice;
319a776ecb6SRiver Riddle   for (unsigned i = firstErasedIndice + 1, e = operands.size(); i < e; ++i)
320a776ecb6SRiver Riddle     if (!eraseIndices.test(i))
321a0391134SRiver Riddle       operands[numOperands++] = std::move(operands[i]);
322a0391134SRiver Riddle   for (OpOperand &operand : operands.drop_front(numOperands))
323a776ecb6SRiver Riddle     operand.~OpOperand();
324a776ecb6SRiver Riddle }
325a776ecb6SRiver Riddle 
3264dfd1b5fSRiver Riddle /// Resize the storage to the given size. Returns the array containing the new
3274dfd1b5fSRiver Riddle /// operands.
3284dfd1b5fSRiver Riddle MutableArrayRef<OpOperand> detail::OperandStorage::resize(Operation *owner,
3294dfd1b5fSRiver Riddle                                                           unsigned newSize) {
3309ffdc930SRiver Riddle   // If the number of operands is less than or equal to the current amount, we
3319ffdc930SRiver Riddle   // can just update in place.
332a0391134SRiver Riddle   MutableArrayRef<OpOperand> origOperands = getOperands();
3334dfd1b5fSRiver Riddle   if (newSize <= numOperands) {
3344dfd1b5fSRiver Riddle     // If the number of new size is less than the current, remove any extra
3354dfd1b5fSRiver Riddle     // operands.
3364dfd1b5fSRiver Riddle     for (unsigned i = newSize; i != numOperands; ++i)
337a0391134SRiver Riddle       origOperands[i].~OpOperand();
3384dfd1b5fSRiver Riddle     numOperands = newSize;
339a0391134SRiver Riddle     return origOperands.take_front(newSize);
3409ffdc930SRiver Riddle   }
3419ffdc930SRiver Riddle 
3424dfd1b5fSRiver Riddle   // If the new size is within the original inline capacity, grow inplace.
343a0391134SRiver Riddle   if (newSize <= capacity) {
344a0391134SRiver Riddle     OpOperand *opBegin = origOperands.data();
3454dfd1b5fSRiver Riddle     for (unsigned e = newSize; numOperands != e; ++numOperands)
3464dfd1b5fSRiver Riddle       new (&opBegin[numOperands]) OpOperand(owner);
3474dfd1b5fSRiver Riddle     return MutableArrayRef<OpOperand>(opBegin, newSize);
3488da0f85eSRiver Riddle   }
3499ffdc930SRiver Riddle 
3504dfd1b5fSRiver Riddle   // Otherwise, we need to allocate a new storage.
3514dfd1b5fSRiver Riddle   unsigned newCapacity =
352a0391134SRiver Riddle       std::max(unsigned(llvm::NextPowerOf2(capacity + 2)), newSize);
353a0391134SRiver Riddle   OpOperand *newOperandStorage =
354a0391134SRiver Riddle       reinterpret_cast<OpOperand *>(malloc(sizeof(OpOperand) * newCapacity));
3554dfd1b5fSRiver Riddle 
3564dfd1b5fSRiver Riddle   // Move the current operands to the new storage.
357a0391134SRiver Riddle   MutableArrayRef<OpOperand> newOperands(newOperandStorage, newSize);
3581bcf21caSBenjamin Kramer   std::uninitialized_move(origOperands.begin(), origOperands.end(),
3594dfd1b5fSRiver Riddle                           newOperands.begin());
3604dfd1b5fSRiver Riddle 
3614dfd1b5fSRiver Riddle   // Destroy the original operands.
362a0391134SRiver Riddle   for (auto &operand : origOperands)
3634dfd1b5fSRiver Riddle     operand.~OpOperand();
3644dfd1b5fSRiver Riddle 
3654dfd1b5fSRiver Riddle   // Initialize any new operands.
3664dfd1b5fSRiver Riddle   for (unsigned e = newSize; numOperands != e; ++numOperands)
3674dfd1b5fSRiver Riddle     new (&newOperands[numOperands]) OpOperand(owner);
3684dfd1b5fSRiver Riddle 
369a0391134SRiver Riddle   // If the current storage is dynamic, free it.
370a0391134SRiver Riddle   if (isStorageDynamic)
371a0391134SRiver Riddle     free(operandStorage);
3724dfd1b5fSRiver Riddle 
3734dfd1b5fSRiver Riddle   // Update the storage representation to use the new dynamic storage.
374a0391134SRiver Riddle   operandStorage = newOperandStorage;
375a0391134SRiver Riddle   capacity = newCapacity;
376a0391134SRiver Riddle   isStorageDynamic = true;
3774dfd1b5fSRiver Riddle   return newOperands;
3789ffdc930SRiver Riddle }
3799ffdc930SRiver Riddle 
3809ed22ae5SRiver Riddle //===----------------------------------------------------------------------===//
3819ed22ae5SRiver Riddle // Operation Value-Iterators
3829ed22ae5SRiver Riddle //===----------------------------------------------------------------------===//
3839ed22ae5SRiver Riddle 
3849ed22ae5SRiver Riddle //===----------------------------------------------------------------------===//
3859ed22ae5SRiver Riddle // OperandRange
3869ed22ae5SRiver Riddle 
387621d7ccaSRiver Riddle unsigned OperandRange::getBeginOperandIndex() const {
388621d7ccaSRiver Riddle   assert(!empty() && "range must not be empty");
389621d7ccaSRiver Riddle   return base->getOperandNumber();
390621d7ccaSRiver Riddle }
391621d7ccaSRiver Riddle 
39258a47508SJeff Niu OperandRangeRange OperandRange::split(DenseI32ArrayAttr segmentSizes) const {
3934e103a12SRiver Riddle   return OperandRangeRange(*this, segmentSizes);
3944e103a12SRiver Riddle }
3954e103a12SRiver Riddle 
3964e103a12SRiver Riddle //===----------------------------------------------------------------------===//
3974e103a12SRiver Riddle // OperandRangeRange
3984e103a12SRiver Riddle 
3994e103a12SRiver Riddle OperandRangeRange::OperandRangeRange(OperandRange operands,
4004e103a12SRiver Riddle                                      Attribute operandSegments)
4014e103a12SRiver Riddle     : OperandRangeRange(OwnerT(operands.getBase(), operandSegments), 0,
402c1fa60b4STres Popp                         llvm::cast<DenseI32ArrayAttr>(operandSegments).size()) {
403c1fa60b4STres Popp }
4044e103a12SRiver Riddle 
4054e103a12SRiver Riddle OperandRange OperandRangeRange::join() const {
4064e103a12SRiver Riddle   const OwnerT &owner = getBase();
407c1fa60b4STres Popp   ArrayRef<int32_t> sizeData = llvm::cast<DenseI32ArrayAttr>(owner.second);
4084e103a12SRiver Riddle   return OperandRange(owner.first,
4094e103a12SRiver Riddle                       std::accumulate(sizeData.begin(), sizeData.end(), 0));
4104e103a12SRiver Riddle }
4114e103a12SRiver Riddle 
4124e103a12SRiver Riddle OperandRange OperandRangeRange::dereference(const OwnerT &object,
4134e103a12SRiver Riddle                                             ptrdiff_t index) {
414c1fa60b4STres Popp   ArrayRef<int32_t> sizeData = llvm::cast<DenseI32ArrayAttr>(object.second);
4154e103a12SRiver Riddle   uint32_t startIndex =
4164e103a12SRiver Riddle       std::accumulate(sizeData.begin(), sizeData.begin() + index, 0);
4174e103a12SRiver Riddle   return OperandRange(object.first + startIndex, *(sizeData.begin() + index));
4184e103a12SRiver Riddle }
4194e103a12SRiver Riddle 
4209ed22ae5SRiver Riddle //===----------------------------------------------------------------------===//
421108abd2fSRiver Riddle // MutableOperandRange
422108abd2fSRiver Riddle 
423108abd2fSRiver Riddle /// Construct a new mutable range from the given operand, operand start index,
424108abd2fSRiver Riddle /// and range length.
425108abd2fSRiver Riddle MutableOperandRange::MutableOperandRange(
426108abd2fSRiver Riddle     Operation *owner, unsigned start, unsigned length,
427108abd2fSRiver Riddle     ArrayRef<OperandSegment> operandSegments)
428108abd2fSRiver Riddle     : owner(owner), start(start), length(length),
4295262865aSKazu Hirata       operandSegments(operandSegments) {
430108abd2fSRiver Riddle   assert((start + length) <= owner->getNumOperands() && "invalid range");
431108abd2fSRiver Riddle }
432108abd2fSRiver Riddle MutableOperandRange::MutableOperandRange(Operation *owner)
433108abd2fSRiver Riddle     : MutableOperandRange(owner, /*start=*/0, owner->getNumOperands()) {}
434108abd2fSRiver Riddle 
4358823e961SMatthias Springer /// Construct a new mutable range for the given OpOperand.
4368823e961SMatthias Springer MutableOperandRange::MutableOperandRange(OpOperand &opOperand)
4378823e961SMatthias Springer     : MutableOperandRange(opOperand.getOwner(),
4388823e961SMatthias Springer                           /*start=*/opOperand.getOperandNumber(),
4398823e961SMatthias Springer                           /*length=*/1) {}
4408823e961SMatthias Springer 
4410752d98cSRiver Riddle /// Slice this range into a sub range, with the additional operand segment.
4420752d98cSRiver Riddle MutableOperandRange
4430752d98cSRiver Riddle MutableOperandRange::slice(unsigned subStart, unsigned subLen,
4440a81ace0SKazu Hirata                            std::optional<OperandSegment> segment) const {
4450752d98cSRiver Riddle   assert((subStart + subLen) <= length && "invalid sub-range");
4460752d98cSRiver Riddle   MutableOperandRange subSlice(owner, start + subStart, subLen,
4470752d98cSRiver Riddle                                operandSegments);
4480752d98cSRiver Riddle   if (segment)
4490752d98cSRiver Riddle     subSlice.operandSegments.push_back(*segment);
4500752d98cSRiver Riddle   return subSlice;
4510752d98cSRiver Riddle }
4520752d98cSRiver Riddle 
453108abd2fSRiver Riddle /// Append the given values to the range.
454108abd2fSRiver Riddle void MutableOperandRange::append(ValueRange values) {
455108abd2fSRiver Riddle   if (values.empty())
456108abd2fSRiver Riddle     return;
457108abd2fSRiver Riddle   owner->insertOperands(start + length, values);
458108abd2fSRiver Riddle   updateLength(length + values.size());
459108abd2fSRiver Riddle }
460108abd2fSRiver Riddle 
461108abd2fSRiver Riddle /// Assign this range to the given values.
462108abd2fSRiver Riddle void MutableOperandRange::assign(ValueRange values) {
463108abd2fSRiver Riddle   owner->setOperands(start, length, values);
464108abd2fSRiver Riddle   if (length != values.size())
465108abd2fSRiver Riddle     updateLength(/*newLength=*/values.size());
466108abd2fSRiver Riddle }
467108abd2fSRiver Riddle 
468108abd2fSRiver Riddle /// Assign the range to the given value.
469108abd2fSRiver Riddle void MutableOperandRange::assign(Value value) {
470108abd2fSRiver Riddle   if (length == 1) {
471108abd2fSRiver Riddle     owner->setOperand(start, value);
472108abd2fSRiver Riddle   } else {
473108abd2fSRiver Riddle     owner->setOperands(start, length, value);
474108abd2fSRiver Riddle     updateLength(/*newLength=*/1);
475108abd2fSRiver Riddle   }
476108abd2fSRiver Riddle }
477108abd2fSRiver Riddle 
478108abd2fSRiver Riddle /// Erase the operands within the given sub-range.
479108abd2fSRiver Riddle void MutableOperandRange::erase(unsigned subStart, unsigned subLen) {
480108abd2fSRiver Riddle   assert((subStart + subLen) <= length && "invalid sub-range");
481108abd2fSRiver Riddle   if (length == 0)
482108abd2fSRiver Riddle     return;
483108abd2fSRiver Riddle   owner->eraseOperands(start + subStart, subLen);
484108abd2fSRiver Riddle   updateLength(length - subLen);
485108abd2fSRiver Riddle }
486108abd2fSRiver Riddle 
487108abd2fSRiver Riddle /// Clear this range and erase all of the operands.
488108abd2fSRiver Riddle void MutableOperandRange::clear() {
489108abd2fSRiver Riddle   if (length != 0) {
490108abd2fSRiver Riddle     owner->eraseOperands(start, length);
491108abd2fSRiver Riddle     updateLength(/*newLength=*/0);
492108abd2fSRiver Riddle   }
493108abd2fSRiver Riddle }
494108abd2fSRiver Riddle 
495a9304edfSThomas Preud'homme /// Explicit conversion to an OperandRange.
496a9304edfSThomas Preud'homme OperandRange MutableOperandRange::getAsOperandRange() const {
497a9304edfSThomas Preud'homme   return owner->getOperands().slice(start, length);
498a9304edfSThomas Preud'homme }
499a9304edfSThomas Preud'homme 
500108abd2fSRiver Riddle /// Allow implicit conversion to an OperandRange.
501108abd2fSRiver Riddle MutableOperandRange::operator OperandRange() const {
502a9304edfSThomas Preud'homme   return getAsOperandRange();
503108abd2fSRiver Riddle }
504108abd2fSRiver Riddle 
5058fb0d77bSMatthias Springer MutableOperandRange::operator MutableArrayRef<OpOperand>() const {
5068fb0d77bSMatthias Springer   return owner->getOpOperands().slice(start, length);
5078fb0d77bSMatthias Springer }
5088fb0d77bSMatthias Springer 
5094e103a12SRiver Riddle MutableOperandRangeRange
5104e103a12SRiver Riddle MutableOperandRange::split(NamedAttribute segmentSizes) const {
5114e103a12SRiver Riddle   return MutableOperandRangeRange(*this, segmentSizes);
5124e103a12SRiver Riddle }
5134e103a12SRiver Riddle 
514108abd2fSRiver Riddle /// Update the length of this range to the one provided.
515108abd2fSRiver Riddle void MutableOperandRange::updateLength(unsigned newLength) {
516108abd2fSRiver Riddle   int32_t diff = int32_t(newLength) - int32_t(length);
517108abd2fSRiver Riddle   length = newLength;
518108abd2fSRiver Riddle 
519108abd2fSRiver Riddle   // Update any of the provided segment attributes.
520108abd2fSRiver Riddle   for (OperandSegment &segment : operandSegments) {
521c1fa60b4STres Popp     auto attr = llvm::cast<DenseI32ArrayAttr>(segment.second.getValue());
52258a47508SJeff Niu     SmallVector<int32_t, 8> segments(attr.asArrayRef());
523108abd2fSRiver Riddle     segments[segment.first] += diff;
5240c7890c8SRiver Riddle     segment.second.setValue(
52558a47508SJeff Niu         DenseI32ArrayAttr::get(attr.getContext(), segments));
5260c7890c8SRiver Riddle     owner->setAttr(segment.second.getName(), segment.second.getValue());
527108abd2fSRiver Riddle   }
528108abd2fSRiver Riddle }
529108abd2fSRiver Riddle 
5300f952cfeSMatthias Springer OpOperand &MutableOperandRange::operator[](unsigned index) const {
5310f952cfeSMatthias Springer   assert(index < length && "index is out of bounds");
5320f952cfeSMatthias Springer   return owner->getOpOperand(start + index);
5330f952cfeSMatthias Springer }
5340f952cfeSMatthias Springer 
5356923a315SMatthias Springer MutableArrayRef<OpOperand>::iterator MutableOperandRange::begin() const {
5366923a315SMatthias Springer   return owner->getOpOperands().slice(start, length).begin();
5376923a315SMatthias Springer }
5386923a315SMatthias Springer 
5396923a315SMatthias Springer MutableArrayRef<OpOperand>::iterator MutableOperandRange::end() const {
5406923a315SMatthias Springer   return owner->getOpOperands().slice(start, length).end();
5416923a315SMatthias Springer }
5426923a315SMatthias Springer 
543108abd2fSRiver Riddle //===----------------------------------------------------------------------===//
5444e103a12SRiver Riddle // MutableOperandRangeRange
5454e103a12SRiver Riddle 
5464e103a12SRiver Riddle MutableOperandRangeRange::MutableOperandRangeRange(
5474e103a12SRiver Riddle     const MutableOperandRange &operands, NamedAttribute operandSegmentAttr)
5484e103a12SRiver Riddle     : MutableOperandRangeRange(
5494e103a12SRiver Riddle           OwnerT(operands, operandSegmentAttr), 0,
550c1fa60b4STres Popp           llvm::cast<DenseI32ArrayAttr>(operandSegmentAttr.getValue()).size()) {
551c1fa60b4STres Popp }
5524e103a12SRiver Riddle 
5534e103a12SRiver Riddle MutableOperandRange MutableOperandRangeRange::join() const {
5544e103a12SRiver Riddle   return getBase().first;
5554e103a12SRiver Riddle }
5564e103a12SRiver Riddle 
5574e103a12SRiver Riddle MutableOperandRangeRange::operator OperandRangeRange() const {
55858a47508SJeff Niu   return OperandRangeRange(getBase().first, getBase().second.getValue());
5594e103a12SRiver Riddle }
5604e103a12SRiver Riddle 
5614e103a12SRiver Riddle MutableOperandRange MutableOperandRangeRange::dereference(const OwnerT &object,
5624e103a12SRiver Riddle                                                           ptrdiff_t index) {
56358a47508SJeff Niu   ArrayRef<int32_t> sizeData =
564c1fa60b4STres Popp       llvm::cast<DenseI32ArrayAttr>(object.second.getValue());
5654e103a12SRiver Riddle   uint32_t startIndex =
5664e103a12SRiver Riddle       std::accumulate(sizeData.begin(), sizeData.begin() + index, 0);
5674e103a12SRiver Riddle   return object.first.slice(
5684e103a12SRiver Riddle       startIndex, *(sizeData.begin() + index),
5694e103a12SRiver Riddle       MutableOperandRange::OperandSegment(index, object.second));
5704e103a12SRiver Riddle }
5714e103a12SRiver Riddle 
5724e103a12SRiver Riddle //===----------------------------------------------------------------------===//
573aea3026eSRiver Riddle // ResultRange
574aea3026eSRiver Riddle 
575015192c6SRiver Riddle ResultRange::ResultRange(OpResult result)
576015192c6SRiver Riddle     : ResultRange(static_cast<detail::OpResultImpl *>(Value(result).getImpl()),
577015192c6SRiver Riddle                   1) {}
578015192c6SRiver Riddle 
579aea3026eSRiver Riddle ResultRange::use_range ResultRange::getUses() const {
580aea3026eSRiver Riddle   return {use_begin(), use_end()};
581aea3026eSRiver Riddle }
582aea3026eSRiver Riddle ResultRange::use_iterator ResultRange::use_begin() const {
583aea3026eSRiver Riddle   return use_iterator(*this);
584aea3026eSRiver Riddle }
585aea3026eSRiver Riddle ResultRange::use_iterator ResultRange::use_end() const {
586aea3026eSRiver Riddle   return use_iterator(*this, /*end=*/true);
587aea3026eSRiver Riddle }
588aea3026eSRiver Riddle ResultRange::user_range ResultRange::getUsers() {
589aea3026eSRiver Riddle   return {user_begin(), user_end()};
590aea3026eSRiver Riddle }
591aea3026eSRiver Riddle ResultRange::user_iterator ResultRange::user_begin() {
592aea3026eSRiver Riddle   return user_iterator(use_begin());
593aea3026eSRiver Riddle }
594aea3026eSRiver Riddle ResultRange::user_iterator ResultRange::user_end() {
595aea3026eSRiver Riddle   return user_iterator(use_end());
596aea3026eSRiver Riddle }
597aea3026eSRiver Riddle 
598aea3026eSRiver Riddle ResultRange::UseIterator::UseIterator(ResultRange results, bool end)
599aea3026eSRiver Riddle     : it(end ? results.end() : results.begin()), endIt(results.end()) {
600aea3026eSRiver Riddle   // Only initialize current use if there are results/can be uses.
601aea3026eSRiver Riddle   if (it != endIt)
602aea3026eSRiver Riddle     skipOverResultsWithNoUsers();
603aea3026eSRiver Riddle }
604aea3026eSRiver Riddle 
605aea3026eSRiver Riddle ResultRange::UseIterator &ResultRange::UseIterator::operator++() {
606aea3026eSRiver Riddle   // We increment over uses, if we reach the last use then move to next
607aea3026eSRiver Riddle   // result.
608aea3026eSRiver Riddle   if (use != (*it).use_end())
609aea3026eSRiver Riddle     ++use;
610aea3026eSRiver Riddle   if (use == (*it).use_end()) {
611aea3026eSRiver Riddle     ++it;
612aea3026eSRiver Riddle     skipOverResultsWithNoUsers();
613aea3026eSRiver Riddle   }
614aea3026eSRiver Riddle   return *this;
615aea3026eSRiver Riddle }
616aea3026eSRiver Riddle 
617aea3026eSRiver Riddle void ResultRange::UseIterator::skipOverResultsWithNoUsers() {
618aea3026eSRiver Riddle   while (it != endIt && (*it).use_empty())
619aea3026eSRiver Riddle     ++it;
620aea3026eSRiver Riddle 
621aea3026eSRiver Riddle   // If we are at the last result, then set use to first use of
622aea3026eSRiver Riddle   // first result (sentinel value used for end).
623aea3026eSRiver Riddle   if (it == endIt)
624aea3026eSRiver Riddle     use = {};
625aea3026eSRiver Riddle   else
626aea3026eSRiver Riddle     use = (*it).use_begin();
627aea3026eSRiver Riddle }
628aea3026eSRiver Riddle 
629015192c6SRiver Riddle void ResultRange::replaceAllUsesWith(Operation *op) {
630015192c6SRiver Riddle   replaceAllUsesWith(op->getResults());
631015192c6SRiver Riddle }
632015192c6SRiver Riddle 
6338a583dd2SUday Bondhugula void ResultRange::replaceUsesWithIf(
6348a583dd2SUday Bondhugula     Operation *op, function_ref<bool(OpOperand &)> shouldReplace) {
6358a583dd2SUday Bondhugula   replaceUsesWithIf(op->getResults(), shouldReplace);
6368a583dd2SUday Bondhugula }
6378a583dd2SUday Bondhugula 
638aea3026eSRiver Riddle //===----------------------------------------------------------------------===//
6399ed22ae5SRiver Riddle // ValueRange
6409ed22ae5SRiver Riddle 
641ab46543cSRiver Riddle ValueRange::ValueRange(ArrayRef<Value> values)
6429ed22ae5SRiver Riddle     : ValueRange(values.data(), values.size()) {}
6439ed22ae5SRiver Riddle ValueRange::ValueRange(OperandRange values)
6449ed22ae5SRiver Riddle     : ValueRange(values.begin().getBase(), values.size()) {}
6459ed22ae5SRiver Riddle ValueRange::ValueRange(ResultRange values)
6463dfa8614SRiver Riddle     : ValueRange(values.getBase(), values.size()) {}
6479ed22ae5SRiver Riddle 
648204c3b55SRiver Riddle /// See `llvm::detail::indexed_accessor_range_base` for details.
6499ed22ae5SRiver Riddle ValueRange::OwnerT ValueRange::offset_base(const OwnerT &owner,
6509ed22ae5SRiver Riddle                                            ptrdiff_t index) {
65168f58812STres Popp   if (const auto *value = llvm::dyn_cast_if_present<const Value *>(owner))
652fd01d862SRiver Riddle     return {value + index};
65368f58812STres Popp   if (auto *operand = llvm::dyn_cast_if_present<OpOperand *>(owner))
654fd01d862SRiver Riddle     return {operand + index};
655*fcb1591bSKazu Hirata   return cast<detail::OpResultImpl *>(owner)->getNextResultAtOffset(index);
6569ed22ae5SRiver Riddle }
657204c3b55SRiver Riddle /// See `llvm::detail::indexed_accessor_range_base` for details.
658ab46543cSRiver Riddle Value ValueRange::dereference_iterator(const OwnerT &owner, ptrdiff_t index) {
65968f58812STres Popp   if (const auto *value = llvm::dyn_cast_if_present<const Value *>(owner))
660fd01d862SRiver Riddle     return value[index];
66168f58812STres Popp   if (auto *operand = llvm::dyn_cast_if_present<OpOperand *>(owner))
6629ed22ae5SRiver Riddle     return operand[index].get();
663*fcb1591bSKazu Hirata   return cast<detail::OpResultImpl *>(owner)->getNextResultAtOffset(index);
6649ed22ae5SRiver Riddle }
665df00e466SRiver Riddle 
666df00e466SRiver Riddle //===----------------------------------------------------------------------===//
667df00e466SRiver Riddle // Operation Equivalency
668df00e466SRiver Riddle //===----------------------------------------------------------------------===//
669df00e466SRiver Riddle 
6700be5d1a9SMehdi Amini llvm::hash_code OperationEquivalence::computeHash(
6710be5d1a9SMehdi Amini     Operation *op, function_ref<llvm::hash_code(Value)> hashOperands,
6720be5d1a9SMehdi Amini     function_ref<llvm::hash_code(Value)> hashResults, Flags flags) {
673df00e466SRiver Riddle   // Hash operations based upon their:
674df00e466SRiver Riddle   //   - Operation Name
675df00e466SRiver Riddle   //   - Attributes
676df00e466SRiver Riddle   //   - Result Types
6775e118f93SMehdi Amini   llvm::hash_code hash =
678b336ab42SOleksandr "Alex" Zinenko       llvm::hash_combine(op->getName(), op->getRawDictionaryAttrs(),
6795e118f93SMehdi Amini                          op->getResultTypes(), op->hashProperties());
680df00e466SRiver Riddle 
681fd824782SHideto Ueno   //   - Location if required
682fd824782SHideto Ueno   if (!(flags & Flags::IgnoreLocations))
683fd824782SHideto Ueno     hash = llvm::hash_combine(hash, op->getLoc());
684fd824782SHideto Ueno 
685df00e466SRiver Riddle   //   - Operands
686ee2deb4cSJacques Pienaar   if (op->hasTrait<mlir::OpTrait::IsCommutative>() &&
687ee2deb4cSJacques Pienaar       op->getNumOperands() > 0) {
688ee2deb4cSJacques Pienaar     size_t operandHash = hashOperands(op->getOperand(0));
689ee2deb4cSJacques Pienaar     for (auto operand : op->getOperands().drop_front())
690ee2deb4cSJacques Pienaar       operandHash += hashOperands(operand);
691ee2deb4cSJacques Pienaar     hash = llvm::hash_combine(hash, operandHash);
692ee2deb4cSJacques Pienaar   } else {
6932109587cStomnatan     for (Value operand : op->getOperands())
6940be5d1a9SMehdi Amini       hash = llvm::hash_combine(hash, hashOperands(operand));
695ee2deb4cSJacques Pienaar   }
696bd514967SValentin Clement 
6972109587cStomnatan   //   - Results
6980be5d1a9SMehdi Amini   for (Value result : op->getResults())
6990be5d1a9SMehdi Amini     hash = llvm::hash_combine(hash, hashResults(result));
700469c02d0SRiver Riddle   return hash;
701469c02d0SRiver Riddle }
702df00e466SRiver Riddle 
70331fc47e3SFrederik Gossen /*static*/ bool OperationEquivalence::isRegionEquivalentTo(
70431fc47e3SFrederik Gossen     Region *lhs, Region *rhs,
705c864288dSMatthias Springer     function_ref<LogicalResult(Value, Value)> checkEquivalent,
706c864288dSMatthias Springer     function_ref<void(Value, Value)> markEquivalent,
707ee2deb4cSJacques Pienaar     OperationEquivalence::Flags flags,
708ee2deb4cSJacques Pienaar     function_ref<LogicalResult(ValueRange, ValueRange)>
709ee2deb4cSJacques Pienaar         checkCommutativeEquivalent) {
7100be5d1a9SMehdi Amini   DenseMap<Block *, Block *> blocksMap;
7110be5d1a9SMehdi Amini   auto blocksEquivalent = [&](Block &lBlock, Block &rBlock) {
7120be5d1a9SMehdi Amini     // Check block arguments.
7130be5d1a9SMehdi Amini     if (lBlock.getNumArguments() != rBlock.getNumArguments())
7140be5d1a9SMehdi Amini       return false;
7150be5d1a9SMehdi Amini 
7160be5d1a9SMehdi Amini     // Map the two blocks.
7170be5d1a9SMehdi Amini     auto insertion = blocksMap.insert({&lBlock, &rBlock});
7180be5d1a9SMehdi Amini     if (insertion.first->getSecond() != &rBlock)
7190be5d1a9SMehdi Amini       return false;
7200be5d1a9SMehdi Amini 
7210be5d1a9SMehdi Amini     for (auto argPair :
7220be5d1a9SMehdi Amini          llvm::zip(lBlock.getArguments(), rBlock.getArguments())) {
7230be5d1a9SMehdi Amini       Value curArg = std::get<0>(argPair);
7240be5d1a9SMehdi Amini       Value otherArg = std::get<1>(argPair);
7250be5d1a9SMehdi Amini       if (curArg.getType() != otherArg.getType())
7260be5d1a9SMehdi Amini         return false;
7270be5d1a9SMehdi Amini       if (!(flags & OperationEquivalence::IgnoreLocations) &&
7280be5d1a9SMehdi Amini           curArg.getLoc() != otherArg.getLoc())
7290be5d1a9SMehdi Amini         return false;
730c864288dSMatthias Springer       // Corresponding bbArgs are equivalent.
731c864288dSMatthias Springer       if (markEquivalent)
732c864288dSMatthias Springer         markEquivalent(curArg, otherArg);
7330be5d1a9SMehdi Amini     }
7340be5d1a9SMehdi Amini 
7350be5d1a9SMehdi Amini     auto opsEquivalent = [&](Operation &lOp, Operation &rOp) {
7360be5d1a9SMehdi Amini       // Check for op equality (recursively).
737c864288dSMatthias Springer       if (!OperationEquivalence::isEquivalentTo(&lOp, &rOp, checkEquivalent,
738ee2deb4cSJacques Pienaar                                                 markEquivalent, flags,
739ee2deb4cSJacques Pienaar                                                 checkCommutativeEquivalent))
7400be5d1a9SMehdi Amini         return false;
7410be5d1a9SMehdi Amini       // Check successor mapping.
7420be5d1a9SMehdi Amini       for (auto successorsPair :
7430be5d1a9SMehdi Amini            llvm::zip(lOp.getSuccessors(), rOp.getSuccessors())) {
7440be5d1a9SMehdi Amini         Block *curSuccessor = std::get<0>(successorsPair);
7450be5d1a9SMehdi Amini         Block *otherSuccessor = std::get<1>(successorsPair);
7460be5d1a9SMehdi Amini         auto insertion = blocksMap.insert({curSuccessor, otherSuccessor});
7470be5d1a9SMehdi Amini         if (insertion.first->getSecond() != otherSuccessor)
7480be5d1a9SMehdi Amini           return false;
7490be5d1a9SMehdi Amini       }
7500be5d1a9SMehdi Amini       return true;
7510be5d1a9SMehdi Amini     };
7520be5d1a9SMehdi Amini     return llvm::all_of_zip(lBlock, rBlock, opsEquivalent);
7530be5d1a9SMehdi Amini   };
7540be5d1a9SMehdi Amini   return llvm::all_of_zip(*lhs, *rhs, blocksEquivalent);
7550be5d1a9SMehdi Amini }
7560be5d1a9SMehdi Amini 
75731fc47e3SFrederik Gossen // Value equivalence cache to be used with `isRegionEquivalentTo` and
75831fc47e3SFrederik Gossen // `isEquivalentTo`.
75931fc47e3SFrederik Gossen struct ValueEquivalenceCache {
76031fc47e3SFrederik Gossen   DenseMap<Value, Value> equivalentValues;
76131fc47e3SFrederik Gossen   LogicalResult checkEquivalent(Value lhsValue, Value rhsValue) {
76231fc47e3SFrederik Gossen     return success(lhsValue == rhsValue ||
76331fc47e3SFrederik Gossen                    equivalentValues.lookup(lhsValue) == rhsValue);
76431fc47e3SFrederik Gossen   }
765ee2deb4cSJacques Pienaar   LogicalResult checkCommutativeEquivalent(ValueRange lhsRange,
766ee2deb4cSJacques Pienaar                                            ValueRange rhsRange) {
767ee2deb4cSJacques Pienaar     // Handle simple case where sizes mismatch.
768ee2deb4cSJacques Pienaar     if (lhsRange.size() != rhsRange.size())
769ee2deb4cSJacques Pienaar       return failure();
770ee2deb4cSJacques Pienaar 
771ee2deb4cSJacques Pienaar     // Handle where operands in order are equivalent.
772ee2deb4cSJacques Pienaar     auto lhsIt = lhsRange.begin();
773ee2deb4cSJacques Pienaar     auto rhsIt = rhsRange.begin();
774ee2deb4cSJacques Pienaar     for (; lhsIt != lhsRange.end(); ++lhsIt, ++rhsIt) {
775ee2deb4cSJacques Pienaar       if (failed(checkEquivalent(*lhsIt, *rhsIt)))
776ee2deb4cSJacques Pienaar         break;
777ee2deb4cSJacques Pienaar     }
778ee2deb4cSJacques Pienaar     if (lhsIt == lhsRange.end())
779ee2deb4cSJacques Pienaar       return success();
780ee2deb4cSJacques Pienaar 
781ee2deb4cSJacques Pienaar     // Handle another simple case where operands are just a permutation.
782ee2deb4cSJacques Pienaar     // Note: This is not sufficient, this handles simple cases relatively
783ee2deb4cSJacques Pienaar     // cheaply.
784ee2deb4cSJacques Pienaar     auto sortValues = [](ValueRange values) {
785ee2deb4cSJacques Pienaar       SmallVector<Value> sortedValues = llvm::to_vector(values);
786ee2deb4cSJacques Pienaar       llvm::sort(sortedValues, [](Value a, Value b) {
787ee2deb4cSJacques Pienaar         return a.getAsOpaquePointer() < b.getAsOpaquePointer();
788ee2deb4cSJacques Pienaar       });
789ee2deb4cSJacques Pienaar       return sortedValues;
790ee2deb4cSJacques Pienaar     };
791ee2deb4cSJacques Pienaar     auto lhsSorted = sortValues({lhsIt, lhsRange.end()});
792ee2deb4cSJacques Pienaar     auto rhsSorted = sortValues({rhsIt, rhsRange.end()});
793ee2deb4cSJacques Pienaar     return success(lhsSorted == rhsSorted);
794ee2deb4cSJacques Pienaar   }
79531fc47e3SFrederik Gossen   void markEquivalent(Value lhsResult, Value rhsResult) {
79631fc47e3SFrederik Gossen     auto insertion = equivalentValues.insert({lhsResult, rhsResult});
79731fc47e3SFrederik Gossen     // Make sure that the value was not already marked equivalent to some other
79831fc47e3SFrederik Gossen     // value.
79931fc47e3SFrederik Gossen     (void)insertion;
80031fc47e3SFrederik Gossen     assert(insertion.first->second == rhsResult &&
80131fc47e3SFrederik Gossen            "inconsistent OperationEquivalence state");
80231fc47e3SFrederik Gossen   }
80331fc47e3SFrederik Gossen };
80431fc47e3SFrederik Gossen 
80531fc47e3SFrederik Gossen /*static*/ bool
80631fc47e3SFrederik Gossen OperationEquivalence::isRegionEquivalentTo(Region *lhs, Region *rhs,
80731fc47e3SFrederik Gossen                                            OperationEquivalence::Flags flags) {
80831fc47e3SFrederik Gossen   ValueEquivalenceCache cache;
80931fc47e3SFrederik Gossen   return isRegionEquivalentTo(
81031fc47e3SFrederik Gossen       lhs, rhs,
81131fc47e3SFrederik Gossen       [&](Value lhsValue, Value rhsValue) -> LogicalResult {
81231fc47e3SFrederik Gossen         return cache.checkEquivalent(lhsValue, rhsValue);
81331fc47e3SFrederik Gossen       },
81431fc47e3SFrederik Gossen       [&](Value lhsResult, Value rhsResult) {
81531fc47e3SFrederik Gossen         cache.markEquivalent(lhsResult, rhsResult);
81631fc47e3SFrederik Gossen       },
817ee2deb4cSJacques Pienaar       flags,
818ee2deb4cSJacques Pienaar       [&](ValueRange lhs, ValueRange rhs) -> LogicalResult {
819ee2deb4cSJacques Pienaar         return cache.checkCommutativeEquivalent(lhs, rhs);
820ee2deb4cSJacques Pienaar       });
82131fc47e3SFrederik Gossen }
82231fc47e3SFrederik Gossen 
82331fc47e3SFrederik Gossen /*static*/ bool OperationEquivalence::isEquivalentTo(
8240be5d1a9SMehdi Amini     Operation *lhs, Operation *rhs,
825c864288dSMatthias Springer     function_ref<LogicalResult(Value, Value)> checkEquivalent,
826ee2deb4cSJacques Pienaar     function_ref<void(Value, Value)> markEquivalent, Flags flags,
827ee2deb4cSJacques Pienaar     function_ref<LogicalResult(ValueRange, ValueRange)>
828ee2deb4cSJacques Pienaar         checkCommutativeEquivalent) {
829df00e466SRiver Riddle   if (lhs == rhs)
830df00e466SRiver Riddle     return true;
831df00e466SRiver Riddle 
832c864288dSMatthias Springer   // 1. Compare the operation properties.
8330be5d1a9SMehdi Amini   if (lhs->getName() != rhs->getName() ||
834b336ab42SOleksandr "Alex" Zinenko       lhs->getRawDictionaryAttrs() != rhs->getRawDictionaryAttrs() ||
8350be5d1a9SMehdi Amini       lhs->getNumRegions() != rhs->getNumRegions() ||
8360be5d1a9SMehdi Amini       lhs->getNumSuccessors() != rhs->getNumSuccessors() ||
8370be5d1a9SMehdi Amini       lhs->getNumOperands() != rhs->getNumOperands() ||
838bbe5bf17SMehdi Amini       lhs->getNumResults() != rhs->getNumResults() ||
839ed8bd717SDudeldu       !lhs->getName().compareOpProperties(lhs->getPropertiesStorage(),
840ed8bd717SDudeldu                                           rhs->getPropertiesStorage()))
841df00e466SRiver Riddle     return false;
8420be5d1a9SMehdi Amini   if (!(flags & IgnoreLocations) && lhs->getLoc() != rhs->getLoc())
843df00e466SRiver Riddle     return false;
8440be5d1a9SMehdi Amini 
845c864288dSMatthias Springer   // 2. Compare operands.
846ee2deb4cSJacques Pienaar   if (checkCommutativeEquivalent &&
847ee2deb4cSJacques Pienaar       lhs->hasTrait<mlir::OpTrait::IsCommutative>()) {
848ee2deb4cSJacques Pienaar     auto lhsRange = lhs->getOperands();
849ee2deb4cSJacques Pienaar     auto rhsRange = rhs->getOperands();
850ee2deb4cSJacques Pienaar     if (failed(checkCommutativeEquivalent(lhsRange, rhsRange)))
851ee2deb4cSJacques Pienaar       return false;
852ee2deb4cSJacques Pienaar   } else {
853ee2deb4cSJacques Pienaar     // Check pair wise for equivalence.
8542109587cStomnatan     for (auto operandPair : llvm::zip(lhs->getOperands(), rhs->getOperands())) {
8550be5d1a9SMehdi Amini       Value curArg = std::get<0>(operandPair);
8560be5d1a9SMehdi Amini       Value otherArg = std::get<1>(operandPair);
857c864288dSMatthias Springer       if (curArg == otherArg)
858c864288dSMatthias Springer         continue;
8590be5d1a9SMehdi Amini       if (curArg.getType() != otherArg.getType())
860df00e466SRiver Riddle         return false;
861c864288dSMatthias Springer       if (failed(checkEquivalent(curArg, otherArg)))
862df00e466SRiver Riddle         return false;
8630be5d1a9SMehdi Amini     }
864ee2deb4cSJacques Pienaar   }
865c864288dSMatthias Springer 
866c864288dSMatthias Springer   // 3. Compare result types and mark results as equivalent.
867c864288dSMatthias Springer   for (auto resultPair : llvm::zip(lhs->getResults(), rhs->getResults())) {
868c864288dSMatthias Springer     Value curArg = std::get<0>(resultPair);
869c864288dSMatthias Springer     Value otherArg = std::get<1>(resultPair);
870c864288dSMatthias Springer     if (curArg.getType() != otherArg.getType())
8710be5d1a9SMehdi Amini       return false;
872c864288dSMatthias Springer     if (markEquivalent)
873c864288dSMatthias Springer       markEquivalent(curArg, otherArg);
874c864288dSMatthias Springer   }
875c864288dSMatthias Springer 
876c864288dSMatthias Springer   // 4. Compare regions.
8770be5d1a9SMehdi Amini   for (auto regionPair : llvm::zip(lhs->getRegions(), rhs->getRegions()))
8780be5d1a9SMehdi Amini     if (!isRegionEquivalentTo(&std::get<0>(regionPair),
879c864288dSMatthias Springer                               &std::get<1>(regionPair), checkEquivalent,
880c864288dSMatthias Springer                               markEquivalent, flags))
8810be5d1a9SMehdi Amini       return false;
882c864288dSMatthias Springer 
8830be5d1a9SMehdi Amini   return true;
884df00e466SRiver Riddle }
8852e36dadbSIvan Butygin 
88631fc47e3SFrederik Gossen /*static*/ bool OperationEquivalence::isEquivalentTo(Operation *lhs,
88731fc47e3SFrederik Gossen                                                      Operation *rhs,
888c864288dSMatthias Springer                                                      Flags flags) {
88931fc47e3SFrederik Gossen   ValueEquivalenceCache cache;
89031fc47e3SFrederik Gossen   return OperationEquivalence::isEquivalentTo(
89131fc47e3SFrederik Gossen       lhs, rhs,
89231fc47e3SFrederik Gossen       [&](Value lhsValue, Value rhsValue) -> LogicalResult {
89331fc47e3SFrederik Gossen         return cache.checkEquivalent(lhsValue, rhsValue);
89431fc47e3SFrederik Gossen       },
89531fc47e3SFrederik Gossen       [&](Value lhsResult, Value rhsResult) {
89631fc47e3SFrederik Gossen         cache.markEquivalent(lhsResult, rhsResult);
89731fc47e3SFrederik Gossen       },
898ee2deb4cSJacques Pienaar       flags,
899ee2deb4cSJacques Pienaar       [&](ValueRange lhs, ValueRange rhs) -> LogicalResult {
900ee2deb4cSJacques Pienaar         return cache.checkCommutativeEquivalent(lhs, rhs);
901ee2deb4cSJacques Pienaar       });
902c864288dSMatthias Springer }
903c864288dSMatthias Springer 
9042e36dadbSIvan Butygin //===----------------------------------------------------------------------===//
9052e36dadbSIvan Butygin // OperationFingerPrint
9062e36dadbSIvan Butygin //===----------------------------------------------------------------------===//
9072e36dadbSIvan Butygin 
9082e36dadbSIvan Butygin template <typename T>
9092e36dadbSIvan Butygin static void addDataToHash(llvm::SHA1 &hasher, const T &data) {
9102e36dadbSIvan Butygin   hasher.update(
9112e36dadbSIvan Butygin       ArrayRef<uint8_t>(reinterpret_cast<const uint8_t *>(&data), sizeof(T)));
9122e36dadbSIvan Butygin }
9132e36dadbSIvan Butygin 
9145fdf8c6fSMatthias Springer OperationFingerPrint::OperationFingerPrint(Operation *topOp,
9155fdf8c6fSMatthias Springer                                            bool includeNested) {
9162e36dadbSIvan Butygin   llvm::SHA1 hasher;
9172e36dadbSIvan Butygin 
9185fdf8c6fSMatthias Springer   // Helper function that hashes an operation based on its mutable bits:
9195fdf8c6fSMatthias Springer   auto addOperationToHash = [&](Operation *op) {
9202e36dadbSIvan Butygin     //   - Operation pointer
9212e36dadbSIvan Butygin     addDataToHash(hasher, op);
922350b4d05SMatthias Springer     //   - Parent operation pointer (to take into account the nesting structure)
923350b4d05SMatthias Springer     if (op != topOp)
924350b4d05SMatthias Springer       addDataToHash(hasher, op->getParentOp());
9252e36dadbSIvan Butygin     //   - Attributes
926b336ab42SOleksandr "Alex" Zinenko     addDataToHash(hasher, op->getRawDictionaryAttrs());
927bbe5bf17SMehdi Amini     //   - Properties
928bbe5bf17SMehdi Amini     addDataToHash(hasher, op->hashProperties());
9292e36dadbSIvan Butygin     //   - Blocks in Regions
9302e36dadbSIvan Butygin     for (Region &region : op->getRegions()) {
9312e36dadbSIvan Butygin       for (Block &block : region) {
9322e36dadbSIvan Butygin         addDataToHash(hasher, &block);
9332e36dadbSIvan Butygin         for (BlockArgument arg : block.getArguments())
9342e36dadbSIvan Butygin           addDataToHash(hasher, arg);
9352e36dadbSIvan Butygin       }
9362e36dadbSIvan Butygin     }
9372e36dadbSIvan Butygin     //   - Location
9382e36dadbSIvan Butygin     addDataToHash(hasher, op->getLoc().getAsOpaquePointer());
9392e36dadbSIvan Butygin     //   - Operands
9402e36dadbSIvan Butygin     for (Value operand : op->getOperands())
9412e36dadbSIvan Butygin       addDataToHash(hasher, operand);
9422e36dadbSIvan Butygin     //   - Successors
9432e36dadbSIvan Butygin     for (unsigned i = 0, e = op->getNumSuccessors(); i != e; ++i)
9442e36dadbSIvan Butygin       addDataToHash(hasher, op->getSuccessor(i));
945ae9288d4SMatthias Springer     //   - Result types
946ae9288d4SMatthias Springer     for (Type t : op->getResultTypes())
947ae9288d4SMatthias Springer       addDataToHash(hasher, t);
9485fdf8c6fSMatthias Springer   };
9495fdf8c6fSMatthias Springer 
9505fdf8c6fSMatthias Springer   if (includeNested)
9515fdf8c6fSMatthias Springer     topOp->walk(addOperationToHash);
9525fdf8c6fSMatthias Springer   else
9535fdf8c6fSMatthias Springer     addOperationToHash(topOp);
9545fdf8c6fSMatthias Springer 
9552e36dadbSIvan Butygin   hash = hasher.result();
9562e36dadbSIvan Butygin }
957