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> &®ion) { 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> ®ion : 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 ®ion : 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