blob: a75ee0ef51260782445eb68139aa8933e1a9467d [file] [log] [blame]
Zachary Turnercffff262015-02-10 21:17:52 +00001//===- DIASession.cpp - DIA implementation of IPDBSession -------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
Zachary Turner819e77d2016-05-06 20:51:57 +00009#include "llvm/DebugInfo/PDB/DIA/DIASession.h"
Zachary Turnerbe6d1e42015-02-10 23:46:48 +000010#include "llvm/ADT/STLExtras.h"
Zachary Turnercffff262015-02-10 21:17:52 +000011#include "llvm/DebugInfo/PDB/DIA/DIAEnumDebugStreams.h"
Zachary Turner679aead2018-03-13 17:46:06 +000012#include "llvm/DebugInfo/PDB/DIA/DIAEnumInjectedSources.h"
Zachary Turner4b083542015-04-17 22:40:36 +000013#include "llvm/DebugInfo/PDB/DIA/DIAEnumLineNumbers.h"
Aaron Smith523de052018-03-22 04:08:15 +000014#include "llvm/DebugInfo/PDB/DIA/DIAEnumSectionContribs.h"
Zachary Turnera5549172015-02-10 22:43:25 +000015#include "llvm/DebugInfo/PDB/DIA/DIAEnumSourceFiles.h"
Aaron Smith89bca9e2017-11-16 14:33:09 +000016#include "llvm/DebugInfo/PDB/DIA/DIAEnumTables.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000017#include "llvm/DebugInfo/PDB/DIA/DIAError.h"
Zachary Turnercffff262015-02-10 21:17:52 +000018#include "llvm/DebugInfo/PDB/DIA/DIARawSymbol.h"
Zachary Turnercffff262015-02-10 21:17:52 +000019#include "llvm/DebugInfo/PDB/DIA/DIASourceFile.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000020#include "llvm/DebugInfo/PDB/DIA/DIASupport.h"
21#include "llvm/DebugInfo/PDB/GenericError.h"
22#include "llvm/DebugInfo/PDB/PDB.h"
Chandler Carruth71f308a2015-02-13 09:09:03 +000023#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
24#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
Zachary Turnercffff262015-02-10 21:17:52 +000025#include "llvm/Support/ConvertUTF.h"
Zachary Turner38380322016-10-19 16:42:20 +000026#include "llvm/Support/Format.h"
Zachary Turner16901642017-04-24 17:47:24 +000027#include "llvm/Support/FormatVariadic.h"
Zachary Turner38380322016-10-19 16:42:20 +000028#include "llvm/Support/raw_ostream.h"
Zachary Turnercffff262015-02-10 21:17:52 +000029
30using namespace llvm;
Zachary Turnerec28fc32016-05-04 20:32:13 +000031using namespace llvm::pdb;
Zachary Turnercffff262015-02-10 21:17:52 +000032
Zachary Turner16901642017-04-24 17:47:24 +000033template <typename... Ts>
34static Error ErrorFromHResult(HRESULT Result, const char *Str, Ts &&... Args) {
35 SmallString<64> MessageStorage;
36 StringRef Context;
37 if (sizeof...(Args) > 0) {
38 MessageStorage = formatv(Str, std::forward<Ts>(Args)...).str();
39 Context = MessageStorage;
40 } else
41 Context = Str;
42
Zachary Turner819e77d2016-05-06 20:51:57 +000043 switch (Result) {
44 case E_PDB_NOT_FOUND:
Zachary Turner38380322016-10-19 16:42:20 +000045 return make_error<GenericError>(generic_error_code::invalid_path, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000046 case E_PDB_FORMAT:
Zachary Turner38380322016-10-19 16:42:20 +000047 return make_error<DIAError>(dia_error_code::invalid_file_format, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000048 case E_INVALIDARG:
Zachary Turner38380322016-10-19 16:42:20 +000049 return make_error<DIAError>(dia_error_code::invalid_parameter, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000050 case E_UNEXPECTED:
Zachary Turner38380322016-10-19 16:42:20 +000051 return make_error<DIAError>(dia_error_code::already_loaded, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000052 case E_PDB_INVALID_SIG:
53 case E_PDB_INVALID_AGE:
Zachary Turner38380322016-10-19 16:42:20 +000054 return make_error<DIAError>(dia_error_code::debug_info_mismatch, Context);
55 default: {
56 std::string S;
57 raw_string_ostream OS(S);
58 OS << "HRESULT: " << format_hex(static_cast<DWORD>(Result), 10, true)
59 << ": " << Context;
60 return make_error<DIAError>(dia_error_code::unspecified, OS.str());
61 }
Zachary Turner819e77d2016-05-06 20:51:57 +000062 }
63}
64
Reid Klecknerfb58be82016-10-12 21:51:14 +000065static Error LoadDIA(CComPtr<IDiaDataSource> &DiaDataSource) {
Nico Weber73853ab2016-04-01 22:21:51 +000066 if (SUCCEEDED(CoCreateInstance(CLSID_DiaSource, nullptr, CLSCTX_INPROC_SERVER,
67 IID_IDiaDataSource,
68 reinterpret_cast<LPVOID *>(&DiaDataSource))))
Zachary Turner819e77d2016-05-06 20:51:57 +000069 return Error::success();
Nico Weber73853ab2016-04-01 22:21:51 +000070
Zachary Turner819e77d2016-05-06 20:51:57 +000071// If the CoCreateInstance call above failed, msdia*.dll is not registered.
72// Try loading the DLL corresponding to the #included DIA SDK.
Nico Weber73853ab2016-04-01 22:21:51 +000073#if !defined(_MSC_VER)
Zachary Turner819e77d2016-05-06 20:51:57 +000074 return llvm::make_error<GenericError>(
75 "DIA is only supported when using MSVC.");
Reid Klecknerfb58be82016-10-12 21:51:14 +000076#else
Nico Weber73853ab2016-04-01 22:21:51 +000077 const wchar_t *msdia_dll = nullptr;
Reid Klecknerfb58be82016-10-12 21:51:14 +000078#if _MSC_VER >= 1900 && _MSC_VER < 2000
Nico Weber73853ab2016-04-01 22:21:51 +000079 msdia_dll = L"msdia140.dll"; // VS2015
Reid Klecknerfb58be82016-10-12 21:51:14 +000080#elif _MSC_VER >= 1800
Nico Weber73853ab2016-04-01 22:21:51 +000081 msdia_dll = L"msdia120.dll"; // VS2013
82#else
83#error "Unknown Visual Studio version."
84#endif
Zachary Turner23ee87b2016-04-19 17:36:58 +000085
Zachary Turner819e77d2016-05-06 20:51:57 +000086 HRESULT HR;
87 if (FAILED(HR = NoRegCoCreate(msdia_dll, CLSID_DiaSource, IID_IDiaDataSource,
88 reinterpret_cast<LPVOID *>(&DiaDataSource))))
Zachary Turner38380322016-10-19 16:42:20 +000089 return ErrorFromHResult(HR, "Calling NoRegCoCreate");
Zachary Turner819e77d2016-05-06 20:51:57 +000090 return Error::success();
Reid Klecknerfb58be82016-10-12 21:51:14 +000091#endif
Nico Weber73853ab2016-04-01 22:21:51 +000092}
Zachary Turnercffff262015-02-10 21:17:52 +000093
94DIASession::DIASession(CComPtr<IDiaSession> DiaSession) : Session(DiaSession) {}
95
Zachary Turner819e77d2016-05-06 20:51:57 +000096Error DIASession::createFromPdb(StringRef Path,
97 std::unique_ptr<IPDBSession> &Session) {
Zachary Turnerccf04152015-02-28 20:23:18 +000098 CComPtr<IDiaDataSource> DiaDataSource;
99 CComPtr<IDiaSession> DiaSession;
Zachary Turnercffff262015-02-10 21:17:52 +0000100
101 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +0000102 if (auto E = LoadDIA(DiaDataSource))
103 return E;
Zachary Turnercffff262015-02-10 21:17:52 +0000104
105 llvm::SmallVector<UTF16, 128> Path16;
106 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +0000107 return make_error<GenericError>(generic_error_code::invalid_path);
Zachary Turnercffff262015-02-10 21:17:52 +0000108
109 const wchar_t *Path16Str = reinterpret_cast<const wchar_t*>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +0000110 HRESULT HR;
Zachary Turner16901642017-04-24 17:47:24 +0000111 if (FAILED(HR = DiaDataSource->loadDataFromPdb(Path16Str))) {
112 return ErrorFromHResult(HR, "Calling loadDataFromPdb {0}", Path);
113 }
Zachary Turnercffff262015-02-10 21:17:52 +0000114
Zachary Turner819e77d2016-05-06 20:51:57 +0000115 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
Zachary Turner38380322016-10-19 16:42:20 +0000116 return ErrorFromHResult(HR, "Calling openSession");
Zachary Turnerccf04152015-02-28 20:23:18 +0000117
118 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +0000119 return Error::success();
Zachary Turnercffff262015-02-10 21:17:52 +0000120}
121
Zachary Turner819e77d2016-05-06 20:51:57 +0000122Error DIASession::createFromExe(StringRef Path,
123 std::unique_ptr<IPDBSession> &Session) {
Zachary Turner4b083542015-04-17 22:40:36 +0000124 CComPtr<IDiaDataSource> DiaDataSource;
125 CComPtr<IDiaSession> DiaSession;
126
127 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +0000128 if (auto EC = LoadDIA(DiaDataSource))
129 return EC;
Zachary Turner4b083542015-04-17 22:40:36 +0000130
131 llvm::SmallVector<UTF16, 128> Path16;
132 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +0000133 return make_error<GenericError>(generic_error_code::invalid_path, Path);
Zachary Turner4b083542015-04-17 22:40:36 +0000134
135 const wchar_t *Path16Str = reinterpret_cast<const wchar_t *>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +0000136 HRESULT HR;
137 if (FAILED(HR = DiaDataSource->loadDataForExe(Path16Str, nullptr, nullptr)))
Zachary Turner38380322016-10-19 16:42:20 +0000138 return ErrorFromHResult(HR, "Calling loadDataForExe");
Zachary Turner4b083542015-04-17 22:40:36 +0000139
Zachary Turner819e77d2016-05-06 20:51:57 +0000140 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
Zachary Turner38380322016-10-19 16:42:20 +0000141 return ErrorFromHResult(HR, "Calling openSession");
Zachary Turner4b083542015-04-17 22:40:36 +0000142
143 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +0000144 return Error::success();
Zachary Turner4b083542015-04-17 22:40:36 +0000145}
146
Zachary Turnercffff262015-02-10 21:17:52 +0000147uint64_t DIASession::getLoadAddress() const {
148 uint64_t LoadAddress;
149 bool success = (S_OK == Session->get_loadAddress(&LoadAddress));
150 return (success) ? LoadAddress : 0;
151}
152
Aaron Smith89a19ac2018-02-23 00:02:27 +0000153bool DIASession::setLoadAddress(uint64_t Address) {
154 return (S_OK == Session->put_loadAddress(Address));
Zachary Turnercffff262015-02-10 21:17:52 +0000155}
156
Adrian McCarthy6a4b0802017-06-22 18:42:23 +0000157std::unique_ptr<PDBSymbolExe> DIASession::getGlobalScope() {
Zachary Turnercffff262015-02-10 21:17:52 +0000158 CComPtr<IDiaSymbol> GlobalScope;
159 if (S_OK != Session->get_globalScope(&GlobalScope))
160 return nullptr;
161
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000162 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, GlobalScope);
Zachary Turnercffff262015-02-10 21:17:52 +0000163 auto PdbSymbol(PDBSymbol::create(*this, std::move(RawSymbol)));
164 std::unique_ptr<PDBSymbolExe> ExeSymbol(
165 static_cast<PDBSymbolExe *>(PdbSymbol.release()));
166 return ExeSymbol;
167}
168
169std::unique_ptr<PDBSymbol> DIASession::getSymbolById(uint32_t SymbolId) const {
170 CComPtr<IDiaSymbol> LocatedSymbol;
171 if (S_OK != Session->symbolById(SymbolId, &LocatedSymbol))
172 return nullptr;
173
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000174 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, LocatedSymbol);
Zachary Turnercffff262015-02-10 21:17:52 +0000175 return PDBSymbol::create(*this, std::move(RawSymbol));
176}
177
Zachary Turner4b083542015-04-17 22:40:36 +0000178std::unique_ptr<PDBSymbol>
Zachary Turnere5cb2692015-05-01 20:24:26 +0000179DIASession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
180 enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
181
Zachary Turner4b083542015-04-17 22:40:36 +0000182 CComPtr<IDiaSymbol> Symbol;
Zachary Turnere5cb2692015-05-01 20:24:26 +0000183 if (S_OK != Session->findSymbolByVA(Address, EnumVal, &Symbol)) {
184 ULONGLONG LoadAddr = 0;
185 if (S_OK != Session->get_loadAddress(&LoadAddr))
186 return nullptr;
187 DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
188 if (S_OK != Session->findSymbolByRVA(RVA, EnumVal, &Symbol))
189 return nullptr;
190 }
Zachary Turner4b083542015-04-17 22:40:36 +0000191 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
192 return PDBSymbol::create(*this, std::move(RawSymbol));
193}
194
195std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner43ec3af2016-02-18 18:47:29 +0000196DIASession::findLineNumbers(const PDBSymbolCompiland &Compiland,
197 const IPDBSourceFile &File) const {
198 const DIARawSymbol &RawCompiland =
199 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
200 const DIASourceFile &RawFile = static_cast<const DIASourceFile &>(File);
201
202 CComPtr<IDiaEnumLineNumbers> LineNumbers;
203 if (S_OK !=
204 Session->findLines(RawCompiland.getDiaSymbol(), RawFile.getDiaFile(),
205 &LineNumbers))
206 return nullptr;
207
208 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
209}
210
211std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner4b083542015-04-17 22:40:36 +0000212DIASession::findLineNumbersByAddress(uint64_t Address, uint32_t Length) const {
213 CComPtr<IDiaEnumLineNumbers> LineNumbers;
Aaron Smith40198f52018-03-15 06:04:51 +0000214 if (S_OK != Session->findLinesByVA(Address, Length, &LineNumbers)) {
215 ULONGLONG LoadAddr = 0;
216 if (S_OK != Session->get_loadAddress(&LoadAddr))
217 return nullptr;
218 DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
219 if (S_OK != Session->findLinesByRVA(RVA, Length, &LineNumbers))
220 return nullptr;
221 }
222 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
223}
224
225std::unique_ptr<IPDBEnumLineNumbers>
226DIASession::findLineNumbersBySectOffset(uint32_t Section, uint32_t Offset,
227 uint32_t Length) const {
228 CComPtr<IDiaEnumLineNumbers> LineNumbers;
229 if (S_OK != Session->findLinesByAddr(Section, Offset, Length, &LineNumbers))
Zachary Turner4b083542015-04-17 22:40:36 +0000230 return nullptr;
231
232 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
233}
234
Zachary Turner43ec3af2016-02-18 18:47:29 +0000235std::unique_ptr<IPDBEnumSourceFiles>
236DIASession::findSourceFiles(const PDBSymbolCompiland *Compiland,
237 llvm::StringRef Pattern,
238 PDB_NameSearchFlags Flags) const {
239 IDiaSymbol *DiaCompiland = nullptr;
240 CComBSTR Utf16Pattern;
241 if (!Pattern.empty())
242 Utf16Pattern = CComBSTR(Pattern.data());
243
244 if (Compiland)
245 DiaCompiland = static_cast<const DIARawSymbol &>(Compiland->getRawSymbol())
246 .getDiaSymbol();
247
248 Flags = static_cast<PDB_NameSearchFlags>(
249 Flags | PDB_NameSearchFlags::NS_FileNameExtMatch);
250 CComPtr<IDiaEnumSourceFiles> SourceFiles;
251 if (S_OK !=
252 Session->findFile(DiaCompiland, Utf16Pattern.m_str, Flags, &SourceFiles))
253 return nullptr;
254 return llvm::make_unique<DIAEnumSourceFiles>(*this, SourceFiles);
255}
256
257std::unique_ptr<IPDBSourceFile>
258DIASession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
259 llvm::StringRef Pattern,
260 PDB_NameSearchFlags Flags) const {
261 auto SourceFiles = findSourceFiles(Compiland, Pattern, Flags);
262 if (!SourceFiles || SourceFiles->getChildCount() == 0)
263 return nullptr;
264 return SourceFiles->getNext();
265}
266
267std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
268DIASession::findCompilandsForSourceFile(llvm::StringRef Pattern,
269 PDB_NameSearchFlags Flags) const {
270 auto File = findOneSourceFile(nullptr, Pattern, Flags);
271 if (!File)
272 return nullptr;
273 return File->getCompilands();
274}
275
276std::unique_ptr<PDBSymbolCompiland>
277DIASession::findOneCompilandForSourceFile(llvm::StringRef Pattern,
278 PDB_NameSearchFlags Flags) const {
279 auto Compilands = findCompilandsForSourceFile(Pattern, Flags);
280 if (!Compilands || Compilands->getChildCount() == 0)
281 return nullptr;
282 return Compilands->getNext();
283}
284
Zachary Turnera5549172015-02-10 22:43:25 +0000285std::unique_ptr<IPDBEnumSourceFiles> DIASession::getAllSourceFiles() const {
286 CComPtr<IDiaEnumSourceFiles> Files;
287 if (S_OK != Session->findFile(nullptr, nullptr, nsNone, &Files))
288 return nullptr;
289
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000290 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000291}
292
293std::unique_ptr<IPDBEnumSourceFiles> DIASession::getSourceFilesForCompiland(
294 const PDBSymbolCompiland &Compiland) const {
295 CComPtr<IDiaEnumSourceFiles> Files;
296
297 const DIARawSymbol &RawSymbol =
298 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
299 if (S_OK !=
300 Session->findFile(RawSymbol.getDiaSymbol(), nullptr, nsNone, &Files))
301 return nullptr;
302
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000303 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000304}
305
Zachary Turnercffff262015-02-10 21:17:52 +0000306std::unique_ptr<IPDBSourceFile>
307DIASession::getSourceFileById(uint32_t FileId) const {
308 CComPtr<IDiaSourceFile> LocatedFile;
309 if (S_OK != Session->findFileById(FileId, &LocatedFile))
310 return nullptr;
311
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000312 return llvm::make_unique<DIASourceFile>(*this, LocatedFile);
Zachary Turnercffff262015-02-10 21:17:52 +0000313}
314
315std::unique_ptr<IPDBEnumDataStreams> DIASession::getDebugStreams() const {
316 CComPtr<IDiaEnumDebugStreams> DiaEnumerator;
317 if (S_OK != Session->getEnumDebugStreams(&DiaEnumerator))
318 return nullptr;
319
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000320 return llvm::make_unique<DIAEnumDebugStreams>(DiaEnumerator);
Zachary Turnercffff262015-02-10 21:17:52 +0000321}
Aaron Smith89bca9e2017-11-16 14:33:09 +0000322
323std::unique_ptr<IPDBEnumTables> DIASession::getEnumTables() const {
324 CComPtr<IDiaEnumTables> DiaEnumerator;
325 if (S_OK != Session->getEnumTables(&DiaEnumerator))
326 return nullptr;
327
328 return llvm::make_unique<DIAEnumTables>(DiaEnumerator);
329}
Zachary Turner679aead2018-03-13 17:46:06 +0000330
Aaron Smith58a32a42018-03-22 03:57:06 +0000331template <class T>
332static CComPtr<T>
333getTableEnumerator(IDiaSession &Session) {
334 CComPtr<T> Enumerator;
Zachary Turner679aead2018-03-13 17:46:06 +0000335 CComPtr<IDiaEnumTables> ET;
336 CComPtr<IDiaTable> Table;
337 ULONG Count = 0;
338
339 if (Session.getEnumTables(&ET) != S_OK)
340 return nullptr;
341
342 while (ET->Next(1, &Table, &Count) == S_OK && Count == 1) {
343 // There is only one table that matches the given iid
344 if (S_OK ==
Aaron Smith58a32a42018-03-22 03:57:06 +0000345 Table->QueryInterface(__uuidof(T), (void **)&Enumerator))
Zachary Turner679aead2018-03-13 17:46:06 +0000346 break;
347 Table.Release();
348 }
Aaron Smith58a32a42018-03-22 03:57:06 +0000349 return Enumerator;
Zachary Turner679aead2018-03-13 17:46:06 +0000350}
351std::unique_ptr<IPDBEnumInjectedSources>
352DIASession::getInjectedSources() const {
Aaron Smith58a32a42018-03-22 03:57:06 +0000353 CComPtr<IDiaEnumInjectedSources> Files =
354 getTableEnumerator<IDiaEnumInjectedSources>(*Session);
Zachary Turner679aead2018-03-13 17:46:06 +0000355 if (!Files)
356 return nullptr;
357
358 return llvm::make_unique<DIAEnumInjectedSources>(*this, Files);
359}
Aaron Smith523de052018-03-22 04:08:15 +0000360
361std::unique_ptr<IPDBEnumSectionContribs>
362DIASession::getSectionContribs() const {
363 CComPtr<IDiaEnumSectionContribs> Sections =
364 getTableEnumerator<IDiaEnumSectionContribs>(*Session);
365 if (!Sections)
366 return nullptr;
367
368 return llvm::make_unique<DIAEnumSectionContribs>(*this, Sections);
369}