27#include "mlir/Conversion/ArithToLLVM/ArithToLLVM.h"
28#include "mlir/Conversion/ControlFlowToLLVM/ControlFlowToLLVM.h"
29#include "mlir/Conversion/FuncToLLVM/ConvertFuncToLLVM.h"
30#include "mlir/Conversion/IndexToLLVM/IndexToLLVM.h"
31#include "mlir/Conversion/LLVMCommon/ConversionTarget.h"
32#include "mlir/Conversion/LLVMCommon/TypeConverter.h"
33#include "mlir/Conversion/SCFToControlFlow/SCFToControlFlow.h"
34#include "mlir/Dialect/ControlFlow/IR/ControlFlowOps.h"
35#include "mlir/Dialect/Func/IR/FuncOps.h"
36#include "mlir/Dialect/Index/IR/IndexOps.h"
37#include "mlir/Dialect/LLVMIR/FunctionCallUtils.h"
38#include "mlir/Dialect/LLVMIR/LLVMAttrs.h"
39#include "mlir/Dialect/LLVMIR/LLVMDialect.h"
40#include "mlir/Dialect/SCF/IR/SCF.h"
41#include "mlir/IR/Builders.h"
42#include "mlir/IR/BuiltinDialect.h"
43#include "mlir/Pass/Pass.h"
44#include "mlir/Transforms/DialectConversion.h"
45#include "llvm/Support/Debug.h"
46#include "llvm/Support/FormatVariadic.h"
50#define DEBUG_TYPE "lower-arc-to-llvm"
53#define GEN_PASS_DEF_LOWERARCTOLLVM
54#include "circt/Conversion/Passes.h.inc"
61using namespace runtime;
68 return modelName +
"_eval";
74 using OpConversionPattern::OpConversionPattern;
76 matchAndRewrite(arc::ModelOp op, OpAdaptor adaptor,
77 ConversionPatternRewriter &rewriter)
const final {
79 IRRewriter::InsertionGuard guard(rewriter);
80 rewriter.setInsertionPointToEnd(&op.getBodyBlock());
81 func::ReturnOp::create(rewriter, op.getLoc());
86 rewriter.getFunctionType(op.getBody().getArgumentTypes(), {});
88 mlir::func::FuncOp::create(rewriter, op.getLoc(), funcName, funcType);
89 rewriter.inlineRegionBefore(op.getRegion(), func.getBody(), func.end());
95struct AllocStorageOpLowering
97 using OpConversionPattern::OpConversionPattern;
99 matchAndRewrite(arc::AllocStorageOp op, OpAdaptor adaptor,
100 ConversionPatternRewriter &rewriter)
const final {
101 auto type = typeConverter->convertType(op.getType());
102 if (!op.getOffset().has_value())
104 rewriter.replaceOpWithNewOp<LLVM::GEPOp>(op, type, rewriter.getI8Type(),
106 LLVM::GEPArg(*op.getOffset()));
111template <
class ConcreteOp>
115 using OpAdaptor =
typename ConcreteOp::Adaptor;
118 matchAndRewrite(ConcreteOp op, OpAdaptor adaptor,
119 ConversionPatternRewriter &rewriter)
const final {
121 auto offsetAttr = op->template getAttrOfType<IntegerAttr>(
"offset");
124 Value ptr = LLVM::GEPOp::create(
125 rewriter, op->getLoc(), adaptor.getStorage().getType(),
126 rewriter.getI8Type(), adaptor.getStorage(),
127 LLVM::GEPArg(offsetAttr.getValue().getZExtValue()));
128 rewriter.replaceOp(op, ptr);
134 using OpConversionPattern::OpConversionPattern;
136 matchAndRewrite(arc::StateReadOp op, OpAdaptor adaptor,
137 ConversionPatternRewriter &rewriter)
const final {
139 if (isa<ArrayRefType>(op.getType())) {
140 rewriter.replaceOp(op, adaptor.getState());
144 auto type = typeConverter->convertType(op.getType());
145 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, type, adaptor.getState());
151 using OpConversionPattern::OpConversionPattern;
153 matchAndRewrite(arc::StateWriteOp op, OpAdaptor adaptor,
154 ConversionPatternRewriter &rewriter)
const final {
155 if (!isa<ArrayRefType>(op.getValue().getType())) {
156 rewriter.replaceOpWithNewOp<LLVM::StoreOp>(op, adaptor.getValue(),
161 int numBytes = op.getState().getType().getByteWidth();
162 Value size = LLVM::ConstantOp::create(rewriter, op.getLoc(),
163 rewriter.getI64Type(), numBytes);
164 rewriter.replaceOpWithNewOp<LLVM::MemcpyOp>(
165 op, adaptor.getState(), adaptor.getValue(), size,
false);
175 using OpConversionPattern::OpConversionPattern;
177 matchAndRewrite(arc::CurrentTimeOp op, OpAdaptor adaptor,
178 ConversionPatternRewriter &rewriter)
const final {
180 Value ptr = adaptor.getStorage();
181 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, rewriter.getI64Type(), ptr);
189struct ConstantTimeOpLowering
191 using OpConversionPattern::OpConversionPattern;
193 matchAndRewrite(llhd::ConstantTimeOp op, OpAdaptor adaptor,
194 ConversionPatternRewriter &rewriter)
const final {
195 auto attr = op.getValue();
196 if (attr.getDelta() != 0 || attr.getEpsilon() != 0)
197 return rewriter.notifyMatchFailure(
198 op,
"non-zero delta or epsilon time components are not supported");
199 uint64_t value = attr.getTime();
200 StringRef unit = attr.getTimeUnit();
204 else if (unit ==
"ps")
206 else if (unit ==
"ns")
207 scale = 1'000'000ULL;
208 else if (unit ==
"us")
209 scale = 1'000'000'000ULL;
210 else if (unit ==
"ms")
211 scale = 1'000'000'000'000ULL;
212 else if (unit ==
"s")
213 scale = 1'000'000'000'000'000ULL;
215 return rewriter.notifyMatchFailure(
216 op,
"time units smaller than `fs` are not supported");
217 if (value > std::numeric_limits<uint64_t>::max() / scale)
218 return rewriter.notifyMatchFailure(
219 op,
"time value does not fit into `i64` femtoseconds");
220 rewriter.replaceOpWithNewOp<LLVM::ConstantOp>(op, rewriter.getI64Type(),
228 using OpConversionPattern::OpConversionPattern;
230 matchAndRewrite(llhd::IntToTimeOp op, OpAdaptor adaptor,
231 ConversionPatternRewriter &rewriter)
const final {
232 rewriter.replaceOp(op, adaptor.getInput());
239 using OpConversionPattern::OpConversionPattern;
241 matchAndRewrite(llhd::TimeToIntOp op, OpAdaptor adaptor,
242 ConversionPatternRewriter &rewriter)
const final {
243 rewriter.replaceOp(op, adaptor.getInput());
253 using OpConversionPattern::OpConversionPattern;
255 matchAndRewrite(arc::AllocMemoryOp op, OpAdaptor adaptor,
256 ConversionPatternRewriter &rewriter)
const final {
257 auto offsetAttr = op->getAttrOfType<IntegerAttr>(
"offset");
260 Value ptr = LLVM::GEPOp::create(
261 rewriter, op.getLoc(), adaptor.getStorage().getType(),
262 rewriter.getI8Type(), adaptor.getStorage(),
263 LLVM::GEPArg(offsetAttr.getValue().getZExtValue()));
265 rewriter.replaceOp(op, ptr);
271 using OpConversionPattern::OpConversionPattern;
273 matchAndRewrite(arc::StorageGetOp op, OpAdaptor adaptor,
274 ConversionPatternRewriter &rewriter)
const final {
275 Value offset = LLVM::ConstantOp::create(
276 rewriter, op.getLoc(), rewriter.getI32Type(), op.getOffsetAttr());
277 Value ptr = LLVM::GEPOp::create(
278 rewriter, op.getLoc(), adaptor.getStorage().getType(),
279 rewriter.getI8Type(), adaptor.getStorage(), offset);
280 rewriter.replaceOp(op, ptr);
290static MemoryAccess prepareMemoryAccess(Location loc, Value memory,
291 Value address, MemoryType type,
292 ConversionPatternRewriter &rewriter) {
293 auto zextAddrType = rewriter.getIntegerType(
294 cast<IntegerType>(address.getType()).getWidth() + 1);
295 Value
addr = LLVM::ZExtOp::create(rewriter, loc, zextAddrType, address);
297 LLVM::ConstantOp::create(rewriter, loc, zextAddrType,
298 rewriter.getI32IntegerAttr(type.getNumWords()));
299 Value withinBounds = LLVM::ICmpOp::create(
300 rewriter, loc, LLVM::ICmpPredicate::ult, addr, addrLimit);
301 Value ptr = LLVM::GEPOp::create(
302 rewriter, loc, LLVM::LLVMPointerType::get(memory.getContext()),
303 rewriter.getIntegerType(type.getStride() * 8), memory, ValueRange{addr});
304 return {ptr, withinBounds};
308 using OpConversionPattern::OpConversionPattern;
310 matchAndRewrite(arc::MemoryReadOp op, OpAdaptor adaptor,
311 ConversionPatternRewriter &rewriter)
const final {
312 auto type = typeConverter->convertType(op.getType());
313 auto memoryType = cast<MemoryType>(op.getMemory().getType());
315 prepareMemoryAccess(op.getLoc(), adaptor.getMemory(),
316 adaptor.getAddress(), memoryType, rewriter);
320 rewriter.replaceOpWithNewOp<scf::IfOp>(
321 op, access.withinBounds,
322 [&](
auto &builder,
auto loc) {
323 Value loadOp = LLVM::LoadOp::create(
324 builder, loc, memoryType.getWordType(), access.ptr);
325 scf::YieldOp::create(builder, loc, loadOp);
327 [&](
auto &builder,
auto loc) {
328 Value zeroValue = LLVM::ConstantOp::create(
329 builder, loc, type, builder.getI64IntegerAttr(0));
330 scf::YieldOp::create(builder, loc, zeroValue);
337 using OpConversionPattern::OpConversionPattern;
339 matchAndRewrite(arc::MemoryWriteOp op, OpAdaptor adaptor,
340 ConversionPatternRewriter &rewriter)
const final {
341 auto access = prepareMemoryAccess(
342 op.getLoc(), adaptor.getMemory(), adaptor.getAddress(),
343 cast<MemoryType>(op.getMemory().getType()), rewriter);
344 auto enable = access.withinBounds;
347 rewriter.replaceOpWithNewOp<scf::IfOp>(
348 op, enable, [&](
auto &builder,
auto loc) {
349 LLVM::StoreOp::create(builder, loc, adaptor.getData(), access.ptr);
350 scf::YieldOp::create(builder, loc);
358 using OpConversionPattern::OpConversionPattern;
360 matchAndRewrite(seq::ClockGateOp op, OpAdaptor adaptor,
361 ConversionPatternRewriter &rewriter)
const final {
362 rewriter.replaceOpWithNewOp<LLVM::AndOp>(op, adaptor.getInput(),
363 adaptor.getEnable());
370 using OpConversionPattern::OpConversionPattern;
372 matchAndRewrite(seq::ClockInverterOp op, OpAdaptor adaptor,
373 ConversionPatternRewriter &rewriter)
const final {
374 auto constTrue = LLVM::ConstantOp::create(rewriter, op->getLoc(),
375 rewriter.getI1Type(), 1);
376 rewriter.replaceOpWithNewOp<LLVM::XOrOp>(op, adaptor.getInput(), constTrue);
382 using OpConversionPattern::OpConversionPattern;
384 matchAndRewrite(arc::ZeroCountOp op, OpAdaptor adaptor,
385 ConversionPatternRewriter &rewriter)
const override {
387 IntegerAttr isZeroPoison = rewriter.getBoolAttr(
true);
389 if (op.getPredicate() == arc::ZeroCountPredicate::leading) {
390 rewriter.replaceOpWithNewOp<LLVM::CountLeadingZerosOp>(
391 op, adaptor.getInput().getType(), adaptor.getInput(), isZeroPoison);
395 rewriter.replaceOpWithNewOp<LLVM::CountTrailingZerosOp>(
396 op, adaptor.getInput().getType(), adaptor.getInput(), isZeroPoison);
402 using OpConversionPattern::OpConversionPattern;
404 matchAndRewrite(seq::ConstClockOp op, OpAdaptor adaptor,
405 ConversionPatternRewriter &rewriter)
const override {
406 rewriter.replaceOpWithNewOp<LLVM::ConstantOp>(
407 op, rewriter.getI1Type(),
static_cast<int64_t
>(op.getValue()));
412template <
typename OpTy>
415 using OpAdaptor =
typename OpTy::Adaptor;
417 matchAndRewrite(OpTy op, OpAdaptor adaptor,
418 ConversionPatternRewriter &rewriter)
const override {
419 rewriter.replaceOp(op, adaptor.getInput());
433 size_t numStateBytes;
434 llvm::DenseMap<StringRef, StateInfo> states;
435 mlir::FlatSymbolRefAttr initialFnSymbol;
436 mlir::FlatSymbolRefAttr finalFnSymbol;
439template <
typename OpTy>
441 ModelAwarePattern(
const TypeConverter &typeConverter, MLIRContext *
context,
442 llvm::DenseMap<StringRef, ModelInfoMap> &modelInfo)
444 modelInfo(modelInfo) {}
447 Value createPtrToPortState(ConversionPatternRewriter &rewriter, Location loc,
448 Value state,
const StateInfo &port)
const {
449 MLIRContext *ctx = rewriter.getContext();
450 return LLVM::GEPOp::create(rewriter, loc, LLVM::LLVMPointerType::get(ctx),
451 IntegerType::get(ctx, 8), state,
452 LLVM::GEPArg(port.offset));
455 llvm::DenseMap<StringRef, ModelInfoMap> &modelInfo;
460struct SimInstantiateOpLowering
461 :
public ModelAwarePattern<arc::SimInstantiateOp> {
462 using ModelAwarePattern::ModelAwarePattern;
465 matchAndRewrite(arc::SimInstantiateOp op, OpAdaptor adaptor,
466 ConversionPatternRewriter &rewriter)
const final {
467 auto modelIt = modelInfo.find(
468 cast<SimModelInstanceType>(op.getBody().getArgument(0).getType())
471 ModelInfoMap &model = modelIt->second;
473 bool useRuntime = op.getRuntimeModel().has_value();
475 ModuleOp moduleOp = op->getParentOfType<ModuleOp>();
479 ConversionPatternRewriter::InsertionGuard guard(rewriter);
483 Type convertedIndex = typeConverter->convertType(rewriter.getIndexType());
484 Location loc = op.getLoc();
489 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
493 if (op.getRuntimeArgs().has_value()) {
494 SmallVector<int8_t> argStringVec(op.getRuntimeArgsAttr().begin(),
495 op.getRuntimeArgsAttr().end());
496 argStringVec.push_back(
'\0');
497 auto strAttr = mlir::DenseElementsAttr::get(
498 mlir::RankedTensorType::get({(int64_t)argStringVec.size()},
499 rewriter.getI8Type()),
500 llvm::ArrayRef(argStringVec));
502 auto arrayCst = LLVM::ConstantOp::create(
504 LLVM::LLVMArrayType::get(rewriter.getI8Type(), argStringVec.size()),
506 auto cst1 = LLVM::ConstantOp::create(rewriter, loc,
507 rewriter.getI32IntegerAttr(1));
508 runtimeArgs = LLVM::AllocaOp::create(rewriter, loc, ptrTy,
509 arrayCst.getType(), cst1);
510 LLVM::LifetimeStartOp::create(rewriter, loc, runtimeArgs);
511 LLVM::StoreOp::create(rewriter, loc, arrayCst, runtimeArgs);
513 runtimeArgs = LLVM::ZeroOp::create(rewriter, loc, ptrTy).getResult();
516 auto rtModelPtr = LLVM::AddressOfOp::create(rewriter, loc, ptrTy,
517 op.getRuntimeModelAttr())
520 LLVM::CallOp::create(rewriter, loc, {ptrTy},
521 runtime::APICallbacks::symNameAllocInstance,
522 {rtModelPtr, runtimeArgs})
525 if (op.getRuntimeArgs().has_value())
526 LLVM::LifetimeEndOp::create(rewriter, loc, runtimeArgs);
530 FailureOr<LLVM::LLVMFuncOp> mallocFunc =
531 LLVM::lookupOrCreateMallocFn(rewriter, moduleOp, convertedIndex);
532 if (failed(mallocFunc))
535 Value numStateBytes = LLVM::ConstantOp::create(
536 rewriter, loc, convertedIndex, model.numStateBytes);
537 allocated = LLVM::CallOp::create(rewriter, loc, mallocFunc.value(),
538 ValueRange{numStateBytes})
541 LLVM::ConstantOp::create(rewriter, loc, rewriter.getI8Type(), 0);
542 LLVM::MemsetOp::create(rewriter, loc, allocated, zero, numStateBytes,
547 if (model.initialFnSymbol) {
548 auto initialFnType = LLVM::LLVMFunctionType::get(
549 LLVM::LLVMVoidType::get(op.getContext()),
550 {LLVM::LLVMPointerType::get(op.getContext())});
551 LLVM::CallOp::create(rewriter, loc, initialFnType, model.initialFnSymbol,
552 ValueRange{allocated});
557 LLVM::CallOp::create(rewriter, loc, TypeRange{},
558 runtime::APICallbacks::symNameOnInitialized,
562 rewriter.inlineBlockBefore(&adaptor.getBody().getBlocks().front(), op,
566 if (model.finalFnSymbol) {
567 auto finalFnType = LLVM::LLVMFunctionType::get(
568 LLVM::LLVMVoidType::get(op.getContext()),
569 {LLVM::LLVMPointerType::get(op.getContext())});
570 LLVM::CallOp::create(rewriter, loc, finalFnType, model.finalFnSymbol,
571 ValueRange{allocated});
575 LLVM::CallOp::create(rewriter, loc, TypeRange{},
576 runtime::APICallbacks::symNameDeleteInstance,
579 FailureOr<LLVM::LLVMFuncOp> freeFunc =
580 LLVM::lookupOrCreateFreeFn(rewriter, moduleOp);
581 if (failed(freeFunc))
584 LLVM::CallOp::create(rewriter, loc, freeFunc.value(),
585 ValueRange{allocated});
588 rewriter.eraseOp(op);
593struct SimSetInputOpLowering :
public ModelAwarePattern<arc::SimSetInputOp> {
594 using ModelAwarePattern::ModelAwarePattern;
597 matchAndRewrite(arc::SimSetInputOp op, OpAdaptor adaptor,
598 ConversionPatternRewriter &rewriter)
const final {
600 modelInfo.find(cast<SimModelInstanceType>(op.getInstance().getType())
603 ModelInfoMap &model = modelIt->second;
605 auto portIt = model.states.find(op.getInput());
606 if (portIt == model.states.end()) {
609 rewriter.eraseOp(op);
613 StateInfo &port = portIt->second;
614 Value statePtr = createPtrToPortState(rewriter, op.getLoc(),
615 adaptor.getInstance(), port);
616 rewriter.replaceOpWithNewOp<LLVM::StoreOp>(op, adaptor.getValue(),
623struct SimGetPortOpLowering :
public ModelAwarePattern<arc::SimGetPortOp> {
624 using ModelAwarePattern::ModelAwarePattern;
627 matchAndRewrite(arc::SimGetPortOp op, OpAdaptor adaptor,
628 ConversionPatternRewriter &rewriter)
const final {
630 modelInfo.find(cast<SimModelInstanceType>(op.getInstance().getType())
633 ModelInfoMap &model = modelIt->second;
635 auto type = typeConverter->convertType(op.getValue().getType());
638 auto portIt = model.states.find(op.getPort());
639 if (portIt == model.states.end()) {
642 rewriter.replaceOpWithNewOp<LLVM::ConstantOp>(op, type, 0);
646 StateInfo &port = portIt->second;
647 Value statePtr = createPtrToPortState(rewriter, op.getLoc(),
648 adaptor.getInstance(), port);
649 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, type, statePtr);
655struct SimStepOpLowering :
public ModelAwarePattern<arc::SimStepOp> {
656 using ModelAwarePattern::ModelAwarePattern;
659 matchAndRewrite(arc::SimStepOp op, OpAdaptor adaptor,
660 ConversionPatternRewriter &rewriter)
const final {
661 StringRef modelName = cast<SimModelInstanceType>(op.getInstance().getType())
665 if (adaptor.getTimePostIncrement()) {
667 OpBuilder::InsertionGuard g(rewriter);
668 rewriter.setInsertionPointAfter(op);
670 arc::SimGetTimeOp::create(rewriter, op.getLoc(), op.getInstance());
671 auto newTime = LLVM::AddOp::create(rewriter, op.getLoc(), oldTime,
672 adaptor.getTimePostIncrement());
673 arc::SimSetTimeOp::create(rewriter, op.getLoc(), op.getInstance(),
677 StringAttr evalFunc =
679 rewriter.replaceOpWithNewOp<LLVM::CallOp>(op, mlir::TypeRange(), evalFunc,
680 adaptor.getInstance());
689 using OpConversionPattern::OpConversionPattern;
692 matchAndRewrite(arc::SimGetTimeOp op, OpAdaptor adaptor,
693 ConversionPatternRewriter &rewriter)
const final {
695 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, rewriter.getI64Type(),
696 adaptor.getInstance());
704 using OpConversionPattern::OpConversionPattern;
707 matchAndRewrite(arc::SimSetTimeOp op, OpAdaptor adaptor,
708 ConversionPatternRewriter &rewriter)
const final {
710 rewriter.replaceOpWithNewOp<LLVM::StoreOp>(op, adaptor.getTime(),
711 adaptor.getInstance());
718struct SimGetNextWakeupOpLowering
720 using OpConversionPattern::OpConversionPattern;
723 matchAndRewrite(arc::SimGetNextWakeupOp op, OpAdaptor adaptor,
724 ConversionPatternRewriter &rewriter)
const final {
725 auto loc = op.getLoc();
726 auto ptrType = LLVM::LLVMPointerType::get(rewriter.getContext());
727 Value slotPtr = LLVM::GEPOp::create(
728 rewriter, loc, ptrType, rewriter.getI8Type(), adaptor.getInstance(),
729 ArrayRef<LLVM::GEPArg>{arc::kNextWakeupOffset});
730 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, rewriter.getI64Type(),
739 Value getOrCreate(OpBuilder &b, StringRef formatStr) {
740 auto it = cache.find(formatStr);
741 if (it != cache.end()) {
742 return LLVM::AddressOfOp::create(b, b.getUnknownLoc(), it->second);
745 Location loc = b.getUnknownLoc();
746 LLVM::GlobalOp global;
748 OpBuilder::InsertionGuard guard(b);
750 b.getInsertionBlock()->getParent()->getParentOfType<ModuleOp>();
751 b.setInsertionPointToStart(m.getBody());
753 SmallVector<char> strVec(formatStr.begin(), formatStr.end());
756 auto name = llvm::formatv(
"_arc_str_{0}", cache.size()).str();
757 auto globalType = LLVM::LLVMArrayType::get(b.getI8Type(), strVec.size());
758 global = LLVM::GlobalOp::create(b, loc, globalType,
true,
759 LLVM::Linkage::Internal,
760 name, b.getStringAttr(strVec),
764 cache[formatStr] = global;
765 return LLVM::AddressOfOp::create(b, loc, global);
769 llvm::StringMap<LLVM::GlobalOp> cache;
772FailureOr<LLVM::CallOp> emitPrintfCall(OpBuilder &builder, Location loc,
773 StringCache &cache, StringRef formatStr,
776 builder.getInsertionBlock()->getParent()->getParentOfType<ModuleOp>();
778 MLIRContext *ctx = builder.getContext();
779 auto printfFunc = LLVM::lookupOrCreateFn(builder, moduleOp,
"printf",
780 LLVM::LLVMPointerType::get(ctx),
781 LLVM::LLVMVoidType::get(ctx),
true);
782 if (failed(printfFunc))
785 Value formatStrPtr = cache.getOrCreate(builder, formatStr);
786 SmallVector<Value> argsVec(1, formatStrPtr);
787 argsVec.append(args.begin(), args.end());
788 return LLVM::CallOp::create(builder, loc, printfFunc.value(), argsVec);
794struct SimEmitValueOpLowering
796 SimEmitValueOpLowering(
const TypeConverter &typeConverter,
797 MLIRContext *
context, StringCache &formatStringCache)
799 formatStringCache(formatStringCache) {}
802 matchAndRewrite(arc::SimEmitValueOp op, OpAdaptor adaptor,
803 ConversionPatternRewriter &rewriter)
const final {
804 auto valueType = dyn_cast<IntegerType>(adaptor.getValue().getType());
808 Location loc = op.getLoc();
810 ModuleOp moduleOp = op->getParentOfType<ModuleOp>();
814 SmallVector<Value> printfVariadicArgs;
815 SmallString<16> printfFormatStr;
816 int remainingBits = valueType.getWidth();
817 Value value = adaptor.getValue();
821 constexpr llvm::StringRef intFormatter =
"llx";
822 auto intType = IntegerType::get(getContext(), 64);
823 Value shiftValue = LLVM::ConstantOp::create(
824 rewriter, loc, rewriter.getIntegerAttr(valueType, intType.getWidth()));
826 if (valueType.getWidth() < intType.getWidth()) {
827 int width = llvm::divideCeil(valueType.getWidth(), 4);
828 printfFormatStr = llvm::formatv(
"%0{0}{1}", width, intFormatter);
829 printfVariadicArgs.push_back(
830 LLVM::ZExtOp::create(rewriter, loc, intType, value));
835 int otherChunkWidth = intType.getWidth() / 4;
836 int firstChunkWidth =
837 llvm::divideCeil(valueType.getWidth() % intType.getWidth(), 4);
838 if (firstChunkWidth == 0) {
839 firstChunkWidth = otherChunkWidth;
842 std::string firstChunkFormat =
843 llvm::formatv(
"%0{0}{1}", firstChunkWidth, intFormatter);
844 std::string otherChunkFormat =
845 llvm::formatv(
"%0{0}{1}", otherChunkWidth, intFormatter);
847 for (
int i = 0; remainingBits > 0; ++i) {
850 printfVariadicArgs.push_back(
851 LLVM::TruncOp::create(rewriter, loc, intType, value));
854 printfFormatStr.append(i == 0 ? firstChunkFormat : otherChunkFormat);
857 LLVM::LShrOp::create(rewriter, loc, value, shiftValue).getResult();
858 remainingBits -= intType.getWidth();
862 std::reverse(printfVariadicArgs.begin(), printfVariadicArgs.end());
864 SmallString<16> formatStr = adaptor.getValueName();
865 formatStr.append(
" = ");
866 formatStr.append(printfFormatStr);
867 formatStr.append(
"\n");
869 auto callOp = emitPrintfCall(rewriter, op->getLoc(), formatStringCache,
870 formatStr, printfVariadicArgs);
873 rewriter.replaceOp(op, *callOp);
878 StringCache &formatStringCache;
887 SmallVector<FmtDescriptor> descriptors;
888 SmallVector<Value> args;
895static Value reg2mem(ConversionPatternRewriter &rewriter, Location loc,
898 int64_t origBitwidth = cast<IntegerType>(value.getType()).getWidth();
899 int64_t bitwidth = llvm::divideCeil(origBitwidth, 64) * 64;
900 int64_t numWords = bitwidth / 64;
903 LLVM::ConstantOp alloca_size =
904 LLVM::ConstantOp::create(rewriter, loc, rewriter.getI32Type(), numWords);
905 auto ptrType = LLVM::LLVMPointerType::get(rewriter.getContext());
906 auto allocaOp = LLVM::AllocaOp::create(rewriter, loc, ptrType,
907 rewriter.getI64Type(), alloca_size);
908 LLVM::LifetimeStartOp::create(rewriter, loc, allocaOp);
912 for (int64_t wordIdx = 0; wordIdx < numWords; ++wordIdx) {
913 Value cst = LLVM::ConstantOp::create(
914 rewriter, loc, rewriter.getIntegerType(origBitwidth), wordIdx * 64);
915 Value v = LLVM::LShrOp::create(rewriter, loc, value, cst);
916 if (origBitwidth > 64) {
917 v = LLVM::TruncOp::create(rewriter, loc, rewriter.getI64Type(), v);
918 }
else if (origBitwidth < 64) {
919 v = LLVM::ZExtOp::create(rewriter, loc, rewriter.getI64Type(), v);
921 Value gep = LLVM::GEPOp::create(rewriter, loc, ptrType,
922 rewriter.getI64Type(), allocaOp, {wordIdx});
923 LLVM::StoreOp::create(rewriter, loc, v, gep);
930static FailureOr<FormatInfo>
931foldFormatString(ConversionPatternRewriter &rewriter, Value fstringValue,
932 StringCache &cache) {
933 Operation *op = fstringValue.getDefiningOp();
934 return llvm::TypeSwitch<Operation *, FailureOr<FormatInfo>>(op)
935 .Case<sim::FormatCharOp>(
936 [&](sim::FormatCharOp op) -> FailureOr<FormatInfo> {
937 FmtDescriptor
d = FmtDescriptor::createChar();
938 return FormatInfo{{
d}, {op.getValue()}};
940 .Case<sim::FormatDecOp>([&](sim::FormatDecOp op)
941 -> FailureOr<FormatInfo> {
942 FmtDescriptor
d = FmtDescriptor::createInt(
943 op.getValue().getType().getWidth(), 10, op.getIsLeftAligned(),
944 op.getSpecifierWidth().value_or(-1), op.getPaddingChar(),
false,
946 return FormatInfo{{
d}, {reg2mem(rewriter, op.getLoc(), op.getValue())}};
948 .Case<sim::FormatHexOp>([&](sim::FormatHexOp op)
949 -> FailureOr<FormatInfo> {
950 FmtDescriptor
d = FmtDescriptor::createInt(
951 op.getValue().getType().getWidth(), 16, op.getIsLeftAligned(),
952 op.getSpecifierWidth().value_or(-1), op.getPaddingChar(),
953 op.getIsHexUppercase(),
false);
954 return FormatInfo{{
d}, {reg2mem(rewriter, op.getLoc(), op.getValue())}};
956 .Case<sim::FormatOctOp>([&](sim::FormatOctOp op)
957 -> FailureOr<FormatInfo> {
958 FmtDescriptor
d = FmtDescriptor::createInt(
959 op.getValue().getType().getWidth(), 8, op.getIsLeftAligned(),
960 op.getSpecifierWidth().value_or(-1), op.getPaddingChar(),
false,
962 return FormatInfo{{
d}, {reg2mem(rewriter, op.getLoc(), op.getValue())}};
964 .Case<sim::FormatLiteralOp>(
965 [&](sim::FormatLiteralOp op) -> FailureOr<FormatInfo> {
966 if (op.getLiteral().size() < 8 &&
967 op.getLiteral().find(
'\0') == StringRef::npos) {
970 FmtDescriptor::createSmallLiteral(op.getLiteral());
971 return FormatInfo{{
d}, {}};
974 FmtDescriptor::createLiteral(op.getLiteral().size());
975 Value value = cache.getOrCreate(rewriter, op.getLiteral());
976 return FormatInfo{{
d}, {value}};
978 .Case<sim::FormatStringConcatOp>(
979 [&](sim::FormatStringConcatOp op) -> FailureOr<FormatInfo> {
980 auto fmt = foldFormatString(rewriter, op.getInputs()[0], cache);
983 for (
auto input : op.getInputs().drop_front()) {
984 auto next = foldFormatString(rewriter, input, cache);
987 fmt->descriptors.append(next->descriptors);
988 fmt->args.append(next->args);
993 [](Operation *op) -> FailureOr<FormatInfo> {
return failure(); });
996FailureOr<LLVM::CallOp> emitFmtCall(OpBuilder &builder, Location loc,
997 StringCache &stringCache,
998 ArrayRef<FmtDescriptor> descriptors,
1001 builder.getInsertionBlock()->getParent()->getParentOfType<ModuleOp>();
1003 MLIRContext *ctx = builder.getContext();
1004 auto func = LLVM::lookupOrCreateFn(
1005 builder, moduleOp, runtime::APICallbacks::symNameFormat,
1006 LLVM::LLVMPointerType::get(ctx), LLVM::LLVMVoidType::get(ctx),
true);
1010 StringRef rawDescriptors(
reinterpret_cast<const char *
>(descriptors.data()),
1011 descriptors.size() *
sizeof(FmtDescriptor));
1012 Value fmtPtr = stringCache.getOrCreate(builder, rawDescriptors);
1014 SmallVector<Value> argsVec(1, fmtPtr);
1015 argsVec.append(args.begin(), args.end());
1016 auto result = LLVM::CallOp::create(builder, loc, func.value(), argsVec);
1018 for (Value arg : args) {
1019 Operation *definingOp = arg.getDefiningOp();
1020 if (
auto alloca = dyn_cast_if_present<LLVM::AllocaOp>(definingOp)) {
1021 LLVM::LifetimeEndOp::create(builder, loc, arg);
1028struct SimPrintFormattedProcOpLowering
1030 SimPrintFormattedProcOpLowering(
const TypeConverter &typeConverter,
1032 StringCache &stringCache)
1034 stringCache(stringCache) {}
1037 matchAndRewrite(sim::PrintFormattedProcOp op, OpAdaptor adaptor,
1038 ConversionPatternRewriter &rewriter)
const override {
1039 auto formatInfo = foldFormatString(rewriter, op.getInput(), stringCache);
1040 if (failed(formatInfo))
1041 return rewriter.notifyMatchFailure(op,
"unsupported format string");
1044 formatInfo->descriptors.push_back(FmtDescriptor());
1046 auto result = emitFmtCall(rewriter, op.getLoc(), stringCache,
1047 formatInfo->descriptors, formatInfo->args);
1050 rewriter.replaceOp(op, result.value());
1055 StringCache &stringCache;
1059 using OpConversionPattern::OpConversionPattern;
1062 matchAndRewrite(arc::TerminateOp op, OpAdaptor adaptor,
1063 ConversionPatternRewriter &rewriter)
const override {
1064 auto loc = op.getLoc();
1066 auto i8Type = rewriter.getI8Type();
1067 auto ptrType = LLVM::LLVMPointerType::get(rewriter.getContext());
1069 Value flagPtr = LLVM::GEPOp::create(
1070 rewriter, loc, ptrType, i8Type, adaptor.getStorage(),
1071 ArrayRef<LLVM::GEPArg>{arc::kTerminateFlagOffset});
1073 uint8_t statusCode = op.getSuccess() ? 1 : 2;
1074 Value codeVal = LLVM::ConstantOp::create(
1075 rewriter, loc, i8Type, rewriter.getI8IntegerAttr(statusCode));
1077 LLVM::StoreOp::create(rewriter, loc, codeVal, flagPtr);
1079 rewriter.eraseOp(op);
1086struct GetNextWakeupOpLowering
1088 using OpConversionPattern::OpConversionPattern;
1091 matchAndRewrite(arc::GetNextWakeupOp op, OpAdaptor adaptor,
1092 ConversionPatternRewriter &rewriter)
const override {
1093 auto loc = op.getLoc();
1094 auto ptrType = LLVM::LLVMPointerType::get(rewriter.getContext());
1095 Value slotPtr = LLVM::GEPOp::create(
1096 rewriter, loc, ptrType, rewriter.getI8Type(), adaptor.getStorage(),
1097 ArrayRef<LLVM::GEPArg>{arc::kNextWakeupOffset});
1098 rewriter.replaceOpWithNewOp<LLVM::LoadOp>(op, rewriter.getI64Type(),
1106struct SetNextWakeupOpLowering
1108 using OpConversionPattern::OpConversionPattern;
1111 matchAndRewrite(arc::SetNextWakeupOp op, OpAdaptor adaptor,
1112 ConversionPatternRewriter &rewriter)
const override {
1113 auto loc = op.getLoc();
1114 auto ptrType = LLVM::LLVMPointerType::get(rewriter.getContext());
1115 Value slotPtr = LLVM::GEPOp::create(
1116 rewriter, loc, ptrType, rewriter.getI8Type(), adaptor.getStorage(),
1117 ArrayRef<LLVM::GEPArg>{arc::kNextWakeupOffset});
1118 rewriter.replaceOpWithNewOp<LLVM::StoreOp>(op, adaptor.getTime(), slotPtr);
1125static LogicalResult
convert(arc::ExecuteOp op, arc::ExecuteOp::Adaptor adaptor,
1126 ConversionPatternRewriter &rewriter,
1127 const TypeConverter &converter) {
1129 if (failed(rewriter.convertRegionTypes(&op.getBody(), converter)))
1135 auto *blockBefore = rewriter.getInsertionBlock();
1137 rewriter.splitBlock(blockBefore, rewriter.getInsertionPoint());
1140 rewriter.setInsertionPointToEnd(blockBefore);
1141 mlir::cf::BranchOp::create(rewriter, op.getLoc(), &op.getBody().front(),
1142 adaptor.getInputs());
1146 for (
auto &block : op.getBody()) {
1147 auto outputOp = dyn_cast<arc::OutputOp>(block.getTerminator());
1150 rewriter.setInsertionPointToEnd(&block);
1151 rewriter.replaceOpWithNewOp<mlir::cf::BranchOp>(outputOp, blockAfter,
1152 outputOp.getOperands());
1156 rewriter.inlineRegionBefore(op.getBody(), blockAfter);
1160 SmallVector<Value> args;
1161 args.reserve(op.getNumResults());
1162 for (
auto result : op.getResults())
1163 args.push_back(blockAfter->addArgument(result.getType(), result.getLoc()));
1164 rewriter.replaceOp(op, args);
1165 auto conversion = converter.convertBlockSignature(blockAfter);
1168 rewriter.applySignatureConversion(blockAfter, *conversion, &converter);
1176template <typename T, typename = std::enable_if_t<std::is_integral<T>::value>>
1177static LLVM::GlobalOp
1179 SmallVectorImpl<T> &data,
1180 unsigned alignment =
alignof(T)) {
1181 auto intType = builder.getIntegerType(8 *
sizeof(T));
1182 Attribute denseAttr = mlir::DenseElementsAttr::get(
1183 mlir::RankedTensorType::get({(int64_t)data.size()}, intType),
1184 llvm::ArrayRef(data));
1185 auto globalOp = LLVM::GlobalOp::create(
1186 builder, loc, LLVM::LLVMArrayType::get(intType, data.size()),
1187 true, LLVM::Linkage::Internal,
1188 builder.getStringAttr(symName), denseAttr);
1189 globalOp.setAlignmentAttr(builder.getI64IntegerAttr(alignment));
1194template <
typename T>
1195static LLVM::GlobalOp
1198 SmallVectorImpl<T> &array) {
1200 static_assert(std::is_standard_layout<T>(),
1201 "Runtime struct must have standard layout");
1202 int64_t numBytes =
sizeof(T) * array.size();
1203 Attribute denseAttr = mlir::DenseElementsAttr::get(
1204 mlir::RankedTensorType::get({numBytes}, builder.getI8Type()),
1205 llvm::ArrayRef(
reinterpret_cast<uint8_t *
>(array.data()), numBytes));
1206 auto globalOp = LLVM::GlobalOp::create(
1207 builder, loc, LLVM::LLVMArrayType::get(builder.getI8Type(), numBytes),
1208 true, LLVM::Linkage::Internal,
1209 builder.getStringAttr(symName), denseAttr,
alignof(T));
1215 using OpConversionPattern::OpConversionPattern;
1222 ConversionPatternRewriter &rewriter)
const {
1223 if (!op.getTraceTaps().has_value() || op.getTraceTaps()->empty())
1226 SmallVector<char> namesArray;
1227 SmallVector<ArcTraceTap> tapArray;
1228 tapArray.reserve(op.getTraceTaps()->size());
1229 for (
auto attr : op.getTraceTapsAttr()) {
1230 auto tap = cast<TraceTapAttr>(attr);
1231 assert(!tap.getNames().empty() &&
1232 "Expected trace tap to have at least one name");
1233 for (
auto alias : tap.getNames()) {
1234 auto aliasStr = cast<StringAttr>(alias);
1235 namesArray.append(aliasStr.begin(), aliasStr.end());
1236 namesArray.push_back(
'\0');
1240 tapStruct.
nameOffset = namesArray.size() - 1;
1241 tapStruct.
typeBits = tap.getSigType().getValue().getIntOrFloatBitWidth();
1243 tapArray.emplace_back(tapStruct);
1245 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1247 rewriter, op.getLoc(),
"_arc_tap_names_" + op.getName(), namesArray);
1249 rewriter, op.getLoc(),
"_arc_trace_taps_" + op.getName(), tapArray);
1259 auto traceInfoStructType = LLVM::LLVMStructType::getLiteral(
1261 {rewriter.getI64Type(), ptrTy, ptrTy, rewriter.getI64Type()});
1263 "Unexpected size of ArcModelTraceInfo struct");
1265 auto globalSymName =
1266 rewriter.getStringAttr(
"_arc_trace_info_" + op.getName());
1267 auto traceInfoGlobalOp = LLVM::GlobalOp::create(
1268 rewriter, op.getLoc(), traceInfoStructType,
1269 false, LLVM::Linkage::Internal, globalSymName,
1271 OpBuilder::InsertionGuard g(rewriter);
1274 Region &initRegion = traceInfoGlobalOp.getInitializerRegion();
1275 Block *initBlock = rewriter.createBlock(&initRegion);
1276 rewriter.setInsertionPointToStart(initBlock);
1278 auto numTraceTapsCst = LLVM::ConstantOp::create(
1279 rewriter, op.getLoc(), rewriter.getI64IntegerAttr(tapArray.size()));
1280 auto traceTapArrayAddr =
1281 LLVM::AddressOfOp::create(rewriter, op.getLoc(), traceTapsArrayGlobal);
1282 auto tapNameArrayAddr =
1283 LLVM::AddressOfOp::create(rewriter, op.getLoc(), namesGlobal);
1284 auto bufferCapacityCst = LLVM::ConstantOp::create(
1285 rewriter, op.getLoc(),
1286 rewriter.getI64IntegerAttr(runtime::defaultTraceBufferCapacity));
1289 LLVM::PoisonOp::create(rewriter, op.getLoc(), traceInfoStructType);
1293 LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1294 numTraceTapsCst, ArrayRef<int64_t>{0});
1296 "Unexpected offset of field numTraceTaps");
1299 LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1300 traceTapArrayAddr, ArrayRef<int64_t>{1});
1302 "Unexpected offset of field traceTaps");
1305 LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1306 tapNameArrayAddr, ArrayRef<int64_t>{2});
1308 "Unexpected offset of field traceTapNames");
1311 LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1312 bufferCapacityCst, ArrayRef<int64_t>{3});
1314 "Unexpected offset of field traceBufferCapacity");
1315 LLVM::ReturnOp::create(rewriter, op.getLoc(), initStruct);
1317 return traceInfoGlobalOp;
1323 ConversionPatternRewriter &rewriter)
const final {
1325 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1326 auto modelInfoStructType = LLVM::LLVMStructType::getLiteral(
1328 {rewriter.getI64Type(), rewriter.getI64Type(), ptrTy, ptrTy});
1330 "Unexpected size of ArcRuntimeModelInfo struct");
1332 rewriter.setInsertionPoint(op);
1336 SmallVector<char, 16> modNameArray(op.getName().begin(),
1337 op.getName().end());
1338 modNameArray.push_back(
'\0');
1339 auto nameGlobalType =
1340 LLVM::LLVMArrayType::get(rewriter.getI8Type(), modNameArray.size());
1341 auto globalSymName =
1342 rewriter.getStringAttr(
"_arc_mod_name_" + op.getName());
1343 auto nameGlobal = LLVM::GlobalOp::create(
1344 rewriter, op.getLoc(), nameGlobalType,
true,
1345 LLVM::Linkage::Internal,
1346 globalSymName, rewriter.getStringAttr(modNameArray),
1353 auto modInfoGlobalOp =
1354 LLVM::GlobalOp::create(rewriter, op.getLoc(), modelInfoStructType,
1355 false, LLVM::Linkage::External,
1356 op.getSymName(), Attribute{});
1359 Region &initRegion = modInfoGlobalOp.getInitializerRegion();
1360 Block *initBlock = rewriter.createBlock(&initRegion);
1361 rewriter.setInsertionPointToStart(initBlock);
1362 auto apiVersionCst = LLVM::ConstantOp::create(
1364 auto numStateBytesCst = LLVM::ConstantOp::create(rewriter, op.getLoc(),
1365 op.getNumStateBytesAttr());
1367 LLVM::AddressOfOp::create(rewriter, op.getLoc(), nameGlobal);
1369 if (traceInfoGlobal)
1371 LLVM::AddressOfOp::create(rewriter, op.getLoc(), traceInfoGlobal);
1373 traceInfoPtr = LLVM::ZeroOp::create(rewriter, op.getLoc(), ptrTy);
1376 LLVM::PoisonOp::create(rewriter, op.getLoc(), modelInfoStructType);
1379 initStruct = LLVM::InsertValueOp::create(
1380 rewriter, op.getLoc(), initStruct, apiVersionCst, ArrayRef<int64_t>{0});
1382 "Unexpected offset of field apiVersion");
1385 LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1386 numStateBytesCst, ArrayRef<int64_t>{1});
1388 "Unexpected offset of field numStateBytes");
1390 initStruct = LLVM::InsertValueOp::create(rewriter, op.getLoc(), initStruct,
1391 nameAddr, ArrayRef<int64_t>{2});
1393 "Unexpected offset of field modelName");
1395 initStruct = LLVM::InsertValueOp::create(
1396 rewriter, op.getLoc(), initStruct, traceInfoPtr, ArrayRef<int64_t>{3});
1398 "Unexpected offset of field traceInfo");
1400 LLVM::ReturnOp::create(rewriter, op.getLoc(), initStruct);
1402 rewriter.replaceOp(op, modInfoGlobalOp);
1412 auto bitWidth = computeLLVMBitWidth(type);
1413 assert(bitWidth.has_value());
1414 return llvm::divideCeil(*bitWidth, 8);
1419 auto arrayBitWidth = computeLLVMBitWidth(arrayRefType);
1420 assert(arrayBitWidth.has_value());
1421 assert(arrayRefType.getNumElements() > 0 &&
1422 "Cannot compute stride for zero sized array");
1423 size_t elementBitWidth = *arrayBitWidth / arrayRefType.getNumElements();
1424 return llvm::divideCeil(elementBitWidth, 8);
1428 using OpConversionPattern::OpConversionPattern;
1432 ConversionPatternRewriter &rewriter)
const override {
1433 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1434 auto i8Ty = rewriter.getI8Type();
1435 ArrayRefType arrayRefType = op.getType();
1437 auto size = LLVM::ConstantOp::create(rewriter, op.getLoc(),
1438 rewriter.getI64Type(), byteWidth);
1441 auto alloc = LLVM::AllocaOp::create(rewriter, op.getLoc(), ptrTy, i8Ty,
1444 if (op.getInitAttr()) {
1445 ArrayAttr initAttr = op.getInitAttr();
1447 auto i8Ty = rewriter.getI8Type();
1448 auto zero = LLVM::ConstantOp::create(rewriter, op.getLoc(), i8Ty, 0);
1449 LLVM::MemsetOp::create(rewriter, op.getLoc(), alloc, zero, size,
1452 initializeArray(rewriter, op.getLoc(), alloc, initAttr, arrayRefType);
1456 rewriter.replaceOp(op, alloc);
1466 auto dl = DataLayout::closest(op);
1468 hw::ArrayType::get(type.getElementType(), type.getNumElements());
1469 auto llvmType = getTypeConverter()->convertType(hwType);
1471 static_cast<unsigned>(dl.getTypePreferredAlignment(llvmType));
1472 alignment = std::max(4u, alignment);
1478 return llvm::all_of(arrayAttr.getAsValueRange<IntegerAttr>(),
1479 [](APInt i) { return i.isZero(); });
1483 Value alloc, ArrayAttr initAttr,
1484 ArrayRefType arrayRefType)
const {
1486 Type ptrTy = LLVM::LLVMPointerType::get(getContext());
1487 Type i8Ty = rewriter.getI8Type();
1488 for (
unsigned i = 0; i < arrayRefType.getNumElements(); ++i) {
1489 unsigned elemIndex = arrayRefType.getNumElements() - i - 1;
1490 Value elemOffset = LLVM::ConstantOp::create(
1491 rewriter, loc, rewriter.getI64Type(), elemIndex * elemByteWidth);
1493 LLVM::GEPOp::create(rewriter, loc, ptrTy, i8Ty, alloc, elemOffset);
1494 auto elem = LLVM::ConstantOp::create(
1495 rewriter, loc, arrayRefType.getElementType(), initAttr[i]);
1496 LLVM::StoreOp::create(rewriter, loc, elem, elemAddr);
1505 using OpConversionPattern::OpConversionPattern;
1509 ConversionPatternRewriter &rewriter)
const override {
1510 ArrayRefType arrayRefType = cast<ArrayRefType>(op.getType());
1511 Value alloc = adaptor.getInput();
1512 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1513 auto i8Ty = rewriter.getI8Type();
1515 auto elements = adaptor.getElements();
1516 for (
unsigned i = 0; i < elements.size(); ++i) {
1518 unsigned elemIndex = arrayRefType.getNumElements() - i - 1;
1520 LLVM::ConstantOp::create(rewriter, op.getLoc(), rewriter.getI64Type(),
1521 elemIndex * elemByteWidth);
1522 auto elemAddr = LLVM::GEPOp::create(rewriter, op.getLoc(), ptrTy, i8Ty,
1524 LLVM::StoreOp::create(rewriter, op.getLoc(), elements[i], elemAddr);
1526 rewriter.replaceOp(op, alloc);
1532 using OpConversionPattern::OpConversionPattern;
1536 ConversionPatternRewriter &rewriter)
const override {
1537 auto loc = op.getLoc();
1538 ArrayRefType arrayRefType = cast<ArrayRefType>(op.getInput().getType());
1539 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1540 auto i8Ty = rewriter.getI8Type();
1541 auto i64Ty = rewriter.getI64Type();
1543 assert(!isa<ArrayRefType>(arrayRefType.getElementType()));
1546 LLVM::ConstantOp::create(rewriter, loc, i64Ty, elemByteWidth);
1548 LLVM::MulOp::create(rewriter, loc, adaptor.getIndex(), stride);
1551 size_t lastElementByteOffset =
1552 elemByteWidth * (arrayRefType.getNumElements() - 1);
1553 Value lastElementByteOffsetVal =
1554 LLVM::ConstantOp::create(rewriter, loc, i64Ty, lastElementByteOffset);
1555 Value clampedOffset = LLVM::UMinOp::create(rewriter, loc, i64Ty, byteOffset,
1556 lastElementByteOffsetVal);
1557 auto elemAddr = LLVM::GEPOp::create(rewriter, loc, ptrTy, i8Ty,
1558 adaptor.getInput(), clampedOffset);
1559 Value loaded = LLVM::LoadOp::create(
1560 rewriter, loc, typeConverter->convertType(op.getValue().getType()),
1562 rewriter.replaceOp(op, loaded);
1568 using OpConversionPattern::OpConversionPattern;
1572 ConversionPatternRewriter &rewriter)
const override {
1573 auto loc = op.getLoc();
1574 ArrayRefType arrayRefType = cast<ArrayRefType>(op.getInput().getType());
1575 assert(!isa<ArrayRefType>(arrayRefType.getElementType()));
1576 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1577 auto i8Ty = rewriter.getI8Type();
1578 auto i64Ty = rewriter.getI64Type();
1583 LLVM::ConstantOp::create(rewriter, loc, i64Ty, elemByteWidth);
1585 LLVM::MulOp::create(rewriter, loc, adaptor.getIndex(), stride);
1586 Value totalSize = LLVM::ConstantOp::create(rewriter, loc, i64Ty, byteWidth);
1589 Value isInbounds = LLVM::ICmpOp::create(
1590 rewriter, loc, LLVM::ICmpPredicate::ult, byteOffset, totalSize);
1591 scf::IfOp::create(rewriter, loc, isInbounds, [&](OpBuilder &b, Location) {
1592 auto elemAddr = LLVM::GEPOp::create(b, loc, ptrTy, i8Ty,
1593 adaptor.getInput(), byteOffset);
1594 LLVM::StoreOp::create(b, loc, adaptor.getElement(), elemAddr);
1595 scf::YieldOp::create(b, loc);
1600 rewriter.replaceOp(op, adaptor.getInput());
1606 using OpConversionPattern::OpConversionPattern;
1610 ConversionPatternRewriter &rewriter)
const override {
1611 auto loc = op.getLoc();
1613 ArrayRefType inputType = cast<ArrayRefType>(op.getInput().getType());
1614 ArrayRefType resultType = cast<ArrayRefType>(op.getOutput().getType());
1615 auto ptrTy = LLVM::LLVMPointerType::get(getContext());
1616 auto i8Ty = rewriter.getI8Type();
1617 auto i64Ty = rewriter.getI64Type();
1621 size_t maxLowIndex =
1622 inputType.getNumElements() - resultType.getNumElements();
1623 Value maxLowIndexVal =
1624 LLVM::ConstantOp::create(rewriter, loc, i64Ty, maxLowIndex);
1625 Value clampedLowIndex = LLVM::UMinOp::create(
1626 rewriter, loc, i64Ty, adaptor.getLowIndex(), maxLowIndexVal);
1630 LLVM::ConstantOp::create(rewriter, loc, i64Ty, elemByteWidth);
1632 LLVM::MulOp::create(rewriter, loc, clampedLowIndex, stride);
1633 auto sliceAddr = LLVM::GEPOp::create(rewriter, loc, ptrTy, i8Ty,
1634 adaptor.getInput(), byteOffset);
1635 rewriter.replaceOp(op, sliceAddr);
1641 using OpConversionPattern::OpConversionPattern;
1645 ConversionPatternRewriter &rewriter)
const override {
1646 auto loc = op.getLoc();
1647 ArrayRefType arrayRefType = cast<ArrayRefType>(op.getInput().getType());
1648 auto i64Ty = rewriter.getI64Type();
1650 Value size = LLVM::ConstantOp::create(rewriter, loc, i64Ty, byteWidth);
1652 LLVM::MemmoveOp::create(rewriter, loc, adaptor.getInput(),
1653 adaptor.getSource(), size,
1655 rewriter.replaceOp(op, adaptor.getInput());
1661 ArrayRefType arrayRefType,
1662 LLVM::LLVMArrayType llvmType) {
1663 auto i8Ty = builder.getI8Type();
1664 auto ptrTy = LLVM::LLVMPointerType::get(builder.getContext());
1666 Value v = LLVM::PoisonOp::create(builder, llvmType);
1667 int32_t size = arrayRefType.getNumElements();
1668 for (int32_t i = 0; i < size; i++) {
1669 int32_t byteOffset = i * elemByteWidth;
1670 Value gep = LLVM::GEPOp::create(builder, ptrTy, i8Ty, arrayRef,
1671 LLVM::GEPArg{byteOffset});
1672 Value load = LLVM::LoadOp::create(builder, llvmType.getElementType(), gep);
1673 v = LLVM::InsertValueOp::create(builder, v, load, i);
1679 Value arrayRef, ArrayRefType arrayRefType) {
1680 auto i8Ty = builder.getI8Type();
1681 auto ptrTy = LLVM::LLVMPointerType::get(builder.getContext());
1683 int32_t size = arrayRefType.getNumElements();
1684 for (int32_t i = 0; i < size; i++) {
1685 int32_t byteOffset = i * elemByteWidth;
1686 Value gep = LLVM::GEPOp::create(builder, ptrTy, i8Ty, arrayRef,
1687 LLVM::GEPArg{byteOffset});
1688 Value val = LLVM::ExtractValueOp::create(builder, array, i);
1689 LLVM::StoreOp::create(builder, val, gep);
1695 using OpConversionPattern::OpConversionPattern;
1699 ConversionPatternRewriter &rewriter)
const override {
1700 if (!isa<ArrayRefType>(op.getOperand(0).getType()) ||
1701 !isa<LLVM::LLVMArrayType>(op.getResult(0).getType())) {
1705 ImplicitLocOpBuilder b(op.getLoc(), rewriter);
1707 b, adaptor.getInputs().front(),
1708 cast<ArrayRefType>(op.getOperand(0).getType()),
1709 cast<LLVM::LLVMArrayType>(op.getResult(0).getType()));
1710 rewriter.replaceOp(op, loaded);
1717 using OpConversionPattern::OpConversionPattern;
1721 ConversionPatternRewriter &rewriter)
const override {
1722 Type resultType = getTypeConverter()->convertType(op.getResult().getType());
1723 ImplicitLocOpBuilder b(op.getLoc(), rewriter);
1725 b, adaptor.getInput(), cast<ArrayRefType>(op.getInput().getType()),
1726 cast<LLVM::LLVMArrayType>(resultType));
1727 rewriter.replaceOp(op, loaded);
1734 using OpConversionPattern::OpConversionPattern;
1738 ConversionPatternRewriter &rewriter)
const override {
1739 ImplicitLocOpBuilder b(op.getLoc(), rewriter);
1741 cast<ArrayRefType>(op.getInput().getType()));
1742 rewriter.replaceOp(op, adaptor.getInput());
1752struct LowerArcToLLVMPass
1753 :
public circt::impl::LowerArcToLLVMBase<LowerArcToLLVMPass> {
1754 void runOnOperation()
override;
1758void LowerArcToLLVMPass::runOnOperation() {
1761 for (func::FuncOp func : getOperation().getOps<func::FuncOp>()) {
1762 for (
int i = 0, e = func.getNumArguments(); i != e; ++i) {
1763 if (
auto arrayRefType =
1764 dyn_cast<ArrayRefType>(func.getArgumentTypes()[i])) {
1766 Builder builder(&getContext());
1767 func.setArgAttr(i, LLVM::LLVMDialect::getDereferenceableAttrName(),
1768 builder.getI64IntegerAttr(byteWidth));
1784 LLVMConversionTarget target(getContext());
1785 target.addLegalOp<mlir::ModuleOp>();
1786 target.addLegalOp<scf::YieldOp>();
1791 target.addLegalOp<sim::FormatLiteralOp, sim::FormatDecOp, sim::FormatHexOp,
1792 sim::FormatBinOp, sim::FormatOctOp, sim::FormatCharOp,
1793 sim::FormatStringConcatOp>();
1796 LLVMTypeConverter converter(&getContext());
1797 converter.addConversion([&](seq::ClockType type) {
1798 return IntegerType::get(type.getContext(), 1);
1800 converter.addConversion([&](StorageType type) {
1801 return LLVM::LLVMPointerType::get(type.getContext());
1803 converter.addConversion([&](MemoryType type) {
1804 return LLVM::LLVMPointerType::get(type.getContext());
1806 converter.addConversion([&](StateType type) {
1807 return LLVM::LLVMPointerType::get(type.getContext());
1809 converter.addConversion([&](SimModelInstanceType type) {
1810 return LLVM::LLVMPointerType::get(type.getContext());
1812 converter.addConversion([&](sim::FormatStringType type) {
1813 return LLVM::LLVMPointerType::get(type.getContext());
1815 converter.addConversion([&](llhd::TimeType type) {
1817 return IntegerType::get(type.getContext(), 64);
1819 converter.addConversion([&](ArrayRefType type) {
1820 return LLVM::LLVMPointerType::get(type.getContext());
1825 target.addDynamicallyLegalOp<UnrealizedConversionCastOp>([&](Operation *op) {
1826 Type src = op->getOperand(0).getType();
1827 Type dst = op->getResult(0).getType();
1828 bool needsConvert = isa<ArrayRefType>(src) && isa<LLVM::LLVMArrayType>(dst);
1829 return !needsConvert;
1836 populateSCFToControlFlowConversionPatterns(
patterns);
1837 populateFuncToLLVMConversionPatterns(converter,
patterns);
1838 cf::populateControlFlowToLLVMConversionPatterns(converter,
patterns);
1839 arith::populateArithToLLVMConversionPatterns(converter,
patterns);
1840 index::populateIndexToLLVMConversionPatterns(converter,
patterns);
1841 populateAnyFunctionOpInterfaceTypeConversionPattern(
patterns, converter);
1844 DenseMap<std::pair<Type, ArrayAttr>, LLVM::GlobalOp> constAggregateGlobalsMap;
1846 std::optional<HWToLLVMArraySpillCache> spillCacheOpt =
1849 OpBuilder spillBuilder(getOperation());
1850 spillCacheOpt->spillNonHWOps(spillBuilder, converter, getOperation());
1853 constAggregateGlobalsMap, spillCacheOpt);
1861 AllocMemoryOpLowering,
1862 AllocStateLikeOpLowering<arc::AllocStateOp>,
1863 AllocStateLikeOpLowering<arc::RootInputOp>,
1864 AllocStateLikeOpLowering<arc::RootOutputOp>,
1865 AllocStorageOpLowering,
1866 ClockGateOpLowering,
1868 ConstantTimeOpLowering,
1869 CurrentTimeOpLowering,
1870 GetNextWakeupOpLowering,
1871 IntToTimeOpLowering,
1872 MemoryReadOpLowering,
1873 MemoryWriteOpLowering,
1875 ReplaceOpWithInputPattern<seq::ToClockOp>,
1876 ReplaceOpWithInputPattern<seq::FromClockOp>,
1878 SeqConstClockLowering,
1879 SetNextWakeupOpLowering,
1880 SimGetNextWakeupOpLowering,
1881 SimGetTimeOpLowering,
1882 SimSetTimeOpLowering,
1883 StateReadOpLowering,
1884 StateWriteOpLowering,
1885 StorageGetOpLowering,
1886 TerminateOpLowering,
1887 TimeToIntOpLowering,
1888 ZeroCountOpLowering,
1898 >(converter, &getContext());
1902 StringCache stringCache;
1903 patterns.add<SimEmitValueOpLowering, SimPrintFormattedProcOpLowering>(
1904 converter, &getContext(), stringCache);
1906 auto &modelInfo = getAnalysis<ModelInfoAnalysis>();
1907 llvm::DenseMap<StringRef, ModelInfoMap> modelMap(modelInfo.infoMap.size());
1908 for (
auto &[_, modelInfo] : modelInfo.infoMap) {
1909 llvm::DenseMap<StringRef, StateInfo> states(modelInfo.states.size());
1910 for (StateInfo &stateInfo : modelInfo.states)
1911 states.insert({stateInfo.name, stateInfo});
1914 ModelInfoMap{modelInfo.numStateBytes, std::move(states),
1915 modelInfo.initialFnSym, modelInfo.finalFnSym}});
1918 patterns.add<SimInstantiateOpLowering, SimSetInputOpLowering,
1919 SimGetPortOpLowering, SimStepOpLowering>(
1920 converter, &getContext(), modelMap);
1923 ConversionConfig config;
1924 config.allowPatternRollback =
false;
1925 if (failed(applyFullConversion(getOperation(), target, std::move(
patterns),
1927 signalPassFailure();
1931 return std::make_unique<LowerArcToLLVMPass>();
assert(baseType &&"element must be base type")
static std::unique_ptr< Context > context
static LLVM::GlobalOp buildGlobalConstantIntArray(OpBuilder &builder, Location loc, Twine symName, SmallVectorImpl< T > &data, unsigned alignment=alignof(T))
static LLVM::GlobalOp buildGlobalConstantRuntimeStructArray(OpBuilder &builder, Location loc, Twine symName, SmallVectorImpl< T > &array)
static Value loadArrayRefAsArray(ImplicitLocOpBuilder &builder, Value arrayRef, ArrayRefType arrayRefType, LLVM::LLVMArrayType llvmType)
size_t computeByteWidth(ArrayRefType type)
static llvm::Twine evalSymbolFromModelName(StringRef modelName)
size_t computeElementByteWidth(ArrayRefType arrayRefType)
static void storeArrayAsArrayRef(ImplicitLocOpBuilder &builder, Value array, Value arrayRef, ArrayRefType arrayRefType)
static LogicalResult convert(arc::ExecuteOp op, arc::ExecuteOp::Adaptor adaptor, ConversionPatternRewriter &rewriter, const TypeConverter &converter)
Extension of RewritePatternSet that allows adding matchAndRewrite functions with op adaptors and Conv...
A namespace that is used to store existing names and generate new names in some scope within the IR.
void add(mlir::ModuleOp module)
void addDefinitions(mlir::Operation *top)
Populate the symbol cache with all symbol-defining operations within the 'top' operation.
Default symbol cache implementation; stores associations between names (StringAttr's) to mlir::Operat...
#define ARC_RUNTIME_API_VERSION
Version of the combined public and internal API.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
void populateCombToArithConversionPatterns(TypeConverter &converter, RewritePatternSet &patterns)
void populateCombToLLVMConversionPatterns(mlir::LLVMTypeConverter &converter, RewritePatternSet &patterns)
Get the Comb to LLVM conversion patterns.
void populateHWToLLVMTypeConversions(mlir::LLVMTypeConverter &converter)
Get the HW to LLVM type conversions.
void populateHWToLLVMConversionPatterns(mlir::LLVMTypeConverter &converter, RewritePatternSet &patterns, Namespace &globals, DenseMap< std::pair< Type, ArrayAttr >, mlir::LLVM::GlobalOp > &constAggregateGlobalsMap, std::optional< HWToLLVMArraySpillCache > &spillCacheOpt)
Get the HW to LLVM conversion patterns.
std::unique_ptr< OperationPass< ModuleOp > > createLowerArcToLLVMPass()
Static information for a compiled hardware model, generated by the MLIR lowering.
uint32_t typeBits
Bit width of the traced signal.
uint64_t stateOffset
Byte offset of the traced value within the model state.
uint64_t nameOffset
Byte offset to the null terminator of this signal's last alias in the names array.
uint32_t reserved
Padding and reserved for future use.
void initializeArray(ConversionPatternRewriter &rewriter, Location loc, Value alloc, ArrayAttr initAttr, ArrayRefType arrayRefType) const
size_t computeAllocaAlignment(ArrayRefType type, Operation *op) const
LogicalResult matchAndRewrite(ArrayRefAllocOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
bool isZero(ArrayAttr arrayAttr) const
DenseMap< ArrayRefType, size_t > alignmentCache
LogicalResult matchAndRewrite(ArrayRefCopyOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefCreateOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefFromArrayOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefGetOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefInjectOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefSliceOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(ArrayRefToArrayOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LogicalResult matchAndRewrite(UnrealizedConversionCastOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const override
LLVM::GlobalOp buildTraceInfoStruct(arc::RuntimeModelOp &op, ConversionPatternRewriter &rewriter) const
static constexpr uint64_t runtimeApiVersion
LogicalResult matchAndRewrite(arc::RuntimeModelOp op, OpAdaptor adaptor, ConversionPatternRewriter &rewriter) const final
Helper class mapping array values (HW or LLVM Dialect) to pointers to buffers containing the array va...