#pragma once #if defined(__ELF__) && !defined(OS_FREEBSD) #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int ILLEGAL_COLUMN; extern const int ILLEGAL_TYPE_OF_ARGUMENT; extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH; } template class FunctionAddressToLineBase : public IFunction { public: size_t getNumberOfArguments() const override { return 1; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } DataTypePtr getReturnTypeImpl(const ColumnsWithTypeAndName & arguments) const override { if (arguments.size() != 1) throw Exception(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, "Function {} needs exactly one argument; passed {}.", getName(), arguments.size()); const auto & type = arguments[0].type; if (!WhichDataType(type.get()).isUInt64()) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "The only argument for function {} must be UInt64. " "Found {} instead.", getName(), type->getName()); return getDataType(); } bool useDefaultImplementationForConstants() const override { return true; } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr &, size_t input_rows_count) const override { const ColumnPtr & column = arguments[0].column; const ColumnUInt64 * column_concrete = checkAndGetColumn(column.get()); if (!column_concrete) throw Exception(ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of argument of function {}", column->getName(), getName()); const typename ColumnVector::Container & data = column_concrete->getData(); return getResultColumn(data, input_rows_count); } protected: virtual DataTypePtr getDataType() const = 0; virtual ColumnPtr getResultColumn(const typename ColumnVector::Container & data, size_t input_rows_count) const = 0; virtual void setResult(ResultT & result, const Dwarf::LocationInfo & location, const std::vector & frames) const = 0; struct Cache { std::mutex mutex; Arena arena; using Map = HashMap; Map map; std::unordered_map dwarfs; }; mutable Cache cache; ResultT impl(uintptr_t addr) const { const SymbolIndex & symbol_index = SymbolIndex::instance(); if (const auto * object = symbol_index.findObject(reinterpret_cast(addr))) { auto dwarf_it = cache.dwarfs.try_emplace(object->name, object->elf).first; if (!std::filesystem::exists(object->name)) return {}; Dwarf::LocationInfo location; std::vector frames; // NOTE: not used in FAST mode. ResultT result; if (dwarf_it->second.findAddress(addr - uintptr_t(object->address_begin), location, locationInfoMode, frames)) { setResult(result, location, frames); return result; } return {object->name}; } return {}; } ResultT implCached(uintptr_t addr) const { typename Cache::Map::LookupResult it; bool inserted; std::lock_guard lock(cache.mutex); cache.map.emplace(addr, it, inserted); if (inserted) it->getMapped() = impl(addr); return it->getMapped(); } }; } #endif