//===--- ImageInspectionWin32.cpp - Win32 image inspection ----------------===// // // This source file is part of the Swift.org open source project // // Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors // Licensed under Apache License v2.0 with Runtime Library Exception // // See https://swift.org/LICENSE.txt for license information // See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors // //===----------------------------------------------------------------------===// #if !defined(__ELF__) && !defined(__MACH__) #include "ImageInspection.h" #include "ImageInspectionCOFF.h" #if defined(__CYGWIN__) #include #elif defined(_WIN32) #define WIN32_LEAN_AND_MEAN #define NOMINMAX #include #include #endif using namespace swift; namespace { static const swift::MetadataSections *registered = nullptr; void record(const swift::MetadataSections *sections) { if (registered == nullptr) { registered = sections; sections->next = sections->prev = sections; } else { registered->prev->next = sections; sections->next = registered; sections->prev = registered->prev; registered->prev = sections; } } } void swift::initializeProtocolLookup() { const swift::MetadataSections *sections = registered; while (true) { const swift::MetadataSections::Range &protocols = sections->swift5_protocols; if (protocols.length) addImageProtocolsBlockCallbackUnsafe( reinterpret_cast(protocols.start), protocols.length); if (sections->next == registered) break; sections = sections->next; } } void swift::initializeProtocolConformanceLookup() { const swift::MetadataSections *sections = registered; while (true) { const swift::MetadataSections::Range &conformances = sections->swift5_protocol_conformances; if (conformances.length) addImageProtocolConformanceBlockCallbackUnsafe( reinterpret_cast(conformances.start), conformances.length); if (sections->next == registered) break; sections = sections->next; } } void swift::initializeTypeMetadataRecordLookup() { const swift::MetadataSections *sections = registered; while (true) { const swift::MetadataSections::Range &type_metadata = sections->swift5_type_metadata; if (type_metadata.length) addImageTypeMetadataRecordBlockCallbackUnsafe( reinterpret_cast(type_metadata.start), type_metadata.length); if (sections->next == registered) break; sections = sections->next; } } void swift::initializeDynamicReplacementLookup() { } SWIFT_RUNTIME_EXPORT void swift_addNewDSOImage(const void *addr) { const swift::MetadataSections *sections = static_cast(addr); record(sections); const auto &protocols_section = sections->swift5_protocols; const void *protocols = reinterpret_cast(protocols_section.start); if (protocols_section.length) addImageProtocolsBlockCallback(protocols, protocols_section.length); const auto &protocol_conformances = sections->swift5_protocol_conformances; const void *conformances = reinterpret_cast(protocol_conformances.start); if (protocol_conformances.length) addImageProtocolConformanceBlockCallback(conformances, protocol_conformances.length); const auto &type_metadata = sections->swift5_type_metadata; const void *metadata = reinterpret_cast(type_metadata.start); if (type_metadata.length) addImageTypeMetadataRecordBlockCallback(metadata, type_metadata.length); const auto &dynamic_replacements = sections->swift5_repl; const auto *replacements = reinterpret_cast(dynamic_replacements.start); if (dynamic_replacements.length) { const auto &dynamic_replacements_some = sections->swift5_reps; const auto *replacements_some = reinterpret_cast(dynamic_replacements_some.start); addImageDynamicReplacementBlockCallback( replacements, dynamic_replacements.length, replacements_some, dynamic_replacements_some.length); } } int swift::lookupSymbol(const void *address, SymbolInfo *info) { #if defined(__CYGWIN__) Dl_info dlinfo; if (dladdr(address, &dlinfo) == 0) { return 0; } info->fileName = dlinfo.dli_fname; info->baseAddress = dlinfo.dli_fbase; info->symbolName = dli_info.dli_sname; info->symbolAddress = dli_saddr; return 1; #elif defined(_WIN32) static const constexpr size_t kSymbolMaxNameLen = 1024; static bool bInitialized = false; if (bInitialized == false) { if (SymInitialize(GetCurrentProcess(), /*UserSearchPath=*/NULL, /*fInvadeProcess=*/TRUE) == FALSE) return 0; bInitialized = true; } char buffer[sizeof(SYMBOL_INFO) + kSymbolMaxNameLen]; PSYMBOL_INFO pSymbol = reinterpret_cast(buffer); pSymbol->SizeOfStruct = sizeof(SYMBOL_INFO); pSymbol->MaxNameLen = kSymbolMaxNameLen; DWORD64 dwDisplacement = 0; if (SymFromAddr(GetCurrentProcess(), reinterpret_cast(address), &dwDisplacement, pSymbol) == FALSE) return 0; info->fileName = NULL; info->baseAddress = reinterpret_cast(pSymbol->ModBase); info->symbolName.reset(_strdup(pSymbol->Name)); info->symbolAddress = reinterpret_cast(pSymbol->Address); return 1; #else return 0; #endif // defined(__CYGWIN__) || defined(_WIN32) } // This is only used for backward deployment hooks, which we currently only support for // MachO. Add a stub here to make sure it still compiles. void *swift::lookupSection(const char *segment, const char *section, size_t *outSize) { return nullptr; } #endif // !defined(__ELF__) && !defined(__MACH__)