1c33d6970SRiver Riddle //===- DebugTranslation.cpp - MLIR to LLVM Debug conversion ---------------===//
2c33d6970SRiver Riddle //
3c33d6970SRiver Riddle // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4c33d6970SRiver Riddle // See https://llvm.org/LICENSE.txt for license information.
5c33d6970SRiver Riddle // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6c33d6970SRiver Riddle //
7c33d6970SRiver Riddle //===----------------------------------------------------------------------===//
8c33d6970SRiver Riddle 
9c33d6970SRiver Riddle #include "DebugTranslation.h"
10c33d6970SRiver Riddle #include "mlir/Dialect/LLVMIR/LLVMDialect.h"
11c33d6970SRiver Riddle #include "llvm/IR/Metadata.h"
12c33d6970SRiver Riddle #include "llvm/IR/Module.h"
13c33d6970SRiver Riddle #include "llvm/Support/FileSystem.h"
14c33d6970SRiver Riddle #include "llvm/Support/Path.h"
15c33d6970SRiver Riddle 
16c33d6970SRiver Riddle using namespace mlir;
17c33d6970SRiver Riddle using namespace mlir::LLVM;
18c33d6970SRiver Riddle using namespace mlir::LLVM::detail;
19c33d6970SRiver Riddle 
20c33d6970SRiver Riddle /// A utility walker that interrupts if the operation has valid debug
21c33d6970SRiver Riddle /// information.
interruptIfValidLocation(Operation * op)22c33d6970SRiver Riddle static WalkResult interruptIfValidLocation(Operation *op) {
23c33d6970SRiver Riddle   return op->getLoc().isa<UnknownLoc>() ? WalkResult::advance()
24c33d6970SRiver Riddle                                         : WalkResult::interrupt();
25c33d6970SRiver Riddle }
26c33d6970SRiver Riddle 
DebugTranslation(Operation * module,llvm::Module & llvmModule)27c33d6970SRiver Riddle DebugTranslation::DebugTranslation(Operation *module, llvm::Module &llvmModule)
28c33d6970SRiver Riddle     : builder(llvmModule), llvmCtx(llvmModule.getContext()),
29c33d6970SRiver Riddle       compileUnit(nullptr) {
30c33d6970SRiver Riddle 
31c33d6970SRiver Riddle   // If the module has no location information, there is nothing to do.
32c33d6970SRiver Riddle   if (!module->walk(interruptIfValidLocation).wasInterrupted())
33c33d6970SRiver Riddle     return;
34c33d6970SRiver Riddle 
359db53a18SRiver Riddle   // TODO: Several parts of this are incorrect. Different source
36c33d6970SRiver Riddle   // languages may interpret different parts of the debug information
37c33d6970SRiver Riddle   // differently. Frontends will also want to pipe in various information, like
38c33d6970SRiver Riddle   // flags. This is fine for now as we only emit line-table information and not
39c33d6970SRiver Riddle   // types or variables. This should disappear as the debug information story
40c33d6970SRiver Riddle   // evolves; i.e. when we have proper attributes for LLVM debug metadata.
41c33d6970SRiver Riddle   compileUnit = builder.createCompileUnit(
42c33d6970SRiver Riddle       llvm::dwarf::DW_LANG_C,
43c33d6970SRiver Riddle       builder.createFile(llvmModule.getModuleIdentifier(), "/"),
44c33d6970SRiver Riddle       /*Producer=*/"mlir", /*isOptimized=*/true, /*Flags=*/"", /*RV=*/0);
45c33d6970SRiver Riddle 
46c33d6970SRiver Riddle   // Mark this module as having debug information.
47c33d6970SRiver Riddle   StringRef debugVersionKey = "Debug Info Version";
48c33d6970SRiver Riddle   if (!llvmModule.getModuleFlag(debugVersionKey))
49c33d6970SRiver Riddle     llvmModule.addModuleFlag(llvm::Module::Warning, debugVersionKey,
50c33d6970SRiver Riddle                              llvm::DEBUG_METADATA_VERSION);
51c9e9cc3fSScott Todd 
52c9e9cc3fSScott Todd   if (auto targetTripleAttr =
53c9e9cc3fSScott Todd           module->getAttr(LLVM::LLVMDialect::getTargetTripleAttrName())) {
54c9e9cc3fSScott Todd     auto targetTriple =
55c9e9cc3fSScott Todd         llvm::Triple(targetTripleAttr.cast<StringAttr>().getValue());
56c9e9cc3fSScott Todd     if (targetTriple.isKnownWindowsMSVCEnvironment()) {
57c9e9cc3fSScott Todd       // Dwarf debugging files will be generated by default, unless "CodeView"
58c9e9cc3fSScott Todd       // is set explicitly. Windows/MSVC should use CodeView instead.
59c9e9cc3fSScott Todd       llvmModule.addModuleFlag(llvm::Module::Warning, "CodeView", 1);
60c9e9cc3fSScott Todd     }
61c9e9cc3fSScott Todd   }
62c33d6970SRiver Riddle }
63c33d6970SRiver Riddle 
64c33d6970SRiver Riddle /// Finalize the translation of debug information.
finalize()65c33d6970SRiver Riddle void DebugTranslation::finalize() { builder.finalize(); }
66c33d6970SRiver Riddle 
67c33d6970SRiver Riddle /// Attempt to extract a filename for the given loc.
extractFileLoc(Location loc)68c33d6970SRiver Riddle static FileLineColLoc extractFileLoc(Location loc) {
69c33d6970SRiver Riddle   if (auto fileLoc = loc.dyn_cast<FileLineColLoc>())
70c33d6970SRiver Riddle     return fileLoc;
71c33d6970SRiver Riddle   if (auto nameLoc = loc.dyn_cast<NameLoc>())
72c33d6970SRiver Riddle     return extractFileLoc(nameLoc.getChildLoc());
73c33d6970SRiver Riddle   if (auto opaqueLoc = loc.dyn_cast<OpaqueLoc>())
74c33d6970SRiver Riddle     return extractFileLoc(opaqueLoc.getFallbackLocation());
75c33d6970SRiver Riddle   return FileLineColLoc();
76c33d6970SRiver Riddle }
77c33d6970SRiver Riddle 
78c33d6970SRiver Riddle /// Translate the debug information for the given function.
translate(LLVMFuncOp func,llvm::Function & llvmFunc)79c33d6970SRiver Riddle void DebugTranslation::translate(LLVMFuncOp func, llvm::Function &llvmFunc) {
80c33d6970SRiver Riddle   // If the function doesn't have location information, there is nothing to
81c33d6970SRiver Riddle   // translate.
82c33d6970SRiver Riddle   if (!compileUnit || !func.walk(interruptIfValidLocation).wasInterrupted())
83c33d6970SRiver Riddle     return;
84c33d6970SRiver Riddle 
85f0506e49STim Shen   // If we are to create debug info for the function, we need to ensure that all
86f0506e49STim Shen   // inlinable calls in it are with debug info, otherwise the LLVM verifier will
87f0506e49STim Shen   // complain. For now, be more restricted and treat all calls as inlinable.
88f0506e49STim Shen   const bool hasCallWithoutDebugInfo =
89*7ecb7efcSIan Bearman       func.walk([&](LLVM::CallOp call) {
90*7ecb7efcSIan Bearman             return call.getLoc()->walk([](Location l) {
91*7ecb7efcSIan Bearman               return l.isa<UnknownLoc>() ? WalkResult::interrupt()
92f0506e49STim Shen                                          : WalkResult::advance();
93*7ecb7efcSIan Bearman             });
94f0506e49STim Shen           })
95f0506e49STim Shen           .wasInterrupted();
96f0506e49STim Shen   if (hasCallWithoutDebugInfo)
97f0506e49STim Shen     return;
98f0506e49STim Shen 
99c33d6970SRiver Riddle   FileLineColLoc fileLoc = extractFileLoc(func.getLoc());
100a4bb667dSRiver Riddle   auto *file =
101a4bb667dSRiver Riddle       translateFile(fileLoc ? fileLoc.getFilename().strref() : "<unknown>");
102c33d6970SRiver Riddle   unsigned line = fileLoc ? fileLoc.getLine() : 0;
103c33d6970SRiver Riddle 
1049db53a18SRiver Riddle   // TODO: This is the bare essentials for now. We will likely end
105c33d6970SRiver Riddle   // up with wrapper metadata around LLVMs metadata in the future, so this
106c33d6970SRiver Riddle   // doesn't need to be smart until then.
107c33d6970SRiver Riddle   llvm::DISubroutineType *type =
108c33d6970SRiver Riddle       builder.createSubroutineType(builder.getOrCreateTypeArray(llvm::None));
109c33d6970SRiver Riddle   llvm::DISubprogram::DISPFlags spFlags = llvm::DISubprogram::SPFlagDefinition |
110c33d6970SRiver Riddle                                           llvm::DISubprogram::SPFlagOptimized;
111c33d6970SRiver Riddle   llvm::DISubprogram *program =
112c33d6970SRiver Riddle       builder.createFunction(compileUnit, func.getName(), func.getName(), file,
113c33d6970SRiver Riddle                              line, type, line, llvm::DINode::FlagZero, spFlags);
114c33d6970SRiver Riddle   llvmFunc.setSubprogram(program);
115c33d6970SRiver Riddle   builder.finalizeSubprogram(program);
116c33d6970SRiver Riddle }
117c33d6970SRiver Riddle 
118c33d6970SRiver Riddle //===----------------------------------------------------------------------===//
119c33d6970SRiver Riddle // Locations
120c33d6970SRiver Riddle //===----------------------------------------------------------------------===//
121c33d6970SRiver Riddle 
122c33d6970SRiver Riddle /// Translate the given location to an llvm debug location.
123c33d6970SRiver Riddle const llvm::DILocation *
translateLoc(Location loc,llvm::DILocalScope * scope)124c33d6970SRiver Riddle DebugTranslation::translateLoc(Location loc, llvm::DILocalScope *scope) {
125c33d6970SRiver Riddle   if (!compileUnit)
126c33d6970SRiver Riddle     return nullptr;
127c33d6970SRiver Riddle   return translateLoc(loc, scope, /*inlinedAt=*/nullptr);
128c33d6970SRiver Riddle }
129c33d6970SRiver Riddle 
130c33d6970SRiver Riddle /// Translate the given location to an llvm DebugLoc.
131c33d6970SRiver Riddle const llvm::DILocation *
translateLoc(Location loc,llvm::DILocalScope * scope,const llvm::DILocation * inlinedAt)132c33d6970SRiver Riddle DebugTranslation::translateLoc(Location loc, llvm::DILocalScope *scope,
133c33d6970SRiver Riddle                                const llvm::DILocation *inlinedAt) {
134c33d6970SRiver Riddle   // LLVM doesn't have a representation for unknown.
135c33d6970SRiver Riddle   if (!scope || loc.isa<UnknownLoc>())
136c33d6970SRiver Riddle     return nullptr;
137c33d6970SRiver Riddle 
138c33d6970SRiver Riddle   // Check for a cached instance.
1396fab33b2SRiver Riddle   auto existingIt = locationToLoc.find(std::make_pair(loc, scope));
1406fab33b2SRiver Riddle   if (existingIt != locationToLoc.end())
1416fab33b2SRiver Riddle     return existingIt->second;
142c33d6970SRiver Riddle 
1436fab33b2SRiver Riddle   const llvm::DILocation *llvmLoc = nullptr;
144fff39b62SRiver Riddle   if (auto callLoc = loc.dyn_cast<CallSiteLoc>()) {
145c33d6970SRiver Riddle     // For callsites, the caller is fed as the inlinedAt for the callee.
146c33d6970SRiver Riddle     const auto *callerLoc = translateLoc(callLoc.getCaller(), scope, inlinedAt);
147c33d6970SRiver Riddle     llvmLoc = translateLoc(callLoc.getCallee(), scope, callerLoc);
148fff39b62SRiver Riddle 
149fff39b62SRiver Riddle   } else if (auto fileLoc = loc.dyn_cast<FileLineColLoc>()) {
150c33d6970SRiver Riddle     auto *file = translateFile(fileLoc.getFilename());
151c33d6970SRiver Riddle     auto *fileScope = builder.createLexicalBlockFile(scope, file);
152c33d6970SRiver Riddle     llvmLoc = llvm::DILocation::get(llvmCtx, fileLoc.getLine(),
153c33d6970SRiver Riddle                                     fileLoc.getColumn(), fileScope,
154c33d6970SRiver Riddle                                     const_cast<llvm::DILocation *>(inlinedAt));
155fff39b62SRiver Riddle 
156fff39b62SRiver Riddle   } else if (auto fusedLoc = loc.dyn_cast<FusedLoc>()) {
157c33d6970SRiver Riddle     ArrayRef<Location> locations = fusedLoc.getLocations();
158c33d6970SRiver Riddle 
159c33d6970SRiver Riddle     // For fused locations, merge each of the nodes.
160c33d6970SRiver Riddle     llvmLoc = translateLoc(locations.front(), scope, inlinedAt);
161c33d6970SRiver Riddle     for (Location locIt : locations.drop_front()) {
162c33d6970SRiver Riddle       llvmLoc = llvm::DILocation::getMergedLocation(
163c33d6970SRiver Riddle           llvmLoc, translateLoc(locIt, scope, inlinedAt));
164c33d6970SRiver Riddle     }
165fff39b62SRiver Riddle 
166fff39b62SRiver Riddle   } else if (auto nameLoc = loc.dyn_cast<NameLoc>()) {
167c33d6970SRiver Riddle     llvmLoc = translateLoc(loc.cast<NameLoc>().getChildLoc(), scope, inlinedAt);
168fff39b62SRiver Riddle 
169fff39b62SRiver Riddle   } else if (auto opaqueLoc = loc.dyn_cast<OpaqueLoc>()) {
170c33d6970SRiver Riddle     llvmLoc = translateLoc(loc.cast<OpaqueLoc>().getFallbackLocation(), scope,
171c33d6970SRiver Riddle                            inlinedAt);
172fff39b62SRiver Riddle   } else {
173c33d6970SRiver Riddle     llvm_unreachable("unknown location kind");
174c33d6970SRiver Riddle   }
175fff39b62SRiver Riddle 
1766fab33b2SRiver Riddle   locationToLoc.try_emplace(std::make_pair(loc, scope), llvmLoc);
177c33d6970SRiver Riddle   return llvmLoc;
178c33d6970SRiver Riddle }
179c33d6970SRiver Riddle 
180c33d6970SRiver Riddle /// Create an llvm debug file for the given file path.
translateFile(StringRef fileName)181c33d6970SRiver Riddle llvm::DIFile *DebugTranslation::translateFile(StringRef fileName) {
182c33d6970SRiver Riddle   auto *&file = fileMap[fileName];
183c33d6970SRiver Riddle   if (file)
184c33d6970SRiver Riddle     return file;
185c33d6970SRiver Riddle 
186c33d6970SRiver Riddle   // Make sure the current working directory is up-to-date.
187c33d6970SRiver Riddle   if (currentWorkingDir.empty())
188c33d6970SRiver Riddle     llvm::sys::fs::current_path(currentWorkingDir);
189c33d6970SRiver Riddle 
190c33d6970SRiver Riddle   StringRef directory = currentWorkingDir;
191c33d6970SRiver Riddle   SmallString<128> dirBuf;
192c33d6970SRiver Riddle   SmallString<128> fileBuf;
193c33d6970SRiver Riddle   if (llvm::sys::path::is_absolute(fileName)) {
194c33d6970SRiver Riddle     // Strip the common prefix (if it is more than just "/") from current
195c33d6970SRiver Riddle     // directory and FileName for a more space-efficient encoding.
196c33d6970SRiver Riddle     auto fileIt = llvm::sys::path::begin(fileName);
197c33d6970SRiver Riddle     auto fileE = llvm::sys::path::end(fileName);
198c33d6970SRiver Riddle     auto curDirIt = llvm::sys::path::begin(directory);
199c33d6970SRiver Riddle     auto curDirE = llvm::sys::path::end(directory);
200c33d6970SRiver Riddle     for (; curDirIt != curDirE && *curDirIt == *fileIt; ++curDirIt, ++fileIt)
201c33d6970SRiver Riddle       llvm::sys::path::append(dirBuf, *curDirIt);
202c33d6970SRiver Riddle     if (std::distance(llvm::sys::path::begin(directory), curDirIt) == 1) {
203c33d6970SRiver Riddle       // Don't strip the common prefix if it is only the root "/"  since that
204c33d6970SRiver Riddle       // would make LLVM diagnostic locations confusing.
205c33d6970SRiver Riddle       directory = StringRef();
206c33d6970SRiver Riddle     } else {
207c33d6970SRiver Riddle       for (; fileIt != fileE; ++fileIt)
208c33d6970SRiver Riddle         llvm::sys::path::append(fileBuf, *fileIt);
209c33d6970SRiver Riddle       directory = dirBuf;
210c33d6970SRiver Riddle       fileName = fileBuf;
211c33d6970SRiver Riddle     }
212c33d6970SRiver Riddle   }
213c33d6970SRiver Riddle   return (file = builder.createFile(fileName, directory));
214c33d6970SRiver Riddle }
215