MLIR  20.0.0git
MLIRContext.cpp
Go to the documentation of this file.
1 //===- MLIRContext.cpp - MLIR Type 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/MLIRContext.h"
10 #include "AffineExprDetail.h"
11 #include "AffineMapDetail.h"
12 #include "AttributeDetail.h"
13 #include "IntegerSetDetail.h"
14 #include "TypeDetail.h"
15 #include "mlir/IR/Action.h"
16 #include "mlir/IR/AffineExpr.h"
17 #include "mlir/IR/AffineMap.h"
18 #include "mlir/IR/Attributes.h"
20 #include "mlir/IR/BuiltinDialect.h"
21 #include "mlir/IR/Diagnostics.h"
22 #include "mlir/IR/Dialect.h"
24 #include "mlir/IR/IntegerSet.h"
25 #include "mlir/IR/Location.h"
28 #include "mlir/IR/Types.h"
29 #include "llvm/ADT/DenseMap.h"
30 #include "llvm/ADT/DenseSet.h"
31 #include "llvm/ADT/SmallString.h"
32 #include "llvm/ADT/StringSet.h"
33 #include "llvm/ADT/Twine.h"
34 #include "llvm/Support/Allocator.h"
35 #include "llvm/Support/CommandLine.h"
36 #include "llvm/Support/Compiler.h"
37 #include "llvm/Support/Debug.h"
38 #include "llvm/Support/Mutex.h"
39 #include "llvm/Support/RWMutex.h"
40 #include "llvm/Support/ThreadPool.h"
41 #include "llvm/Support/raw_ostream.h"
42 #include <memory>
43 #include <optional>
44 
45 #define DEBUG_TYPE "mlircontext"
46 
47 using namespace mlir;
48 using namespace mlir::detail;
49 
50 //===----------------------------------------------------------------------===//
51 // MLIRContext CommandLine Options
52 //===----------------------------------------------------------------------===//
53 
54 namespace {
55 /// This struct contains command line options that can be used to initialize
56 /// various bits of an MLIRContext. This uses a struct wrapper to avoid the need
57 /// for global command line options.
58 struct MLIRContextOptions {
59  llvm::cl::opt<bool> disableThreading{
60  "mlir-disable-threading",
61  llvm::cl::desc("Disable multi-threading within MLIR, overrides any "
62  "further call to MLIRContext::enableMultiThreading()")};
63 
64  llvm::cl::opt<bool> printOpOnDiagnostic{
65  "mlir-print-op-on-diagnostic",
66  llvm::cl::desc("When a diagnostic is emitted on an operation, also print "
67  "the operation as an attached note"),
68  llvm::cl::init(true)};
69 
70  llvm::cl::opt<bool> printStackTraceOnDiagnostic{
71  "mlir-print-stacktrace-on-diagnostic",
72  llvm::cl::desc("When a diagnostic is emitted, also print the stack trace "
73  "as an attached note")};
74 };
75 } // namespace
76 
77 static llvm::ManagedStatic<MLIRContextOptions> clOptions;
78 
80 #if LLVM_ENABLE_THREADS != 0
81  return clOptions.isConstructed() && clOptions->disableThreading;
82 #else
83  return true;
84 #endif
85 }
86 
87 /// Register a set of useful command-line options that can be used to configure
88 /// various flags within the MLIRContext. These flags are used when constructing
89 /// an MLIR context for initialization.
91  // Make sure that the options struct has been initialized.
92  *clOptions;
93 }
94 
95 //===----------------------------------------------------------------------===//
96 // Locking Utilities
97 //===----------------------------------------------------------------------===//
98 
99 namespace {
100 /// Utility writer lock that takes a runtime flag that specifies if we really
101 /// need to lock.
102 struct ScopedWriterLock {
103  ScopedWriterLock(llvm::sys::SmartRWMutex<true> &mutexParam, bool shouldLock)
104  : mutex(shouldLock ? &mutexParam : nullptr) {
105  if (mutex)
106  mutex->lock();
107  }
108  ~ScopedWriterLock() {
109  if (mutex)
110  mutex->unlock();
111  }
112  llvm::sys::SmartRWMutex<true> *mutex;
113 };
114 } // namespace
115 
116 //===----------------------------------------------------------------------===//
117 // MLIRContextImpl
118 //===----------------------------------------------------------------------===//
119 
120 namespace mlir {
121 /// This is the implementation of the MLIRContext class, using the pImpl idiom.
122 /// This class is completely private to this file, so everything is public.
124 public:
125  //===--------------------------------------------------------------------===//
126  // Debugging
127  //===--------------------------------------------------------------------===//
128 
129  /// An action handler for handling actions that are dispatched through this
130  /// context.
131  std::function<void(function_ref<void()>, const tracing::Action &)>
133 
134  //===--------------------------------------------------------------------===//
135  // Diagnostics
136  //===--------------------------------------------------------------------===//
138 
139  //===--------------------------------------------------------------------===//
140  // Options
141  //===--------------------------------------------------------------------===//
142 
143  /// In most cases, creating operation in unregistered dialect is not desired
144  /// and indicate a misconfiguration of the compiler. This option enables to
145  /// detect such use cases
146  bool allowUnregisteredDialects = false;
147 
148  /// Enable support for multi-threading within MLIR.
149  bool threadingIsEnabled = true;
150 
151  /// Track if we are currently executing in a threaded execution environment
152  /// (like the pass-manager): this is only a debugging feature to help reducing
153  /// the chances of data races one some context APIs.
154 #ifndef NDEBUG
155  std::atomic<int> multiThreadedExecutionContext{0};
156 #endif
157 
158  /// If the operation should be attached to diagnostics printed via the
159  /// Operation::emit methods.
160  bool printOpOnDiagnostic = true;
161 
162  /// If the current stack trace should be attached when emitting diagnostics.
163  bool printStackTraceOnDiagnostic = false;
164 
165  //===--------------------------------------------------------------------===//
166  // Other
167  //===--------------------------------------------------------------------===//
168 
169  /// This points to the ThreadPool used when processing MLIR tasks in parallel.
170  /// It can't be nullptr when multi-threading is enabled. Otherwise if
171  /// multi-threading is disabled, and the threadpool wasn't externally provided
172  /// using `setThreadPool`, this will be nullptr.
173  llvm::ThreadPoolInterface *threadPool = nullptr;
174 
175  /// In case where the thread pool is owned by the context, this ensures
176  /// destruction with the context.
177  std::unique_ptr<llvm::ThreadPoolInterface> ownedThreadPool;
178 
179  /// An allocator used for AbstractAttribute and AbstractType objects.
180  llvm::BumpPtrAllocator abstractDialectSymbolAllocator;
181 
182  /// This is a mapping from operation name to the operation info describing it.
183  llvm::StringMap<std::unique_ptr<OperationName::Impl>> operations;
184 
185  /// A vector of operation info specifically for registered operations.
187  llvm::StringMap<RegisteredOperationName> registeredOperationsByName;
188 
189  /// This is a sorted container of registered operations for a deterministic
190  /// and efficient `getRegisteredOperations` implementation.
192 
193  /// This is a list of dialects that are created referring to this context.
194  /// The MLIRContext owns the objects. These need to be declared after the
195  /// registered operations to ensure correct destruction order.
198 
199  /// A mutex used when accessing operation information.
200  llvm::sys::SmartRWMutex<true> operationInfoMutex;
201 
202  //===--------------------------------------------------------------------===//
203  // Affine uniquing
204  //===--------------------------------------------------------------------===//
205 
206  // Affine expression, map and integer set uniquing.
208 
209  //===--------------------------------------------------------------------===//
210  // Type uniquing
211  //===--------------------------------------------------------------------===//
212 
215 
216  /// This is a mapping from type name to the abstract type describing it.
217  /// It is used by `AbstractType::lookup` to get an `AbstractType` from a name.
218  /// As this map needs to be populated before `StringAttr` is loaded, we
219  /// cannot use `StringAttr` as the key. The context does not take ownership
220  /// of the key, so the `StringRef` must outlive the context.
222 
223  /// Cached Type Instances.
224  Float4E2M1FNType f4E2M1FNTy;
225  Float6E2M3FNType f6E2M3FNTy;
226  Float6E3M2FNType f6E3M2FNTy;
227  Float8E5M2Type f8E5M2Ty;
228  Float8E4M3Type f8E4M3Ty;
229  Float8E4M3FNType f8E4M3FNTy;
230  Float8E5M2FNUZType f8E5M2FNUZTy;
231  Float8E4M3FNUZType f8E4M3FNUZTy;
232  Float8E4M3B11FNUZType f8E4M3B11FNUZTy;
233  Float8E3M4Type f8E3M4Ty;
234  Float8E8M0FNUType f8E8M0FNUTy;
235  BFloat16Type bf16Ty;
236  Float16Type f16Ty;
237  FloatTF32Type tf32Ty;
238  Float32Type f32Ty;
239  Float64Type f64Ty;
240  Float80Type f80Ty;
241  Float128Type f128Ty;
242  IndexType indexTy;
243  IntegerType int1Ty, int8Ty, int16Ty, int32Ty, int64Ty, int128Ty;
244  NoneType noneType;
245 
246  //===--------------------------------------------------------------------===//
247  // Attribute uniquing
248  //===--------------------------------------------------------------------===//
249 
252 
253  /// This is a mapping from attribute name to the abstract attribute describing
254  /// it. It is used by `AbstractType::lookup` to get an `AbstractType` from a
255  /// name.
256  /// As this map needs to be populated before `StringAttr` is loaded, we
257  /// cannot use `StringAttr` as the key. The context does not take ownership
258  /// of the key, so the `StringRef` must outlive the context.
260 
261  /// Cached Attribute Instances.
262  BoolAttr falseAttr, trueAttr;
263  UnitAttr unitAttr;
264  UnknownLoc unknownLocAttr;
265  DictionaryAttr emptyDictionaryAttr;
266  StringAttr emptyStringAttr;
267 
268  /// Map of string attributes that may reference a dialect, that are awaiting
269  /// that dialect to be loaded.
273 
274  /// A distinct attribute allocator that allocates every time since the
275  /// address of the distinct attribute storage serves as unique identifier. The
276  /// allocator is thread safe and frees the allocated storage after its
277  /// destruction.
279 
280 public:
281  MLIRContextImpl(bool threadingIsEnabled)
282  : threadingIsEnabled(threadingIsEnabled) {
283  if (threadingIsEnabled) {
284  ownedThreadPool = std::make_unique<llvm::DefaultThreadPool>();
285  threadPool = ownedThreadPool.get();
286  }
287  }
289  for (auto typeMapping : registeredTypes)
290  typeMapping.second->~AbstractType();
291  for (auto attrMapping : registeredAttributes)
292  attrMapping.second->~AbstractAttribute();
293  }
294 };
295 } // namespace mlir
296 
298  : MLIRContext(DialectRegistry(), setting) {}
299 
301  : impl(new MLIRContextImpl(setting == Threading::ENABLED &&
303  // Initialize values based on the command line flags if they were provided.
304  if (clOptions.isConstructed()) {
305  printOpOnDiagnostic(clOptions->printOpOnDiagnostic);
306  printStackTraceOnDiagnostic(clOptions->printStackTraceOnDiagnostic);
307  }
308 
309  // Pre-populate the registry.
310  registry.appendTo(impl->dialectsRegistry);
311 
312  // Ensure the builtin dialect is always pre-loaded.
313  getOrLoadDialect<BuiltinDialect>();
314 
315  // Initialize several common attributes and types to avoid the need to lock
316  // the context when accessing them.
317 
318  //// Types.
319  /// Floating-point Types.
320  impl->f4E2M1FNTy = TypeUniquer::get<Float4E2M1FNType>(this);
321  impl->f6E2M3FNTy = TypeUniquer::get<Float6E2M3FNType>(this);
322  impl->f6E3M2FNTy = TypeUniquer::get<Float6E3M2FNType>(this);
323  impl->f8E5M2Ty = TypeUniquer::get<Float8E5M2Type>(this);
324  impl->f8E4M3Ty = TypeUniquer::get<Float8E4M3Type>(this);
325  impl->f8E4M3FNTy = TypeUniquer::get<Float8E4M3FNType>(this);
326  impl->f8E5M2FNUZTy = TypeUniquer::get<Float8E5M2FNUZType>(this);
327  impl->f8E4M3FNUZTy = TypeUniquer::get<Float8E4M3FNUZType>(this);
328  impl->f8E4M3B11FNUZTy = TypeUniquer::get<Float8E4M3B11FNUZType>(this);
329  impl->f8E3M4Ty = TypeUniquer::get<Float8E3M4Type>(this);
330  impl->f8E8M0FNUTy = TypeUniquer::get<Float8E8M0FNUType>(this);
331  impl->bf16Ty = TypeUniquer::get<BFloat16Type>(this);
332  impl->f16Ty = TypeUniquer::get<Float16Type>(this);
333  impl->tf32Ty = TypeUniquer::get<FloatTF32Type>(this);
334  impl->f32Ty = TypeUniquer::get<Float32Type>(this);
335  impl->f64Ty = TypeUniquer::get<Float64Type>(this);
336  impl->f80Ty = TypeUniquer::get<Float80Type>(this);
337  impl->f128Ty = TypeUniquer::get<Float128Type>(this);
338  /// Index Type.
339  impl->indexTy = TypeUniquer::get<IndexType>(this);
340  /// Integer Types.
341  impl->int1Ty = TypeUniquer::get<IntegerType>(this, 1, IntegerType::Signless);
342  impl->int8Ty = TypeUniquer::get<IntegerType>(this, 8, IntegerType::Signless);
343  impl->int16Ty =
344  TypeUniquer::get<IntegerType>(this, 16, IntegerType::Signless);
345  impl->int32Ty =
346  TypeUniquer::get<IntegerType>(this, 32, IntegerType::Signless);
347  impl->int64Ty =
348  TypeUniquer::get<IntegerType>(this, 64, IntegerType::Signless);
349  impl->int128Ty =
350  TypeUniquer::get<IntegerType>(this, 128, IntegerType::Signless);
351  /// None Type.
352  impl->noneType = TypeUniquer::get<NoneType>(this);
353 
354  //// Attributes.
355  //// Note: These must be registered after the types as they may generate one
356  //// of the above types internally.
357  /// Unknown Location Attribute.
358  impl->unknownLocAttr = AttributeUniquer::get<UnknownLoc>(this);
359  /// Bool Attributes.
360  impl->falseAttr = IntegerAttr::getBoolAttrUnchecked(impl->int1Ty, false);
361  impl->trueAttr = IntegerAttr::getBoolAttrUnchecked(impl->int1Ty, true);
362  /// Unit Attribute.
363  impl->unitAttr = AttributeUniquer::get<UnitAttr>(this);
364  /// The empty dictionary attribute.
365  impl->emptyDictionaryAttr = DictionaryAttr::getEmptyUnchecked(this);
366  /// The empty string attribute.
367  impl->emptyStringAttr = StringAttr::getEmptyStringAttrUnchecked(this);
368 
369  // Register the affine storage objects with the uniquer.
370  impl->affineUniquer
371  .registerParametricStorageType<AffineBinaryOpExprStorage>();
372  impl->affineUniquer
373  .registerParametricStorageType<AffineConstantExprStorage>();
374  impl->affineUniquer.registerParametricStorageType<AffineDimExprStorage>();
375  impl->affineUniquer.registerParametricStorageType<AffineMapStorage>();
376  impl->affineUniquer.registerParametricStorageType<IntegerSetStorage>();
377 }
378 
379 MLIRContext::~MLIRContext() = default;
380 
381 /// Copy the specified array of elements into memory managed by the provided
382 /// bump pointer allocator. This assumes the elements are all PODs.
383 template <typename T>
384 static ArrayRef<T> copyArrayRefInto(llvm::BumpPtrAllocator &allocator,
385  ArrayRef<T> elements) {
386  auto result = allocator.Allocate<T>(elements.size());
387  std::uninitialized_copy(elements.begin(), elements.end(), result);
388  return ArrayRef<T>(result, elements.size());
389 }
390 
391 //===----------------------------------------------------------------------===//
392 // Action Handling
393 //===----------------------------------------------------------------------===//
394 
396  getImpl().actionHandler = std::move(handler);
397 }
398 
399 /// Dispatch the provided action to the handler if any, or just execute it.
400 void MLIRContext::executeActionInternal(function_ref<void()> actionFn,
401  const tracing::Action &action) {
402  assert(getImpl().actionHandler);
403  getImpl().actionHandler(actionFn, action);
404 }
405 
407 
408 //===----------------------------------------------------------------------===//
409 // Diagnostic Handlers
410 //===----------------------------------------------------------------------===//
411 
412 /// Returns the diagnostic engine for this context.
414 
415 //===----------------------------------------------------------------------===//
416 // Dialect and Operation Registration
417 //===----------------------------------------------------------------------===//
418 
420  if (registry.isSubsetOf(impl->dialectsRegistry))
421  return;
422 
423  assert(impl->multiThreadedExecutionContext == 0 &&
424  "appending to the MLIRContext dialect registry while in a "
425  "multi-threaded execution context");
426  registry.appendTo(impl->dialectsRegistry);
427 
428  // For the already loaded dialects, apply any possible extensions immediately.
429  registry.applyExtensions(this);
430 }
431 
433  return impl->dialectsRegistry;
434 }
435 
436 /// Return information about all registered IR dialects.
437 std::vector<Dialect *> MLIRContext::getLoadedDialects() {
438  std::vector<Dialect *> result;
439  result.reserve(impl->loadedDialects.size());
440  for (auto &dialect : impl->loadedDialects)
441  result.push_back(dialect.second.get());
442  llvm::array_pod_sort(result.begin(), result.end(),
443  [](Dialect *const *lhs, Dialect *const *rhs) -> int {
444  return (*lhs)->getNamespace() < (*rhs)->getNamespace();
445  });
446  return result;
447 }
448 std::vector<StringRef> MLIRContext::getAvailableDialects() {
449  std::vector<StringRef> result;
450  for (auto dialect : impl->dialectsRegistry.getDialectNames())
451  result.push_back(dialect);
452  return result;
453 }
454 
455 /// Get a registered IR dialect with the given namespace. If none is found,
456 /// then return nullptr.
458  // Dialects are sorted by name, so we can use binary search for lookup.
459  auto it = impl->loadedDialects.find(name);
460  return (it != impl->loadedDialects.end()) ? it->second.get() : nullptr;
461 }
462 
464  Dialect *dialect = getLoadedDialect(name);
465  if (dialect)
466  return dialect;
467  DialectAllocatorFunctionRef allocator =
468  impl->dialectsRegistry.getDialectAllocator(name);
469  return allocator ? allocator(this) : nullptr;
470 }
471 
472 /// Get a dialect for the provided namespace and TypeID: abort the program if a
473 /// dialect exist for this namespace with different TypeID. Returns a pointer to
474 /// the dialect owned by the context.
475 Dialect *
476 MLIRContext::getOrLoadDialect(StringRef dialectNamespace, TypeID dialectID,
477  function_ref<std::unique_ptr<Dialect>()> ctor) {
478  auto &impl = getImpl();
479  // Get the correct insertion position sorted by namespace.
480  auto dialectIt = impl.loadedDialects.try_emplace(dialectNamespace, nullptr);
481 
482  if (dialectIt.second) {
483  LLVM_DEBUG(llvm::dbgs()
484  << "Load new dialect in Context " << dialectNamespace << "\n");
485 #ifndef NDEBUG
486  if (impl.multiThreadedExecutionContext != 0)
487  llvm::report_fatal_error(
488  "Loading a dialect (" + dialectNamespace +
489  ") while in a multi-threaded execution context (maybe "
490  "the PassManager): this can indicate a "
491  "missing `dependentDialects` in a pass for example.");
492 #endif // NDEBUG
493  // loadedDialects entry is initialized to nullptr, indicating that the
494  // dialect is currently being loaded. Re-lookup the address in
495  // loadedDialects because the table might have been rehashed by recursive
496  // dialect loading in ctor().
497  std::unique_ptr<Dialect> &dialectOwned =
498  impl.loadedDialects[dialectNamespace] = ctor();
499  Dialect *dialect = dialectOwned.get();
500  assert(dialect && "dialect ctor failed");
501 
502  // Refresh all the identifiers dialect field, this catches cases where a
503  // dialect may be loaded after identifier prefixed with this dialect name
504  // were already created.
505  auto stringAttrsIt = impl.dialectReferencingStrAttrs.find(dialectNamespace);
506  if (stringAttrsIt != impl.dialectReferencingStrAttrs.end()) {
507  for (StringAttrStorage *storage : stringAttrsIt->second)
508  storage->referencedDialect = dialect;
509  impl.dialectReferencingStrAttrs.erase(stringAttrsIt);
510  }
511 
512  // Apply any extensions to this newly loaded dialect.
513  impl.dialectsRegistry.applyExtensions(dialect);
514  return dialect;
515  }
516 
517 #ifndef NDEBUG
518  if (dialectIt.first->second == nullptr)
519  llvm::report_fatal_error(
520  "Loading (and getting) a dialect (" + dialectNamespace +
521  ") while the same dialect is still loading: use loadDialect instead "
522  "of getOrLoadDialect.");
523 #endif // NDEBUG
524 
525  // Abort if dialect with namespace has already been registered.
526  std::unique_ptr<Dialect> &dialect = dialectIt.first->second;
527  if (dialect->getTypeID() != dialectID)
528  llvm::report_fatal_error("a dialect with namespace '" + dialectNamespace +
529  "' has already been registered");
530 
531  return dialect.get();
532 }
533 
534 bool MLIRContext::isDialectLoading(StringRef dialectNamespace) {
535  auto it = getImpl().loadedDialects.find(dialectNamespace);
536  // nullptr indicates that the dialect is currently being loaded.
537  return it != getImpl().loadedDialects.end() && it->second == nullptr;
538 }
539 
541  StringRef dialectNamespace, function_ref<void(DynamicDialect *)> ctor) {
542  auto &impl = getImpl();
543  // Get the correct insertion position sorted by namespace.
544  auto dialectIt = impl.loadedDialects.find(dialectNamespace);
545 
546  if (dialectIt != impl.loadedDialects.end()) {
547  if (auto *dynDialect = dyn_cast<DynamicDialect>(dialectIt->second.get()))
548  return dynDialect;
549  llvm::report_fatal_error("a dialect with namespace '" + dialectNamespace +
550  "' has already been registered");
551  }
552 
553  LLVM_DEBUG(llvm::dbgs() << "Load new dynamic dialect in Context "
554  << dialectNamespace << "\n");
555 #ifndef NDEBUG
556  if (impl.multiThreadedExecutionContext != 0)
557  llvm::report_fatal_error(
558  "Loading a dynamic dialect (" + dialectNamespace +
559  ") while in a multi-threaded execution context (maybe "
560  "the PassManager): this can indicate a "
561  "missing `dependentDialects` in a pass for example.");
562 #endif
563 
564  auto name = StringAttr::get(this, dialectNamespace);
565  auto *dialect = new DynamicDialect(name, this);
566  (void)getOrLoadDialect(name, dialect->getTypeID(), [dialect, ctor]() {
567  ctor(dialect);
568  return std::unique_ptr<DynamicDialect>(dialect);
569  });
570  // This is the same result as `getOrLoadDialect` (if it didn't failed),
571  // since it has the same TypeID, and TypeIDs are unique.
572  return dialect;
573 }
574 
576  for (StringRef name : getAvailableDialects())
577  getOrLoadDialect(name);
578 }
579 
580 llvm::hash_code MLIRContext::getRegistryHash() {
581  llvm::hash_code hash(0);
582  // Factor in number of loaded dialects, attributes, operations, types.
583  hash = llvm::hash_combine(hash, impl->loadedDialects.size());
584  hash = llvm::hash_combine(hash, impl->registeredAttributes.size());
585  hash = llvm::hash_combine(hash, impl->registeredOperations.size());
586  hash = llvm::hash_combine(hash, impl->registeredTypes.size());
587  return hash;
588 }
589 
591  return impl->allowUnregisteredDialects;
592 }
593 
595  assert(impl->multiThreadedExecutionContext == 0 &&
596  "changing MLIRContext `allow-unregistered-dialects` configuration "
597  "while in a multi-threaded execution context");
598  impl->allowUnregisteredDialects = allowing;
599 }
600 
601 /// Return true if multi-threading is enabled by the context.
603  return impl->threadingIsEnabled && llvm::llvm_is_multithreaded();
604 }
605 
606 /// Set the flag specifying if multi-threading is disabled by the context.
608  // This API can be overridden by the global debugging flag
609  // --mlir-disable-threading
611  return;
612  assert(impl->multiThreadedExecutionContext == 0 &&
613  "changing MLIRContext `disable-threading` configuration while "
614  "in a multi-threaded execution context");
615 
616  impl->threadingIsEnabled = !disable;
617 
618  // Update the threading mode for each of the uniquers.
619  impl->affineUniquer.disableMultithreading(disable);
620  impl->attributeUniquer.disableMultithreading(disable);
621  impl->typeUniquer.disableMultithreading(disable);
622 
623  // Destroy thread pool (stop all threads) if it is no longer needed, or create
624  // a new one if multithreading was re-enabled.
625  if (disable) {
626  // If the thread pool is owned, explicitly set it to nullptr to avoid
627  // keeping a dangling pointer around. If the thread pool is externally
628  // owned, we don't do anything.
629  if (impl->ownedThreadPool) {
630  assert(impl->threadPool);
631  impl->threadPool = nullptr;
632  impl->ownedThreadPool.reset();
633  }
634  } else if (!impl->threadPool) {
635  // The thread pool isn't externally provided.
636  assert(!impl->ownedThreadPool);
637  impl->ownedThreadPool = std::make_unique<llvm::DefaultThreadPool>();
638  impl->threadPool = impl->ownedThreadPool.get();
639  }
640 }
641 
642 void MLIRContext::setThreadPool(llvm::ThreadPoolInterface &pool) {
643  assert(!isMultithreadingEnabled() &&
644  "expected multi-threading to be disabled when setting a ThreadPool");
645  impl->threadPool = &pool;
646  impl->ownedThreadPool.reset();
648 }
649 
651  if (isMultithreadingEnabled()) {
652  assert(impl->threadPool &&
653  "multi-threading is enabled but threadpool not set");
654  return impl->threadPool->getMaxConcurrency();
655  }
656  // No multithreading or active thread pool. Return 1 thread.
657  return 1;
658 }
659 
660 llvm::ThreadPoolInterface &MLIRContext::getThreadPool() {
661  assert(isMultithreadingEnabled() &&
662  "expected multi-threading to be enabled within the context");
663  assert(impl->threadPool &&
664  "multi-threading is enabled but threadpool not set");
665  return *impl->threadPool;
666 }
667 
669 #ifndef NDEBUG
670  ++impl->multiThreadedExecutionContext;
671 #endif
672 }
674 #ifndef NDEBUG
675  --impl->multiThreadedExecutionContext;
676 #endif
677 }
678 
679 /// Return true if we should attach the operation to diagnostics emitted via
680 /// Operation::emit.
682  return impl->printOpOnDiagnostic;
683 }
684 
685 /// Set the flag specifying if we should attach the operation to diagnostics
686 /// emitted via Operation::emit.
688  assert(impl->multiThreadedExecutionContext == 0 &&
689  "changing MLIRContext `print-op-on-diagnostic` configuration while in "
690  "a multi-threaded execution context");
691  impl->printOpOnDiagnostic = enable;
692 }
693 
694 /// Return true if we should attach the current stacktrace to diagnostics when
695 /// emitted.
697  return impl->printStackTraceOnDiagnostic;
698 }
699 
700 /// Set the flag specifying if we should attach the current stacktrace when
701 /// emitting diagnostics.
703  assert(impl->multiThreadedExecutionContext == 0 &&
704  "changing MLIRContext `print-stacktrace-on-diagnostic` configuration "
705  "while in a multi-threaded execution context");
706  impl->printStackTraceOnDiagnostic = enable;
707 }
708 
709 /// Return information about all registered operations.
711  return impl->sortedRegisteredOperations;
712 }
713 
714 /// Return information for registered operations by dialect.
717  auto lowerBound =
718  std::lower_bound(impl->sortedRegisteredOperations.begin(),
719  impl->sortedRegisteredOperations.end(), dialectName,
720  [](auto &lhs, auto &rhs) {
721  return lhs.getDialect().getNamespace().compare(rhs);
722  });
723 
724  if (lowerBound == impl->sortedRegisteredOperations.end() ||
725  lowerBound->getDialect().getNamespace() != dialectName)
727 
728  auto upperBound =
729  std::upper_bound(lowerBound, impl->sortedRegisteredOperations.end(),
730  dialectName, [](auto &lhs, auto &rhs) {
731  return lhs.compare(rhs.getDialect().getNamespace());
732  });
733 
734  size_t count = std::distance(lowerBound, upperBound);
735  return ArrayRef(&*lowerBound, count);
736 }
737 
739  return RegisteredOperationName::lookup(name, this).has_value();
740 }
741 
742 void Dialect::addType(TypeID typeID, AbstractType &&typeInfo) {
743  auto &impl = context->getImpl();
744  assert(impl.multiThreadedExecutionContext == 0 &&
745  "Registering a new type kind while in a multi-threaded execution "
746  "context");
747  auto *newInfo =
748  new (impl.abstractDialectSymbolAllocator.Allocate<AbstractType>())
749  AbstractType(std::move(typeInfo));
750  if (!impl.registeredTypes.insert({typeID, newInfo}).second)
751  llvm::report_fatal_error("Dialect Type already registered.");
752  if (!impl.nameToType.insert({newInfo->getName(), newInfo}).second)
753  llvm::report_fatal_error("Dialect Type with name " + newInfo->getName() +
754  " is already registered.");
755 }
756 
757 void Dialect::addAttribute(TypeID typeID, AbstractAttribute &&attrInfo) {
758  auto &impl = context->getImpl();
759  assert(impl.multiThreadedExecutionContext == 0 &&
760  "Registering a new attribute kind while in a multi-threaded execution "
761  "context");
762  auto *newInfo =
763  new (impl.abstractDialectSymbolAllocator.Allocate<AbstractAttribute>())
764  AbstractAttribute(std::move(attrInfo));
765  if (!impl.registeredAttributes.insert({typeID, newInfo}).second)
766  llvm::report_fatal_error("Dialect Attribute already registered.");
767  if (!impl.nameToAttribute.insert({newInfo->getName(), newInfo}).second)
768  llvm::report_fatal_error("Dialect Attribute with name " +
769  newInfo->getName() + " is already registered.");
770 }
771 
772 //===----------------------------------------------------------------------===//
773 // AbstractAttribute
774 //===----------------------------------------------------------------------===//
775 
776 /// Get the dialect that registered the attribute with the provided typeid.
778  MLIRContext *context) {
779  const AbstractAttribute *abstract = lookupMutable(typeID, context);
780  if (!abstract)
781  llvm::report_fatal_error("Trying to create an Attribute that was not "
782  "registered in this MLIRContext.");
783  return *abstract;
784 }
785 
786 AbstractAttribute *AbstractAttribute::lookupMutable(TypeID typeID,
787  MLIRContext *context) {
788  auto &impl = context->getImpl();
789  return impl.registeredAttributes.lookup(typeID);
790 }
791 
792 std::optional<std::reference_wrapper<const AbstractAttribute>>
793 AbstractAttribute::lookup(StringRef name, MLIRContext *context) {
794  MLIRContextImpl &impl = context->getImpl();
795  const AbstractAttribute *type = impl.nameToAttribute.lookup(name);
796 
797  if (!type)
798  return std::nullopt;
799  return {*type};
800 }
801 
802 //===----------------------------------------------------------------------===//
803 // OperationName
804 //===----------------------------------------------------------------------===//
805 
806 OperationName::Impl::Impl(StringRef name, Dialect *dialect, TypeID typeID,
807  detail::InterfaceMap interfaceMap)
808  : Impl(StringAttr::get(dialect->getContext(), name), dialect, typeID,
809  std::move(interfaceMap)) {}
810 
811 OperationName::OperationName(StringRef name, MLIRContext *context) {
812  MLIRContextImpl &ctxImpl = context->getImpl();
813 
814  // Check for an existing name in read-only mode.
815  bool isMultithreadingEnabled = context->isMultithreadingEnabled();
816  if (isMultithreadingEnabled) {
817  // Check the registered info map first. In the overwhelmingly common case,
818  // the entry will be in here and it also removes the need to acquire any
819  // locks.
820  auto registeredIt = ctxImpl.registeredOperationsByName.find(name);
821  if (LLVM_LIKELY(registeredIt != ctxImpl.registeredOperationsByName.end())) {
822  impl = registeredIt->second.impl;
823  return;
824  }
825 
826  llvm::sys::SmartScopedReader<true> contextLock(ctxImpl.operationInfoMutex);
827  auto it = ctxImpl.operations.find(name);
828  if (it != ctxImpl.operations.end()) {
829  impl = it->second.get();
830  return;
831  }
832  }
833 
834  // Acquire a writer-lock so that we can safely create the new instance.
835  ScopedWriterLock lock(ctxImpl.operationInfoMutex, isMultithreadingEnabled);
836 
837  auto it = ctxImpl.operations.insert({name, nullptr});
838  if (it.second) {
839  auto nameAttr = StringAttr::get(context, name);
840  it.first->second = std::make_unique<UnregisteredOpModel>(
841  nameAttr, nameAttr.getReferencedDialect(), TypeID::get<void>(),
843  }
844  impl = it.first->second.get();
845 }
846 
848  if (Dialect *dialect = getDialect())
849  return dialect->getNamespace();
850  return getStringRef().split('.').first;
851 }
852 
853 LogicalResult
856  return failure();
857 }
861 
864  llvm::report_fatal_error("getParseAssemblyFn hook called on unregistered op");
865 }
867  const OperationName &, NamedAttrList &) {}
869  Operation *op, OpAsmPrinter &p, StringRef defaultDialect) {
870  p.printGenericOp(op);
871 }
872 LogicalResult
874  return success();
875 }
876 LogicalResult
878  return success();
879 }
880 
881 std::optional<Attribute>
883  StringRef name) {
884  auto dict = dyn_cast_or_null<DictionaryAttr>(getPropertiesAsAttr(op));
885  if (!dict)
886  return std::nullopt;
887  if (Attribute attr = dict.get(name))
888  return attr;
889  return std::nullopt;
890 }
892  StringAttr name,
893  Attribute value) {
894  auto dict = dyn_cast_or_null<DictionaryAttr>(getPropertiesAsAttr(op));
895  assert(dict);
896  NamedAttrList attrs(dict);
897  attrs.set(name, value);
898  *op->getPropertiesStorage().as<Attribute *>() =
899  attrs.getDictionary(op->getContext());
900 }
902  Operation *op, NamedAttrList &attrs) {}
904  OperationName opName, NamedAttrList &attributes,
906  return success();
907 }
909  return sizeof(Attribute);
910 }
912  OperationName opName, OpaqueProperties storage, OpaqueProperties init) {
913  new (storage.as<Attribute *>()) Attribute();
914 }
916  OpaqueProperties prop) {
917  prop.as<Attribute *>()->~Attribute();
918 }
920  OperationName opName, OpaqueProperties properties) {}
922  OperationName opName, OpaqueProperties properties, Attribute attr,
924  *properties.as<Attribute *>() = attr;
925  return success();
926 }
927 Attribute
929  return *op->getPropertiesStorage().as<Attribute *>();
930 }
932  OpaqueProperties rhs) {
933  *lhs.as<Attribute *>() = *rhs.as<Attribute *>();
934 }
937  return *lhs.as<Attribute *>() == *rhs.as<Attribute *>();
938 }
939 llvm::hash_code
941  return llvm::hash_combine(*prop.as<Attribute *>());
942 }
943 
944 //===----------------------------------------------------------------------===//
945 // RegisteredOperationName
946 //===----------------------------------------------------------------------===//
947 
948 std::optional<RegisteredOperationName>
950  auto &impl = ctx->getImpl();
951  auto it = impl.registeredOperations.find(typeID);
952  if (it != impl.registeredOperations.end())
953  return it->second;
954  return std::nullopt;
955 }
956 
957 std::optional<RegisteredOperationName>
959  auto &impl = ctx->getImpl();
960  auto it = impl.registeredOperationsByName.find(name);
961  if (it != impl.registeredOperationsByName.end())
962  return it->getValue();
963  return std::nullopt;
964 }
965 
967  std::unique_ptr<RegisteredOperationName::Impl> ownedImpl,
968  ArrayRef<StringRef> attrNames) {
969  RegisteredOperationName::Impl *impl = ownedImpl.get();
970  MLIRContext *ctx = impl->getDialect()->getContext();
971  auto &ctxImpl = ctx->getImpl();
972  assert(ctxImpl.multiThreadedExecutionContext == 0 &&
973  "registering a new operation kind while in a multi-threaded execution "
974  "context");
975 
976  // Register the attribute names of this operation.
977  MutableArrayRef<StringAttr> cachedAttrNames;
978  if (!attrNames.empty()) {
979  cachedAttrNames = MutableArrayRef<StringAttr>(
980  ctxImpl.abstractDialectSymbolAllocator.Allocate<StringAttr>(
981  attrNames.size()),
982  attrNames.size());
983  for (unsigned i : llvm::seq<unsigned>(0, attrNames.size()))
984  new (&cachedAttrNames[i]) StringAttr(StringAttr::get(ctx, attrNames[i]));
985  impl->attributeNames = cachedAttrNames;
986  }
987  StringRef name = impl->getName().strref();
988  // Insert the operation info if it doesn't exist yet.
989  ctxImpl.operations[name] = std::move(ownedImpl);
990 
991  // Update the registered info for this operation.
992  auto emplaced = ctxImpl.registeredOperations.try_emplace(
993  impl->getTypeID(), RegisteredOperationName(impl));
994  assert(emplaced.second && "operation name registration must be successful");
995  auto emplacedByName = ctxImpl.registeredOperationsByName.try_emplace(
997  (void)emplacedByName;
998  assert(emplacedByName.second &&
999  "operation name registration must be successful");
1000 
1001  // Add emplaced operation name to the sorted operations container.
1002  RegisteredOperationName &value = emplaced.first->second;
1003  ctxImpl.sortedRegisteredOperations.insert(
1004  llvm::upper_bound(ctxImpl.sortedRegisteredOperations, value,
1005  [](auto &lhs, auto &rhs) {
1006  return lhs.getIdentifier().compare(
1007  rhs.getIdentifier());
1008  }),
1009  value);
1010 }
1011 
1012 //===----------------------------------------------------------------------===//
1013 // AbstractType
1014 //===----------------------------------------------------------------------===//
1015 
1017  const AbstractType *type = lookupMutable(typeID, context);
1018  if (!type)
1019  llvm::report_fatal_error(
1020  "Trying to create a Type that was not registered in this MLIRContext.");
1021  return *type;
1022 }
1023 
1024 AbstractType *AbstractType::lookupMutable(TypeID typeID, MLIRContext *context) {
1025  auto &impl = context->getImpl();
1026  return impl.registeredTypes.lookup(typeID);
1027 }
1028 
1029 std::optional<std::reference_wrapper<const AbstractType>>
1030 AbstractType::lookup(StringRef name, MLIRContext *context) {
1031  MLIRContextImpl &impl = context->getImpl();
1032  const AbstractType *type = impl.nameToType.lookup(name);
1033 
1034  if (!type)
1035  return std::nullopt;
1036  return {*type};
1037 }
1038 
1039 //===----------------------------------------------------------------------===//
1040 // Type uniquing
1041 //===----------------------------------------------------------------------===//
1042 
1043 /// Returns the storage uniquer used for constructing type storage instances.
1044 /// This should not be used directly.
1046 
1047 Float4E2M1FNType Float4E2M1FNType::get(MLIRContext *context) {
1048  return context->getImpl().f4E2M1FNTy;
1049 }
1050 Float6E2M3FNType Float6E2M3FNType::get(MLIRContext *context) {
1051  return context->getImpl().f6E2M3FNTy;
1052 }
1053 Float6E3M2FNType Float6E3M2FNType::get(MLIRContext *context) {
1054  return context->getImpl().f6E3M2FNTy;
1055 }
1056 Float8E5M2Type Float8E5M2Type::get(MLIRContext *context) {
1057  return context->getImpl().f8E5M2Ty;
1058 }
1059 Float8E4M3Type Float8E4M3Type::get(MLIRContext *context) {
1060  return context->getImpl().f8E4M3Ty;
1061 }
1062 Float8E4M3FNType Float8E4M3FNType::get(MLIRContext *context) {
1063  return context->getImpl().f8E4M3FNTy;
1064 }
1065 Float8E5M2FNUZType Float8E5M2FNUZType::get(MLIRContext *context) {
1066  return context->getImpl().f8E5M2FNUZTy;
1067 }
1068 Float8E4M3FNUZType Float8E4M3FNUZType::get(MLIRContext *context) {
1069  return context->getImpl().f8E4M3FNUZTy;
1070 }
1071 Float8E4M3B11FNUZType Float8E4M3B11FNUZType::get(MLIRContext *context) {
1072  return context->getImpl().f8E4M3B11FNUZTy;
1073 }
1074 Float8E3M4Type Float8E3M4Type::get(MLIRContext *context) {
1075  return context->getImpl().f8E3M4Ty;
1076 }
1077 Float8E8M0FNUType Float8E8M0FNUType::get(MLIRContext *context) {
1078  return context->getImpl().f8E8M0FNUTy;
1079 }
1080 BFloat16Type BFloat16Type::get(MLIRContext *context) {
1081  return context->getImpl().bf16Ty;
1082 }
1083 Float16Type Float16Type::get(MLIRContext *context) {
1084  return context->getImpl().f16Ty;
1085 }
1086 FloatTF32Type FloatTF32Type::get(MLIRContext *context) {
1087  return context->getImpl().tf32Ty;
1088 }
1089 Float32Type Float32Type::get(MLIRContext *context) {
1090  return context->getImpl().f32Ty;
1091 }
1092 Float64Type Float64Type::get(MLIRContext *context) {
1093  return context->getImpl().f64Ty;
1094 }
1095 Float80Type Float80Type::get(MLIRContext *context) {
1096  return context->getImpl().f80Ty;
1097 }
1098 Float128Type Float128Type::get(MLIRContext *context) {
1099  return context->getImpl().f128Ty;
1100 }
1101 
1102 /// Get an instance of the IndexType.
1103 IndexType IndexType::get(MLIRContext *context) {
1104  return context->getImpl().indexTy;
1105 }
1106 
1107 /// Return an existing integer type instance if one is cached within the
1108 /// context.
1109 static IntegerType
1110 getCachedIntegerType(unsigned width,
1111  IntegerType::SignednessSemantics signedness,
1112  MLIRContext *context) {
1113  if (signedness != IntegerType::Signless)
1114  return IntegerType();
1115 
1116  switch (width) {
1117  case 1:
1118  return context->getImpl().int1Ty;
1119  case 8:
1120  return context->getImpl().int8Ty;
1121  case 16:
1122  return context->getImpl().int16Ty;
1123  case 32:
1124  return context->getImpl().int32Ty;
1125  case 64:
1126  return context->getImpl().int64Ty;
1127  case 128:
1128  return context->getImpl().int128Ty;
1129  default:
1130  return IntegerType();
1131  }
1132 }
1133 
1134 IntegerType IntegerType::get(MLIRContext *context, unsigned width,
1135  IntegerType::SignednessSemantics signedness) {
1136  if (auto cached = getCachedIntegerType(width, signedness, context))
1137  return cached;
1138  return Base::get(context, width, signedness);
1139 }
1140 
1141 IntegerType
1142 IntegerType::getChecked(function_ref<InFlightDiagnostic()> emitError,
1143  MLIRContext *context, unsigned width,
1144  SignednessSemantics signedness) {
1145  if (auto cached = getCachedIntegerType(width, signedness, context))
1146  return cached;
1147  return Base::getChecked(emitError, context, width, signedness);
1148 }
1149 
1150 /// Get an instance of the NoneType.
1151 NoneType NoneType::get(MLIRContext *context) {
1152  if (NoneType cachedInst = context->getImpl().noneType)
1153  return cachedInst;
1154  // Note: May happen when initializing the singleton attributes of the builtin
1155  // dialect.
1156  return Base::get(context);
1157 }
1158 
1159 //===----------------------------------------------------------------------===//
1160 // Attribute uniquing
1161 //===----------------------------------------------------------------------===//
1162 
1163 /// Returns the storage uniquer used for constructing attribute storage
1164 /// instances. This should not be used directly.
1166  return getImpl().attributeUniquer;
1167 }
1168 
1169 /// Initialize the given attribute storage instance.
1170 void AttributeUniquer::initializeAttributeStorage(AttributeStorage *storage,
1171  MLIRContext *ctx,
1172  TypeID attrID) {
1174 }
1175 
1176 BoolAttr BoolAttr::get(MLIRContext *context, bool value) {
1177  return value ? context->getImpl().trueAttr : context->getImpl().falseAttr;
1178 }
1179 
1180 UnitAttr UnitAttr::get(MLIRContext *context) {
1181  return context->getImpl().unitAttr;
1182 }
1183 
1184 UnknownLoc UnknownLoc::get(MLIRContext *context) {
1185  return context->getImpl().unknownLocAttr;
1186 }
1187 
1189 detail::DistinctAttributeUniquer::allocateStorage(MLIRContext *context,
1190  Attribute referencedAttr) {
1191  return context->getImpl().distinctAttributeAllocator.allocate(referencedAttr);
1192 }
1193 
1194 /// Return empty dictionary.
1195 DictionaryAttr DictionaryAttr::getEmpty(MLIRContext *context) {
1196  return context->getImpl().emptyDictionaryAttr;
1197 }
1198 
1199 void StringAttrStorage::initialize(MLIRContext *context) {
1200  // Check for a dialect namespace prefix, if there isn't one we don't need to
1201  // do any additional initialization.
1202  auto dialectNamePair = value.split('.');
1203  if (dialectNamePair.first.empty() || dialectNamePair.second.empty())
1204  return;
1205 
1206  // If one exists, we check to see if this dialect is loaded. If it is, we set
1207  // the dialect now, if it isn't we record this storage for initialization
1208  // later if the dialect ever gets loaded.
1209  if ((referencedDialect = context->getLoadedDialect(dialectNamePair.first)))
1210  return;
1211 
1212  MLIRContextImpl &impl = context->getImpl();
1213  llvm::sys::SmartScopedLock<true> lock(impl.dialectRefStrAttrMutex);
1214  impl.dialectReferencingStrAttrs[dialectNamePair.first].push_back(this);
1215 }
1216 
1217 /// Return an empty string.
1218 StringAttr StringAttr::get(MLIRContext *context) {
1219  return context->getImpl().emptyStringAttr;
1220 }
1221 
1222 //===----------------------------------------------------------------------===//
1223 // AffineMap uniquing
1224 //===----------------------------------------------------------------------===//
1225 
1227  return getImpl().affineUniquer;
1228 }
1229 
1230 AffineMap AffineMap::getImpl(unsigned dimCount, unsigned symbolCount,
1231  ArrayRef<AffineExpr> results,
1232  MLIRContext *context) {
1233  auto &impl = context->getImpl();
1234  auto *storage = impl.affineUniquer.get<AffineMapStorage>(
1235  [&](AffineMapStorage *storage) { storage->context = context; }, dimCount,
1236  symbolCount, results);
1237  return AffineMap(storage);
1238 }
1239 
1240 /// Check whether the arguments passed to the AffineMap::get() are consistent.
1241 /// This method checks whether the highest index of dimensional identifier
1242 /// present in result expressions is less than `dimCount` and the highest index
1243 /// of symbolic identifier present in result expressions is less than
1244 /// `symbolCount`.
1245 LLVM_ATTRIBUTE_UNUSED static bool
1246 willBeValidAffineMap(unsigned dimCount, unsigned symbolCount,
1247  ArrayRef<AffineExpr> results) {
1248  int64_t maxDimPosition = -1;
1249  int64_t maxSymbolPosition = -1;
1250  getMaxDimAndSymbol(ArrayRef<ArrayRef<AffineExpr>>(results), maxDimPosition,
1251  maxSymbolPosition);
1252  if ((maxDimPosition >= dimCount) || (maxSymbolPosition >= symbolCount)) {
1253  LLVM_DEBUG(
1254  llvm::dbgs()
1255  << "maximum dimensional identifier position in result expression must "
1256  "be less than `dimCount` and maximum symbolic identifier position "
1257  "in result expression must be less than `symbolCount`\n");
1258  return false;
1259  }
1260  return true;
1261 }
1262 
1264  return getImpl(/*dimCount=*/0, /*symbolCount=*/0, /*results=*/{}, context);
1265 }
1266 
1267 AffineMap AffineMap::get(unsigned dimCount, unsigned symbolCount,
1268  MLIRContext *context) {
1269  return getImpl(dimCount, symbolCount, /*results=*/{}, context);
1270 }
1271 
1272 AffineMap AffineMap::get(unsigned dimCount, unsigned symbolCount,
1273  AffineExpr result) {
1274  assert(willBeValidAffineMap(dimCount, symbolCount, {result}));
1275  return getImpl(dimCount, symbolCount, {result}, result.getContext());
1276 }
1277 
1278 AffineMap AffineMap::get(unsigned dimCount, unsigned symbolCount,
1279  ArrayRef<AffineExpr> results, MLIRContext *context) {
1280  assert(willBeValidAffineMap(dimCount, symbolCount, results));
1281  return getImpl(dimCount, symbolCount, results, context);
1282 }
1283 
1284 //===----------------------------------------------------------------------===//
1285 // Integer Sets: these are allocated into the bump pointer, and are immutable.
1286 // Unlike AffineMap's, these are uniqued only if they are small.
1287 //===----------------------------------------------------------------------===//
1288 
1289 IntegerSet IntegerSet::get(unsigned dimCount, unsigned symbolCount,
1290  ArrayRef<AffineExpr> constraints,
1291  ArrayRef<bool> eqFlags) {
1292  // The number of constraints can't be zero.
1293  assert(!constraints.empty());
1294  assert(constraints.size() == eqFlags.size());
1295 
1296  auto &impl = constraints[0].getContext()->getImpl();
1297  auto *storage = impl.affineUniquer.get<IntegerSetStorage>(
1298  [](IntegerSetStorage *) {}, dimCount, symbolCount, constraints, eqFlags);
1299  return IntegerSet(storage);
1300 }
1301 
1302 //===----------------------------------------------------------------------===//
1303 // StorageUniquerSupport
1304 //===----------------------------------------------------------------------===//
1305 
1306 /// Utility method to generate a callback that can be used to generate a
1307 /// diagnostic when checking the construction invariants of a storage object.
1308 /// This is defined out-of-line to avoid the need to include Location.h.
1309 llvm::unique_function<InFlightDiagnostic()>
1311  return [ctx] { return emitError(UnknownLoc::get(ctx)); };
1312 }
1313 llvm::unique_function<InFlightDiagnostic()>
1315  return [=] { return emitError(loc); };
1316 }
static MLIRContext * getContext(OpFoldResult val)
static LLVM_ATTRIBUTE_UNUSED bool willBeValidAffineMap(unsigned dimCount, unsigned symbolCount, ArrayRef< AffineExpr > results)
Check whether the arguments passed to the AffineMap::get() are consistent.
static bool isThreadingGloballyDisabled()
Definition: MLIRContext.cpp:79
static ArrayRef< T > copyArrayRefInto(llvm::BumpPtrAllocator &allocator, ArrayRef< T > elements)
Copy the specified array of elements into memory managed by the provided bump pointer allocator.
static llvm::ManagedStatic< MLIRContextOptions > clOptions
Definition: MLIRContext.cpp:77
static IntegerType getCachedIntegerType(unsigned width, IntegerType::SignednessSemantics signedness, MLIRContext *context)
Return an existing integer type instance if one is cached within the context.
This class contains all of the static information common to all instances of a registered Attribute.
static const AbstractAttribute & lookup(TypeID typeID, MLIRContext *context)
Look up the specified abstract attribute in the MLIRContext and return a reference to it.
This class contains all of the static information common to all instances of a registered Type.
Definition: TypeSupport.h:30
static const AbstractType & lookup(TypeID typeID, MLIRContext *context)
Look up the specified abstract type in the MLIRContext and return a reference to it.
Base type for affine expression.
Definition: AffineExpr.h:68
MLIRContext * getContext() const
Definition: AffineExpr.cpp:33
A multi-dimensional affine map Affine map's are immutable like Type's, and they are uniqued.
Definition: AffineMap.h:46
static AffineMap get(MLIRContext *context)
Returns a zero result affine map with no dimensions or symbols: () -> ().
Base storage class appearing in an attribute.
void initializeAbstractAttribute(const AbstractAttribute &abstractAttr)
Set the abstract attribute for this storage instance.
Attributes are known-constant values of operations.
Definition: Attributes.h:25
Special case of IntegerAttr to represent boolean integers, i.e., signless i1 integers.
static BoolAttr get(MLIRContext *context, bool value)
This class is the main interface for diagnostics.
Definition: Diagnostics.h:414
The DialectRegistry maps a dialect namespace to a constructor for the matching dialect.
bool isSubsetOf(const DialectRegistry &rhs) const
Returns true if the current registry is a subset of 'rhs', i.e.
Definition: Dialect.cpp:329
void appendTo(DialectRegistry &destination) const
void applyExtensions(Dialect *dialect) const
Apply any held extensions that require the given dialect.
Definition: Dialect.cpp:255
Dialects are groups of MLIR operations, types and attributes, as well as behavior associated with the...
Definition: Dialect.h:38
A dialect that can be defined at runtime.
This class represents a diagnostic that is inflight and set to be reported.
Definition: Diagnostics.h:314
An integer set representing a conjunction of one or more affine equalities and inequalities.
Definition: IntegerSet.h:44
static IntegerSet get(unsigned dimCount, unsigned symbolCount, ArrayRef< AffineExpr > constraints, ArrayRef< bool > eqFlags)
This class defines the main interface for locations in MLIR and acts as a non-nullable wrapper around...
Definition: Location.h:66
This is the implementation of the MLIRContext class, using the pImpl idiom.
DictionaryAttr emptyDictionaryAttr
llvm::DenseMap< StringRef, AbstractAttribute * > nameToAttribute
This is a mapping from attribute name to the abstract attribute describing it.
DenseMap< StringRef, std::unique_ptr< Dialect > > loadedDialects
This is a list of dialects that are created referring to this context.
llvm::DenseMap< TypeID, RegisteredOperationName > registeredOperations
A vector of operation info specifically for registered operations.
BoolAttr falseAttr
Cached Attribute Instances.
llvm::sys::SmartRWMutex< true > operationInfoMutex
A mutex used when accessing operation information.
Float8E5M2FNUZType f8E5M2FNUZTy
Float6E2M3FNType f6E2M3FNTy
DenseMap< StringRef, SmallVector< StringAttrStorage * > > dialectReferencingStrAttrs
llvm::StringMap< RegisteredOperationName > registeredOperationsByName
StringAttr emptyStringAttr
std::function< void(function_ref< void()>, const tracing::Action &)> actionHandler
An action handler for handling actions that are dispatched through this context.
llvm::sys::SmartMutex< true > dialectRefStrAttrMutex
Map of string attributes that may reference a dialect, that are awaiting that dialect to be loaded.
llvm::DenseMap< StringRef, AbstractType * > nameToType
This is a mapping from type name to the abstract type describing it.
StorageUniquer typeUniquer
Float8E3M4Type f8E3M4Ty
llvm::BumpPtrAllocator abstractDialectSymbolAllocator
An allocator used for AbstractAttribute and AbstractType objects.
Float6E3M2FNType f6E3M2FNTy
DenseMap< TypeID, AbstractType * > registeredTypes
Float8E5M2Type f8E5M2Ty
StorageUniquer affineUniquer
DiagnosticEngine diagEngine
DenseMap< TypeID, AbstractAttribute * > registeredAttributes
Float4E2M1FNType f4E2M1FNTy
Cached Type Instances.
llvm::StringMap< std::unique_ptr< OperationName::Impl > > operations
This is a mapping from operation name to the operation info describing it.
Float8E4M3FNUZType f8E4M3FNUZTy
Float8E4M3B11FNUZType f8E4M3B11FNUZTy
std::unique_ptr< llvm::ThreadPoolInterface > ownedThreadPool
In case where the thread pool is owned by the context, this ensures destruction with the context.
Float8E8M0FNUType f8E8M0FNUTy
SmallVector< RegisteredOperationName, 0 > sortedRegisteredOperations
This is a sorted container of registered operations for a deterministic and efficient getRegisteredOp...
Float8E4M3FNType f8E4M3FNTy
DistinctAttributeAllocator distinctAttributeAllocator
A distinct attribute allocator that allocates every time since the address of the distinct attribute ...
MLIRContextImpl(bool threadingIsEnabled)
StorageUniquer attributeUniquer
Float8E4M3Type f8E4M3Ty
DialectRegistry dialectsRegistry
FloatTF32Type tf32Ty
MLIRContext is the top-level object for a collection of MLIR operations.
Definition: MLIRContext.h:60
void appendDialectRegistry(const DialectRegistry &registry)
Append the contents of the given dialect registry to the registry associated with this context.
bool shouldPrintStackTraceOnDiagnostic()
Return true if we should attach the current stacktrace to diagnostics when emitted.
unsigned getNumThreads()
Return the number of threads used by the thread pool in this context.
bool isOperationRegistered(StringRef name)
Return true if this operation name is registered in this context.
MLIRContext(Threading multithreading=Threading::ENABLED)
Create a new Context.
void disableMultithreading(bool disable=true)
Set the flag specifying if multi-threading is disabled by the context.
void printStackTraceOnDiagnostic(bool enable)
Set the flag specifying if we should attach the current stacktrace when emitting diagnostics.
Dialect * getLoadedDialect(StringRef name)
Get a registered IR dialect with the given namespace.
bool hasActionHandler()
Return true if a valid ActionHandler is set.
void setThreadPool(llvm::ThreadPoolInterface &pool)
Set a new thread pool to be used in this context.
void enableMultithreading(bool enable=true)
Definition: MLIRContext.h:152
std::vector< Dialect * > getLoadedDialects()
Return information about all IR dialects loaded in the context.
ArrayRef< RegisteredOperationName > getRegisteredOperationsByDialect(StringRef dialectName)
Return a sorted array containing the information for registered operations filtered by dialect name.
void printOpOnDiagnostic(bool enable)
Set the flag specifying if we should attach the operation to diagnostics emitted via Operation::emit.
void registerActionHandler(HandlerTy handler)
Register a handler for handling actions that are dispatched through this context.
ArrayRef< RegisteredOperationName > getRegisteredOperations()
Return a sorted array containing the information about all registered operations.
llvm::hash_code getRegistryHash()
Returns a hash of the registry of the context that may be used to give a rough indicator of if the st...
void enterMultiThreadedExecution()
These APIs are tracking whether the context will be used in a multithreading environment: this has no...
const DialectRegistry & getDialectRegistry()
Return the dialect registry associated with this context.
DynamicDialect * getOrLoadDynamicDialect(StringRef dialectNamespace, function_ref< void(DynamicDialect *)> ctor)
Get (or create) a dynamic dialect for the given name.
StorageUniquer & getAttributeUniquer()
Returns the storage uniquer used for constructing attribute storage instances.
StorageUniquer & getAffineUniquer()
Returns the storage uniquer used for creating affine constructs.
StorageUniquer & getTypeUniquer()
Returns the storage uniquer used for constructing type storage instances.
llvm::ThreadPoolInterface & getThreadPool()
Return the thread pool used by this context.
std::vector< StringRef > getAvailableDialects()
Return information about all available dialects in the registry in this context.
bool isMultithreadingEnabled()
Return true if multi-threading is enabled by the context.
void allowUnregisteredDialects(bool allow=true)
Enables creating operations in unregistered dialects.
bool allowsUnregisteredDialects()
Return true if we allow to create operation for unregistered dialects.
T * getLoadedDialect()
Get a registered IR dialect for the given derived dialect type.
Definition: MLIRContext.h:90
DiagnosticEngine & getDiagEngine()
Returns the diagnostic engine for this context.
std::function< void(function_ref< void()>, const tracing::Action &)> HandlerTy
Signatures for the action handler that can be registered with the context.
Definition: MLIRContext.h:254
MLIRContextImpl & getImpl()
Definition: MLIRContext.h:210
T * getOrLoadDialect()
Get (or create) a dialect for the given derived dialect type.
Definition: MLIRContext.h:97
void exitMultiThreadedExecution()
bool shouldPrintOpOnDiagnostic()
Return true if we should attach the operation to diagnostics emitted via Operation::emit.
void loadAllAvailableDialects()
Load all dialects available in the registry in this context.
NamedAttrList is array of NamedAttributes that tracks whether it is sorted and does some basic work t...
DictionaryAttr getDictionary(MLIRContext *context) const
Return a dictionary attribute for the underlying dictionary.
Attribute set(StringAttr name, Attribute value)
If the an attribute exists with the specified name, change it to the new value.
This is a pure-virtual base class that exposes the asmprinter hooks necessary to implement a custom p...
virtual void printGenericOp(Operation *op, bool printOpName=true)=0
Print the entire operation with the default generic assembly form.
Simple wrapper around a void* in order to express generically how to pass in op properties through AP...
Impl(StringRef, Dialect *dialect, TypeID typeID, detail::InterfaceMap interfaceMap)
StringRef getStringRef() const
Return the name of this operation. This always succeeds.
llvm::unique_function< ParseResult(OpAsmParser &, OperationState &)> ParseAssemblyFn
Impl * getImpl() const
Dialect * getDialect() const
Return the dialect this operation is registered to if the dialect is loaded in the context,...
OperationName(StringRef name, MLIRContext *context)
StringRef getDialectNamespace() const
Return the name of the dialect this operation is registered to.
Operation is the basic unit of execution within MLIR.
Definition: Operation.h:88
MLIRContext * getContext()
Return the context this operation is associated with.
Definition: Operation.h:216
OpaqueProperties getPropertiesStorage()
Returns the properties storage.
Definition: Operation.h:901
This is a "type erased" representation of a registered operation.
static void insert(Dialect &dialect)
Register a new operation in a Dialect object.
static std::optional< RegisteredOperationName > lookup(StringRef name, MLIRContext *ctx)
Lookup the registered operation information for the given operation.
A utility class to get or create instances of "storage classes".
This class provides an efficient unique identifier for a specific C++ type.
Definition: TypeID.h:104
An allocator for distinct attribute storage instances.
DistinctAttrStorage * allocate(Attribute referencedAttr)
Allocates a distinct attribute storage using a thread local bump pointer allocator to enable synchron...
This class provides an efficient mapping between a given Interface type, and a particular implementat...
An action is a specific action that is to be taken by the compiler, that can be toggled and controlle...
Definition: Action.h:38
AttrTypeReplacer.
llvm::unique_function< InFlightDiagnostic()> getDefaultDiagnosticEmitFn(MLIRContext *ctx)
Utility method to generate a callback that can be used to generate a diagnostic when checking the con...
Include the generated interface declarations.
InFlightDiagnostic emitError(Location loc)
Utility method to emit an error message using this location.
void registerMLIRContextCLOptions()
Register a set of useful command-line options that can be used to configure various flags within the ...
Definition: MLIRContext.cpp:90
static void getMaxDimAndSymbol(ArrayRef< AffineExprContainer > exprsList, int64_t &maxDim, int64_t &maxSym)
Calculates maximum dimension and symbol positions from the expressions in exprsLists and stores them ...
Definition: AffineMap.h:697
auto get(MLIRContext *context, Ts &&...params)
Helper method that injects context only if needed, this helps unify some of the attribute constructio...
LogicalResult foldHook(Operation *, ArrayRef< Attribute >, SmallVectorImpl< OpFoldResult > &) final
llvm::hash_code hashProperties(OpaqueProperties) final
void populateInherentAttrs(Operation *op, NamedAttrList &attrs) final
void deleteProperties(OpaqueProperties) final
LogicalResult setPropertiesFromAttr(OperationName, OpaqueProperties, Attribute, function_ref< InFlightDiagnostic()> emitError) final
void setInherentAttr(Operation *op, StringAttr name, Attribute value) final
void populateDefaultProperties(OperationName opName, OpaqueProperties properties) final
OperationName::ParseAssemblyFn getParseAssemblyFn() final
LogicalResult verifyRegionInvariants(Operation *) final
std::optional< Attribute > getInherentAttr(Operation *op, StringRef name) final
Implementation for properties.
void getCanonicalizationPatterns(RewritePatternSet &, MLIRContext *) final
void initProperties(OperationName opName, OpaqueProperties storage, OpaqueProperties init) final
void populateDefaultAttrs(const OperationName &, NamedAttrList &) final
void printAssembly(Operation *, OpAsmPrinter &, StringRef) final
LogicalResult verifyInvariants(Operation *) final
void copyProperties(OpaqueProperties, OpaqueProperties) final
LogicalResult verifyInherentAttrs(OperationName opName, NamedAttrList &attributes, function_ref< InFlightDiagnostic()> emitError) final
Attribute getPropertiesAsAttr(Operation *) final
bool compareProperties(OpaqueProperties, OpaqueProperties) final
A binary operation appearing in an affine expression.
An integer constant appearing in affine expression.
A dimensional or symbolic identifier appearing in an affine expression.
An attribute to store a distinct reference to another attribute.
Dialect * referencedDialect
If the string value contains a dialect namespace prefix (e.g.