1 //===-- SBReproducer.cpp --------------------------------------------------===// 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 "SBReproducerPrivate.h" 10 11 #include "lldb/API/LLDB.h" 12 #include "lldb/API/SBAddress.h" 13 #include "lldb/API/SBAttachInfo.h" 14 #include "lldb/API/SBBlock.h" 15 #include "lldb/API/SBBreakpoint.h" 16 #include "lldb/API/SBCommandInterpreter.h" 17 #include "lldb/API/SBCommandInterpreterRunOptions.h" 18 #include "lldb/API/SBData.h" 19 #include "lldb/API/SBDebugger.h" 20 #include "lldb/API/SBDeclaration.h" 21 #include "lldb/API/SBError.h" 22 #include "lldb/API/SBFileSpec.h" 23 #include "lldb/API/SBHostOS.h" 24 #include "lldb/API/SBReproducer.h" 25 #include "lldb/Host/FileSystem.h" 26 #include "lldb/lldb-private.h" 27 28 using namespace lldb; 29 using namespace lldb_private; 30 using namespace lldb_private::repro; 31 32 SBReplayOptions::SBReplayOptions() 33 : m_opaque_up(std::make_unique<ReplayOptions>()){} 34 35 SBReplayOptions::SBReplayOptions(const SBReplayOptions &rhs) 36 : m_opaque_up(std::make_unique<ReplayOptions>(*rhs.m_opaque_up)) {} 37 38 SBReplayOptions::~SBReplayOptions() = default; 39 40 SBReplayOptions &SBReplayOptions::operator=(const SBReplayOptions &rhs) { 41 if (this == &rhs) 42 return *this; 43 *m_opaque_up = *rhs.m_opaque_up; 44 return *this; 45 } 46 47 void SBReplayOptions::SetVerify(bool verify) { m_opaque_up->verify = verify; } 48 49 bool SBReplayOptions::GetVerify() const { return m_opaque_up->verify; } 50 51 void SBReplayOptions::SetCheckVersion(bool check) { 52 m_opaque_up->check_version = check; 53 } 54 55 bool SBReplayOptions::GetCheckVersion() const { 56 return m_opaque_up->check_version; 57 } 58 59 SBRegistry::SBRegistry() { 60 Registry &R = *this; 61 62 RegisterMethods<SBAddress>(R); 63 RegisterMethods<SBAttachInfo>(R); 64 RegisterMethods<SBBlock>(R); 65 RegisterMethods<SBBreakpoint>(R); 66 RegisterMethods<SBBreakpointList>(R); 67 RegisterMethods<SBBreakpointLocation>(R); 68 RegisterMethods<SBBreakpointName>(R); 69 RegisterMethods<SBBroadcaster>(R); 70 RegisterMethods<SBCommandInterpreter>(R); 71 RegisterMethods<SBCommandInterpreterRunOptions>(R); 72 RegisterMethods<SBCommandReturnObject>(R); 73 RegisterMethods<SBCommunication>(R); 74 RegisterMethods<SBCompileUnit>(R); 75 RegisterMethods<SBData>(R); 76 RegisterMethods<SBDebugger>(R); 77 RegisterMethods<SBDeclaration>(R); 78 RegisterMethods<SBEnvironment>(R); 79 RegisterMethods<SBError>(R); 80 RegisterMethods<SBEvent>(R); 81 RegisterMethods<SBExecutionContext>(R); 82 RegisterMethods<SBExpressionOptions>(R); 83 RegisterMethods<SBFile>(R); 84 RegisterMethods<SBFileSpec>(R); 85 RegisterMethods<SBFileSpecList>(R); 86 RegisterMethods<SBFrame>(R); 87 RegisterMethods<SBFunction>(R); 88 RegisterMethods<SBHostOS>(R); 89 RegisterMethods<SBInputReader>(R); 90 RegisterMethods<SBInstruction>(R); 91 RegisterMethods<SBInstructionList>(R); 92 RegisterMethods<SBLanguageRuntime>(R); 93 RegisterMethods<SBLaunchInfo>(R); 94 RegisterMethods<SBLineEntry>(R); 95 RegisterMethods<SBListener>(R); 96 RegisterMethods<SBMemoryRegionInfo>(R); 97 RegisterMethods<SBMemoryRegionInfoList>(R); 98 RegisterMethods<SBModule>(R); 99 RegisterMethods<SBModuleSpec>(R); 100 RegisterMethods<SBPlatform>(R); 101 RegisterMethods<SBPlatformConnectOptions>(R); 102 RegisterMethods<SBPlatformShellCommand>(R); 103 RegisterMethods<SBProcess>(R); 104 RegisterMethods<SBProcessInfo>(R); 105 RegisterMethods<SBQueue>(R); 106 RegisterMethods<SBQueueItem>(R); 107 RegisterMethods<SBSection>(R); 108 RegisterMethods<SBSourceManager>(R); 109 RegisterMethods<SBStream>(R); 110 RegisterMethods<SBStringList>(R); 111 RegisterMethods<SBStructuredData>(R); 112 RegisterMethods<SBSymbol>(R); 113 RegisterMethods<SBSymbolContext>(R); 114 RegisterMethods<SBSymbolContextList>(R); 115 RegisterMethods<SBTarget>(R); 116 RegisterMethods<SBThread>(R); 117 RegisterMethods<SBThreadCollection>(R); 118 RegisterMethods<SBThreadPlan>(R); 119 RegisterMethods<SBTrace>(R); 120 RegisterMethods<SBType>(R); 121 RegisterMethods<SBTypeCategory>(R); 122 RegisterMethods<SBTypeEnumMember>(R); 123 RegisterMethods<SBTypeFilter>(R); 124 RegisterMethods<SBTypeFormat>(R); 125 RegisterMethods<SBTypeNameSpecifier>(R); 126 RegisterMethods<SBTypeSummary>(R); 127 RegisterMethods<SBTypeSummaryOptions>(R); 128 RegisterMethods<SBTypeSynthetic>(R); 129 RegisterMethods<SBUnixSignals>(R); 130 RegisterMethods<SBValue>(R); 131 RegisterMethods<SBValueList>(R); 132 RegisterMethods<SBVariablesOptions>(R); 133 RegisterMethods<SBWatchpoint>(R); 134 } 135 136 const char *SBReproducer::Capture() { 137 static std::string error; 138 if (auto e = Reproducer::Initialize(ReproducerMode::Capture, llvm::None)) { 139 error = llvm::toString(std::move(e)); 140 return error.c_str(); 141 } 142 143 if (auto *g = lldb_private::repro::Reproducer::Instance().GetGenerator()) { 144 auto &p = g->GetOrCreate<SBProvider>(); 145 InstrumentationData::Initialize(p.GetSerializer(), p.GetRegistry()); 146 } 147 148 return nullptr; 149 } 150 151 const char *SBReproducer::Capture(const char *path) { 152 static std::string error; 153 if (auto e = 154 Reproducer::Initialize(ReproducerMode::Capture, FileSpec(path))) { 155 error = llvm::toString(std::move(e)); 156 return error.c_str(); 157 } 158 159 if (auto *g = lldb_private::repro::Reproducer::Instance().GetGenerator()) { 160 auto &p = g->GetOrCreate<SBProvider>(); 161 InstrumentationData::Initialize(p.GetSerializer(), p.GetRegistry()); 162 } 163 164 return nullptr; 165 } 166 167 const char *SBReproducer::PassiveReplay(const char *path) { 168 static std::string error; 169 if (auto e = Reproducer::Initialize(ReproducerMode::PassiveReplay, 170 FileSpec(path))) { 171 error = llvm::toString(std::move(e)); 172 return error.c_str(); 173 } 174 175 if (auto *l = lldb_private::repro::Reproducer::Instance().GetLoader()) { 176 FileSpec file = l->GetFile<SBProvider::Info>(); 177 auto error_or_file = llvm::MemoryBuffer::getFile(file.GetPath()); 178 if (!error_or_file) { 179 error = 180 "unable to read SB API data: " + error_or_file.getError().message(); 181 return error.c_str(); 182 } 183 static ReplayData r(std::move(*error_or_file)); 184 InstrumentationData::Initialize(r.GetDeserializer(), r.GetRegistry()); 185 } 186 187 return nullptr; 188 } 189 190 const char *SBReproducer::Replay(const char *path) { 191 SBReplayOptions options; 192 return SBReproducer::Replay(path, options); 193 } 194 195 const char *SBReproducer::Replay(const char *path, bool skip_version_check) { 196 SBReplayOptions options; 197 options.SetCheckVersion(!skip_version_check); 198 return SBReproducer::Replay(path, options); 199 } 200 201 const char *SBReproducer::Replay(const char *path, 202 const SBReplayOptions &options) { 203 static std::string error; 204 if (auto e = Reproducer::Initialize(ReproducerMode::Replay, FileSpec(path))) { 205 error = llvm::toString(std::move(e)); 206 return error.c_str(); 207 } 208 209 repro::Loader *loader = repro::Reproducer::Instance().GetLoader(); 210 if (!loader) { 211 error = "unable to get replay loader."; 212 return error.c_str(); 213 } 214 215 if (options.GetCheckVersion()) { 216 llvm::Expected<std::string> version = loader->LoadBuffer<VersionProvider>(); 217 if (!version) { 218 error = llvm::toString(version.takeError()); 219 return error.c_str(); 220 } 221 if (lldb_private::GetVersion() != llvm::StringRef(*version).rtrim()) { 222 error = "reproducer capture and replay version don't match:\n"; 223 error.append("reproducer captured with:\n"); 224 error.append(*version); 225 error.append("reproducer replayed with:\n"); 226 error.append(lldb_private::GetVersion()); 227 return error.c_str(); 228 } 229 } 230 231 if (options.GetVerify()) { 232 bool verification_failed = false; 233 llvm::raw_string_ostream os(error); 234 auto error_callback = [&](llvm::StringRef error) { 235 verification_failed = true; 236 os << "\nerror: " << error; 237 }; 238 239 auto warning_callback = [&](llvm::StringRef warning) { 240 verification_failed = true; 241 os << "\nwarning: " << warning; 242 }; 243 244 auto note_callback = [&](llvm::StringRef warning) {}; 245 246 Verifier verifier(loader); 247 verifier.Verify(error_callback, warning_callback, note_callback); 248 249 if (verification_failed) { 250 os.flush(); 251 return error.c_str(); 252 } 253 } 254 255 FileSpec file = loader->GetFile<SBProvider::Info>(); 256 if (!file) { 257 error = "unable to get replay data from reproducer."; 258 return error.c_str(); 259 } 260 261 SBRegistry registry; 262 registry.Replay(file); 263 264 return nullptr; 265 } 266 267 const char *SBReproducer::Finalize(const char *path) { 268 static std::string error; 269 if (auto e = Reproducer::Initialize(ReproducerMode::Replay, FileSpec(path))) { 270 error = llvm::toString(std::move(e)); 271 return error.c_str(); 272 } 273 274 repro::Loader *loader = repro::Reproducer::Instance().GetLoader(); 275 if (!loader) { 276 error = "unable to get replay loader."; 277 return error.c_str(); 278 } 279 280 if (auto e = repro::Finalize(loader)) { 281 error = llvm::toString(std::move(e)); 282 return error.c_str(); 283 } 284 285 return nullptr; 286 } 287 288 bool SBReproducer::Generate() { 289 auto &r = Reproducer::Instance(); 290 if (auto generator = r.GetGenerator()) { 291 generator->Keep(); 292 return true; 293 } 294 return false; 295 } 296 297 bool SBReproducer::SetAutoGenerate(bool b) { 298 auto &r = Reproducer::Instance(); 299 if (auto generator = r.GetGenerator()) { 300 generator->SetAutoGenerate(b); 301 return true; 302 } 303 return false; 304 } 305 306 const char *SBReproducer::GetPath() { 307 ConstString path; 308 auto &r = Reproducer::Instance(); 309 if (FileSpec reproducer_path = Reproducer::Instance().GetReproducerPath()) 310 path = ConstString(r.GetReproducerPath().GetCString()); 311 return path.GetCString(); 312 } 313 314 void SBReproducer::SetWorkingDirectory(const char *path) { 315 if (auto *g = lldb_private::repro::Reproducer::Instance().GetGenerator()) { 316 auto &wp = g->GetOrCreate<repro::WorkingDirectoryProvider>(); 317 wp.SetDirectory(path); 318 auto &fp = g->GetOrCreate<repro::FileProvider>(); 319 fp.RecordInterestingDirectory(wp.GetDirectory()); 320 } 321 } 322 323 char lldb_private::repro::SBProvider::ID = 0; 324 const char *SBProvider::Info::name = "sbapi"; 325 const char *SBProvider::Info::file = "sbapi.bin"; 326