1 //===- AffineMap.cpp - MLIR Affine Map Classes ----------------------------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "mlir/IR/AffineMap.h" 10 #include "AffineMapDetail.h" 11 #include "mlir/IR/BuiltinAttributes.h" 12 #include "mlir/IR/BuiltinTypes.h" 13 #include "mlir/Support/LogicalResult.h" 14 #include "mlir/Support/MathExtras.h" 15 #include "llvm/ADT/SmallSet.h" 16 #include "llvm/ADT/StringRef.h" 17 #include "llvm/Support/raw_ostream.h" 18 19 using namespace mlir; 20 21 namespace { 22 23 // AffineExprConstantFolder evaluates an affine expression using constant 24 // operands passed in 'operandConsts'. Returns an IntegerAttr attribute 25 // representing the constant value of the affine expression evaluated on 26 // constant 'operandConsts', or nullptr if it can't be folded. 27 class AffineExprConstantFolder { 28 public: 29 AffineExprConstantFolder(unsigned numDims, ArrayRef<Attribute> operandConsts) 30 : numDims(numDims), operandConsts(operandConsts) {} 31 32 /// Attempt to constant fold the specified affine expr, or return null on 33 /// failure. 34 IntegerAttr constantFold(AffineExpr expr) { 35 if (auto result = constantFoldImpl(expr)) 36 return IntegerAttr::get(IndexType::get(expr.getContext()), *result); 37 return nullptr; 38 } 39 40 private: 41 Optional<int64_t> constantFoldImpl(AffineExpr expr) { 42 switch (expr.getKind()) { 43 case AffineExprKind::Add: 44 return constantFoldBinExpr( 45 expr, [](int64_t lhs, int64_t rhs) { return lhs + rhs; }); 46 case AffineExprKind::Mul: 47 return constantFoldBinExpr( 48 expr, [](int64_t lhs, int64_t rhs) { return lhs * rhs; }); 49 case AffineExprKind::Mod: 50 return constantFoldBinExpr( 51 expr, [](int64_t lhs, int64_t rhs) { return mod(lhs, rhs); }); 52 case AffineExprKind::FloorDiv: 53 return constantFoldBinExpr( 54 expr, [](int64_t lhs, int64_t rhs) { return floorDiv(lhs, rhs); }); 55 case AffineExprKind::CeilDiv: 56 return constantFoldBinExpr( 57 expr, [](int64_t lhs, int64_t rhs) { return ceilDiv(lhs, rhs); }); 58 case AffineExprKind::Constant: 59 return expr.cast<AffineConstantExpr>().getValue(); 60 case AffineExprKind::DimId: 61 if (auto attr = operandConsts[expr.cast<AffineDimExpr>().getPosition()] 62 .dyn_cast_or_null<IntegerAttr>()) 63 return attr.getInt(); 64 return llvm::None; 65 case AffineExprKind::SymbolId: 66 if (auto attr = operandConsts[numDims + 67 expr.cast<AffineSymbolExpr>().getPosition()] 68 .dyn_cast_or_null<IntegerAttr>()) 69 return attr.getInt(); 70 return llvm::None; 71 } 72 llvm_unreachable("Unknown AffineExpr"); 73 } 74 75 // TODO: Change these to operate on APInts too. 76 Optional<int64_t> constantFoldBinExpr(AffineExpr expr, 77 int64_t (*op)(int64_t, int64_t)) { 78 auto binOpExpr = expr.cast<AffineBinaryOpExpr>(); 79 if (auto lhs = constantFoldImpl(binOpExpr.getLHS())) 80 if (auto rhs = constantFoldImpl(binOpExpr.getRHS())) 81 return op(*lhs, *rhs); 82 return llvm::None; 83 } 84 85 // The number of dimension operands in AffineMap containing this expression. 86 unsigned numDims; 87 // The constant valued operands used to evaluate this AffineExpr. 88 ArrayRef<Attribute> operandConsts; 89 }; 90 91 } // end anonymous namespace 92 93 /// Returns a single constant result affine map. 94 AffineMap AffineMap::getConstantMap(int64_t val, MLIRContext *context) { 95 return get(/*dimCount=*/0, /*symbolCount=*/0, 96 {getAffineConstantExpr(val, context)}); 97 } 98 99 /// Returns an identity affine map (d0, ..., dn) -> (dp, ..., dn) on the most 100 /// minor dimensions. 101 AffineMap AffineMap::getMinorIdentityMap(unsigned dims, unsigned results, 102 MLIRContext *context) { 103 assert(dims >= results && "Dimension mismatch"); 104 auto id = AffineMap::getMultiDimIdentityMap(dims, context); 105 return AffineMap::get(dims, 0, id.getResults().take_back(results), context); 106 } 107 108 bool AffineMap::isMinorIdentity() const { 109 return *this == 110 getMinorIdentityMap(getNumDims(), getNumResults(), getContext()); 111 } 112 113 /// Returns an AffineMap representing a permutation. 114 AffineMap AffineMap::getPermutationMap(ArrayRef<unsigned> permutation, 115 MLIRContext *context) { 116 assert(!permutation.empty() && 117 "Cannot create permutation map from empty permutation vector"); 118 SmallVector<AffineExpr, 4> affExprs; 119 for (auto index : permutation) 120 affExprs.push_back(getAffineDimExpr(index, context)); 121 auto m = std::max_element(permutation.begin(), permutation.end()); 122 auto permutationMap = AffineMap::get(*m + 1, 0, affExprs, context); 123 assert(permutationMap.isPermutation() && "Invalid permutation vector"); 124 return permutationMap; 125 } 126 127 template <typename AffineExprContainer> 128 static void getMaxDimAndSymbol(ArrayRef<AffineExprContainer> exprsList, 129 int64_t &maxDim, int64_t &maxSym) { 130 for (const auto &exprs : exprsList) { 131 for (auto expr : exprs) { 132 expr.walk([&maxDim, &maxSym](AffineExpr e) { 133 if (auto d = e.dyn_cast<AffineDimExpr>()) 134 maxDim = std::max(maxDim, static_cast<int64_t>(d.getPosition())); 135 if (auto s = e.dyn_cast<AffineSymbolExpr>()) 136 maxSym = std::max(maxSym, static_cast<int64_t>(s.getPosition())); 137 }); 138 } 139 } 140 } 141 142 template <typename AffineExprContainer> 143 static SmallVector<AffineMap, 4> 144 inferFromExprList(ArrayRef<AffineExprContainer> exprsList) { 145 assert(!exprsList.empty()); 146 assert(!exprsList[0].empty()); 147 auto context = exprsList[0][0].getContext(); 148 int64_t maxDim = -1, maxSym = -1; 149 getMaxDimAndSymbol(exprsList, maxDim, maxSym); 150 SmallVector<AffineMap, 4> maps; 151 maps.reserve(exprsList.size()); 152 for (const auto &exprs : exprsList) 153 maps.push_back(AffineMap::get(/*dimCount=*/maxDim + 1, 154 /*symbolCount=*/maxSym + 1, exprs, context)); 155 return maps; 156 } 157 158 SmallVector<AffineMap, 4> 159 AffineMap::inferFromExprList(ArrayRef<ArrayRef<AffineExpr>> exprsList) { 160 return ::inferFromExprList(exprsList); 161 } 162 163 SmallVector<AffineMap, 4> 164 AffineMap::inferFromExprList(ArrayRef<SmallVector<AffineExpr, 4>> exprsList) { 165 return ::inferFromExprList(exprsList); 166 } 167 168 AffineMap AffineMap::getMultiDimIdentityMap(unsigned numDims, 169 MLIRContext *context) { 170 SmallVector<AffineExpr, 4> dimExprs; 171 dimExprs.reserve(numDims); 172 for (unsigned i = 0; i < numDims; ++i) 173 dimExprs.push_back(mlir::getAffineDimExpr(i, context)); 174 return get(/*dimCount=*/numDims, /*symbolCount=*/0, dimExprs, context); 175 } 176 177 MLIRContext *AffineMap::getContext() const { return map->context; } 178 179 bool AffineMap::isIdentity() const { 180 if (getNumDims() != getNumResults()) 181 return false; 182 ArrayRef<AffineExpr> results = getResults(); 183 for (unsigned i = 0, numDims = getNumDims(); i < numDims; ++i) { 184 auto expr = results[i].dyn_cast<AffineDimExpr>(); 185 if (!expr || expr.getPosition() != i) 186 return false; 187 } 188 return true; 189 } 190 191 bool AffineMap::isEmpty() const { 192 return getNumDims() == 0 && getNumSymbols() == 0 && getNumResults() == 0; 193 } 194 195 bool AffineMap::isSingleConstant() const { 196 return getNumResults() == 1 && getResult(0).isa<AffineConstantExpr>(); 197 } 198 199 int64_t AffineMap::getSingleConstantResult() const { 200 assert(isSingleConstant() && "map must have a single constant result"); 201 return getResult(0).cast<AffineConstantExpr>().getValue(); 202 } 203 204 unsigned AffineMap::getNumDims() const { 205 assert(map && "uninitialized map storage"); 206 return map->numDims; 207 } 208 unsigned AffineMap::getNumSymbols() const { 209 assert(map && "uninitialized map storage"); 210 return map->numSymbols; 211 } 212 unsigned AffineMap::getNumResults() const { 213 assert(map && "uninitialized map storage"); 214 return map->results.size(); 215 } 216 unsigned AffineMap::getNumInputs() const { 217 assert(map && "uninitialized map storage"); 218 return map->numDims + map->numSymbols; 219 } 220 221 ArrayRef<AffineExpr> AffineMap::getResults() const { 222 assert(map && "uninitialized map storage"); 223 return map->results; 224 } 225 AffineExpr AffineMap::getResult(unsigned idx) const { 226 assert(map && "uninitialized map storage"); 227 return map->results[idx]; 228 } 229 230 unsigned AffineMap::getDimPosition(unsigned idx) const { 231 return getResult(idx).cast<AffineDimExpr>().getPosition(); 232 } 233 234 /// Folds the results of the application of an affine map on the provided 235 /// operands to a constant if possible. Returns false if the folding happens, 236 /// true otherwise. 237 LogicalResult 238 AffineMap::constantFold(ArrayRef<Attribute> operandConstants, 239 SmallVectorImpl<Attribute> &results) const { 240 // Attempt partial folding. 241 SmallVector<int64_t, 2> integers; 242 partialConstantFold(operandConstants, &integers); 243 244 // If all expressions folded to a constant, populate results with attributes 245 // containing those constants. 246 if (integers.empty()) 247 return failure(); 248 249 auto range = llvm::map_range(integers, [this](int64_t i) { 250 return IntegerAttr::get(IndexType::get(getContext()), i); 251 }); 252 results.append(range.begin(), range.end()); 253 return success(); 254 } 255 256 AffineMap 257 AffineMap::partialConstantFold(ArrayRef<Attribute> operandConstants, 258 SmallVectorImpl<int64_t> *results) const { 259 assert(getNumInputs() == operandConstants.size()); 260 261 // Fold each of the result expressions. 262 AffineExprConstantFolder exprFolder(getNumDims(), operandConstants); 263 SmallVector<AffineExpr, 4> exprs; 264 exprs.reserve(getNumResults()); 265 266 for (auto expr : getResults()) { 267 auto folded = exprFolder.constantFold(expr); 268 // If did not fold to a constant, keep the original expression, and clear 269 // the integer results vector. 270 if (folded) { 271 exprs.push_back( 272 getAffineConstantExpr(folded.getInt(), folded.getContext())); 273 if (results) 274 results->push_back(folded.getInt()); 275 } else { 276 exprs.push_back(expr); 277 if (results) { 278 results->clear(); 279 results = nullptr; 280 } 281 } 282 } 283 284 return get(getNumDims(), getNumSymbols(), exprs, getContext()); 285 } 286 287 /// Walk all of the AffineExpr's in this mapping. Each node in an expression 288 /// tree is visited in postorder. 289 void AffineMap::walkExprs(std::function<void(AffineExpr)> callback) const { 290 for (auto expr : getResults()) 291 expr.walk(callback); 292 } 293 294 /// This method substitutes any uses of dimensions and symbols (e.g. 295 /// dim#0 with dimReplacements[0]) in subexpressions and returns the modified 296 /// expression mapping. Because this can be used to eliminate dims and 297 /// symbols, the client needs to specify the number of dims and symbols in 298 /// the result. The returned map always has the same number of results. 299 AffineMap AffineMap::replaceDimsAndSymbols(ArrayRef<AffineExpr> dimReplacements, 300 ArrayRef<AffineExpr> symReplacements, 301 unsigned numResultDims, 302 unsigned numResultSyms) const { 303 SmallVector<AffineExpr, 8> results; 304 results.reserve(getNumResults()); 305 for (auto expr : getResults()) 306 results.push_back( 307 expr.replaceDimsAndSymbols(dimReplacements, symReplacements)); 308 return get(numResultDims, numResultSyms, results, getContext()); 309 } 310 311 /// Sparse replace method. Apply AffineExpr::replace(`expr`, `replacement`) to 312 /// each of the results and return a new AffineMap with the new results and 313 /// with the specified number of dims and symbols. 314 AffineMap AffineMap::replace(AffineExpr expr, AffineExpr replacement, 315 unsigned numResultDims, 316 unsigned numResultSyms) const { 317 SmallVector<AffineExpr, 4> newResults; 318 newResults.reserve(getNumResults()); 319 for (AffineExpr e : getResults()) 320 newResults.push_back(e.replace(expr, replacement)); 321 return AffineMap::get(numResultDims, numResultSyms, newResults, getContext()); 322 } 323 324 /// Sparse replace method. Apply AffineExpr::replace(`map`) to each of the 325 /// results and return a new AffineMap with the new results and with the 326 /// specified number of dims and symbols. 327 AffineMap AffineMap::replace(const DenseMap<AffineExpr, AffineExpr> &map, 328 unsigned numResultDims, 329 unsigned numResultSyms) const { 330 SmallVector<AffineExpr, 4> newResults; 331 newResults.reserve(getNumResults()); 332 for (AffineExpr e : getResults()) 333 newResults.push_back(e.replace(map)); 334 return AffineMap::get(numResultDims, numResultSyms, newResults, getContext()); 335 } 336 337 AffineMap AffineMap::compose(AffineMap map) const { 338 assert(getNumDims() == map.getNumResults() && "Number of results mismatch"); 339 // Prepare `map` by concatenating the symbols and rewriting its exprs. 340 unsigned numDims = map.getNumDims(); 341 unsigned numSymbolsThisMap = getNumSymbols(); 342 unsigned numSymbols = numSymbolsThisMap + map.getNumSymbols(); 343 SmallVector<AffineExpr, 8> newDims(numDims); 344 for (unsigned idx = 0; idx < numDims; ++idx) { 345 newDims[idx] = getAffineDimExpr(idx, getContext()); 346 } 347 SmallVector<AffineExpr, 8> newSymbols(numSymbols - numSymbolsThisMap); 348 for (unsigned idx = numSymbolsThisMap; idx < numSymbols; ++idx) { 349 newSymbols[idx - numSymbolsThisMap] = 350 getAffineSymbolExpr(idx, getContext()); 351 } 352 auto newMap = 353 map.replaceDimsAndSymbols(newDims, newSymbols, numDims, numSymbols); 354 SmallVector<AffineExpr, 8> exprs; 355 exprs.reserve(getResults().size()); 356 for (auto expr : getResults()) 357 exprs.push_back(expr.compose(newMap)); 358 return AffineMap::get(numDims, numSymbols, exprs, map.getContext()); 359 } 360 361 SmallVector<int64_t, 4> AffineMap::compose(ArrayRef<int64_t> values) const { 362 assert(getNumSymbols() == 0 && "Expected symbol-less map"); 363 SmallVector<AffineExpr, 4> exprs; 364 exprs.reserve(values.size()); 365 MLIRContext *ctx = getContext(); 366 for (auto v : values) 367 exprs.push_back(getAffineConstantExpr(v, ctx)); 368 auto resMap = compose(AffineMap::get(0, 0, exprs, ctx)); 369 SmallVector<int64_t, 4> res; 370 res.reserve(resMap.getNumResults()); 371 for (auto e : resMap.getResults()) 372 res.push_back(e.cast<AffineConstantExpr>().getValue()); 373 return res; 374 } 375 376 bool AffineMap::isProjectedPermutation() const { 377 if (getNumSymbols() > 0) 378 return false; 379 SmallVector<bool, 8> seen(getNumInputs(), false); 380 for (auto expr : getResults()) { 381 if (auto dim = expr.dyn_cast<AffineDimExpr>()) { 382 if (seen[dim.getPosition()]) 383 return false; 384 seen[dim.getPosition()] = true; 385 continue; 386 } 387 return false; 388 } 389 return true; 390 } 391 392 bool AffineMap::isPermutation() const { 393 if (getNumDims() != getNumResults()) 394 return false; 395 return isProjectedPermutation(); 396 } 397 398 AffineMap AffineMap::getSubMap(ArrayRef<unsigned> resultPos) const { 399 SmallVector<AffineExpr, 4> exprs; 400 exprs.reserve(resultPos.size()); 401 for (auto idx : resultPos) 402 exprs.push_back(getResult(idx)); 403 return AffineMap::get(getNumDims(), getNumSymbols(), exprs, getContext()); 404 } 405 406 AffineMap AffineMap::getMajorSubMap(unsigned numResults) const { 407 if (numResults == 0) 408 return AffineMap(); 409 if (numResults > getNumResults()) 410 return *this; 411 return getSubMap(llvm::to_vector<4>(llvm::seq<unsigned>(0, numResults))); 412 } 413 414 AffineMap AffineMap::getMinorSubMap(unsigned numResults) const { 415 if (numResults == 0) 416 return AffineMap(); 417 if (numResults > getNumResults()) 418 return *this; 419 return getSubMap(llvm::to_vector<4>( 420 llvm::seq<unsigned>(getNumResults() - numResults, getNumResults()))); 421 } 422 423 AffineMap mlir::compressDims(AffineMap map, 424 const llvm::SmallDenseSet<unsigned> &unusedDims) { 425 unsigned numDims = 0; 426 SmallVector<AffineExpr> dimReplacements; 427 dimReplacements.reserve(map.getNumDims()); 428 MLIRContext *context = map.getContext(); 429 for (unsigned dim = 0, e = map.getNumDims(); dim < e; ++dim) { 430 if (unusedDims.contains(dim)) 431 dimReplacements.push_back(getAffineConstantExpr(0, context)); 432 else 433 dimReplacements.push_back(getAffineDimExpr(numDims++, context)); 434 } 435 SmallVector<AffineExpr> resultExprs; 436 resultExprs.reserve(map.getNumResults()); 437 for (auto e : map.getResults()) 438 resultExprs.push_back(e.replaceDims(dimReplacements)); 439 return AffineMap::get(numDims, map.getNumSymbols(), resultExprs, context); 440 } 441 442 AffineMap mlir::compressUnusedDims(AffineMap map) { 443 llvm::SmallDenseSet<unsigned> usedDims; 444 map.walkExprs([&](AffineExpr expr) { 445 if (auto dimExpr = expr.dyn_cast<AffineDimExpr>()) 446 usedDims.insert(dimExpr.getPosition()); 447 }); 448 llvm::SmallDenseSet<unsigned> unusedDims; 449 for (unsigned d = 0, e = map.getNumDims(); d != e; ++d) 450 if (!usedDims.contains(d)) 451 unusedDims.insert(d); 452 return compressDims(map, unusedDims); 453 } 454 455 AffineMap 456 mlir::compressSymbols(AffineMap map, 457 const llvm::SmallDenseSet<unsigned> &unusedSymbols) { 458 unsigned numSymbols = 0; 459 SmallVector<AffineExpr> symReplacements; 460 symReplacements.reserve(map.getNumSymbols()); 461 MLIRContext *context = map.getContext(); 462 for (unsigned sym = 0, e = map.getNumSymbols(); sym < e; ++sym) { 463 if (unusedSymbols.contains(sym)) 464 symReplacements.push_back(getAffineConstantExpr(0, context)); 465 else 466 symReplacements.push_back(getAffineSymbolExpr(numSymbols++, context)); 467 } 468 SmallVector<AffineExpr> resultExprs; 469 resultExprs.reserve(map.getNumResults()); 470 for (auto e : map.getResults()) 471 resultExprs.push_back(e.replaceSymbols(symReplacements)); 472 return AffineMap::get(map.getNumDims(), numSymbols, resultExprs, context); 473 } 474 475 AffineMap mlir::compressUnusedSymbols(AffineMap map) { 476 llvm::SmallDenseSet<unsigned> usedSymbols; 477 map.walkExprs([&](AffineExpr expr) { 478 if (auto symExpr = expr.dyn_cast<AffineSymbolExpr>()) 479 usedSymbols.insert(symExpr.getPosition()); 480 }); 481 llvm::SmallDenseSet<unsigned> unusedSymbols; 482 for (unsigned d = 0, e = map.getNumSymbols(); d != e; ++d) 483 if (!usedSymbols.contains(d)) 484 unusedSymbols.insert(d); 485 return compressSymbols(map, unusedSymbols); 486 } 487 488 AffineMap mlir::simplifyAffineMap(AffineMap map) { 489 SmallVector<AffineExpr, 8> exprs; 490 for (auto e : map.getResults()) { 491 exprs.push_back( 492 simplifyAffineExpr(e, map.getNumDims(), map.getNumSymbols())); 493 } 494 return AffineMap::get(map.getNumDims(), map.getNumSymbols(), exprs, 495 map.getContext()); 496 } 497 498 AffineMap mlir::removeDuplicateExprs(AffineMap map) { 499 auto results = map.getResults(); 500 SmallVector<AffineExpr, 4> uniqueExprs(results.begin(), results.end()); 501 uniqueExprs.erase(std::unique(uniqueExprs.begin(), uniqueExprs.end()), 502 uniqueExprs.end()); 503 return AffineMap::get(map.getNumDims(), map.getNumSymbols(), uniqueExprs, 504 map.getContext()); 505 } 506 507 AffineMap mlir::inversePermutation(AffineMap map) { 508 if (map.isEmpty()) 509 return map; 510 assert(map.getNumSymbols() == 0 && "expected map without symbols"); 511 SmallVector<AffineExpr, 4> exprs(map.getNumDims()); 512 for (auto en : llvm::enumerate(map.getResults())) { 513 auto expr = en.value(); 514 // Skip non-permutations. 515 if (auto d = expr.dyn_cast<AffineDimExpr>()) { 516 if (exprs[d.getPosition()]) 517 continue; 518 exprs[d.getPosition()] = getAffineDimExpr(en.index(), d.getContext()); 519 } 520 } 521 SmallVector<AffineExpr, 4> seenExprs; 522 seenExprs.reserve(map.getNumDims()); 523 for (auto expr : exprs) 524 if (expr) 525 seenExprs.push_back(expr); 526 if (seenExprs.size() != map.getNumInputs()) 527 return AffineMap(); 528 return AffineMap::get(map.getNumResults(), 0, seenExprs, map.getContext()); 529 } 530 531 AffineMap mlir::concatAffineMaps(ArrayRef<AffineMap> maps) { 532 unsigned numResults = 0, numDims = 0, numSymbols = 0; 533 for (auto m : maps) 534 numResults += m.getNumResults(); 535 SmallVector<AffineExpr, 8> results; 536 results.reserve(numResults); 537 for (auto m : maps) { 538 for (auto res : m.getResults()) 539 results.push_back(res.shiftSymbols(m.getNumSymbols(), numSymbols)); 540 541 numSymbols += m.getNumSymbols(); 542 numDims = std::max(m.getNumDims(), numDims); 543 } 544 return AffineMap::get(numDims, numSymbols, results, 545 maps.front().getContext()); 546 } 547 548 AffineMap 549 mlir::getProjectedMap(AffineMap map, 550 const llvm::SmallDenseSet<unsigned> &unusedDims) { 551 return compressUnusedSymbols(compressDims(map, unusedDims)); 552 } 553 554 //===----------------------------------------------------------------------===// 555 // MutableAffineMap. 556 //===----------------------------------------------------------------------===// 557 558 MutableAffineMap::MutableAffineMap(AffineMap map) 559 : numDims(map.getNumDims()), numSymbols(map.getNumSymbols()), 560 context(map.getContext()) { 561 for (auto result : map.getResults()) 562 results.push_back(result); 563 } 564 565 void MutableAffineMap::reset(AffineMap map) { 566 results.clear(); 567 numDims = map.getNumDims(); 568 numSymbols = map.getNumSymbols(); 569 context = map.getContext(); 570 for (auto result : map.getResults()) 571 results.push_back(result); 572 } 573 574 bool MutableAffineMap::isMultipleOf(unsigned idx, int64_t factor) const { 575 if (results[idx].isMultipleOf(factor)) 576 return true; 577 578 // TODO: use simplifyAffineExpr and FlatAffineConstraints to 579 // complete this (for a more powerful analysis). 580 return false; 581 } 582 583 // Simplifies the result affine expressions of this map. The expressions have to 584 // be pure for the simplification implemented. 585 void MutableAffineMap::simplify() { 586 // Simplify each of the results if possible. 587 // TODO: functional-style map 588 for (unsigned i = 0, e = getNumResults(); i < e; i++) { 589 results[i] = simplifyAffineExpr(getResult(i), numDims, numSymbols); 590 } 591 } 592 593 AffineMap MutableAffineMap::getAffineMap() const { 594 return AffineMap::get(numDims, numSymbols, results, context); 595 } 596