Skip to content

Commit

Permalink
feat: add support for archives
Browse files Browse the repository at this point in the history
This patch allows hotspot to open zip, tar, ... files
  • Loading branch information
lievenhey committed Sep 17, 2024
1 parent 51408f8 commit 7eb77b7
Showing 1 changed file with 137 additions and 65 deletions.
202 changes: 137 additions & 65 deletions src/parsers/perf/perfparser.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,12 @@
#include "settings.h"

#if KFArchive_FOUND
#include <K7Zip>
#include <KCompressionDevice>
#include <KTar>
#include <KZip>

#include <QMimeDatabase>
#endif

Q_LOGGING_CATEGORY(LOG_PERFPARSER, "hotspot.perfparser", QtWarningMsg)
Expand All @@ -56,11 +61,8 @@ QDataStream& operator>>(QDataStream& stream, Record& record)

QDebug operator<<(QDebug stream, const Record& record)
{
stream.noquote().nospace() << "Record{"
<< "pid=" << record.pid << ", "
<< "tid=" << record.tid << ", "
<< "time=" << record.time << ", "
<< "cpu=" << record.cpu << "}";
stream.noquote().nospace() << "Record{" << "pid=" << record.pid << ", " << "tid=" << record.tid << ", "
<< "time=" << record.time << ", " << "cpu=" << record.cpu << "}";
return stream;
}

Expand All @@ -76,8 +78,7 @@ QDataStream& operator>>(QDataStream& stream, StringId& stringId)

QDebug operator<<(QDebug stream, StringId stringId)
{
stream.noquote().nospace() << "String{"
<< "id=" << stringId.id << "}";
stream.noquote().nospace() << "String{" << "id=" << stringId.id << "}";
return stream;
}

Expand Down Expand Up @@ -109,8 +110,7 @@ QDataStream& operator>>(QDataStream& stream, AttributesDefinition& attributesDef

QDebug operator<<(QDebug stream, const AttributesDefinition& attributesDefinition)
{
stream.noquote().nospace() << "AttributesDefinition{"
<< "id=" << attributesDefinition.id << ", "
stream.noquote().nospace() << "AttributesDefinition{" << "id=" << attributesDefinition.id << ", "
<< "type=" << attributesDefinition.type << ", "
<< "config=" << attributesDefinition.config << ", "
<< "name=" << attributesDefinition.name << ", "
Expand All @@ -131,8 +131,8 @@ QDataStream& operator>>(QDataStream& stream, Command& command)

QDebug operator<<(QDebug stream, const Command& command)
{
stream.noquote().nospace() << "Command{" << static_cast<const Record&>(command) << ", "
<< "comm=" << command.comm << "}";
stream.noquote().nospace() << "Command{" << static_cast<const Record&>(command) << ", " << "comm=" << command.comm
<< "}";
return stream;
}

Expand Down Expand Up @@ -189,12 +189,9 @@ QDataStream& operator>>(QDataStream& stream, Location& location)

QDebug operator<<(QDebug stream, const Location& location)
{
stream.noquote().nospace() << "Location{"
<< "address=0x" << Qt::hex << location.address << Qt::dec << ", "
<< "relAddr=" << location.relAddr << ", "
<< "file=" << location.file << ", "
<< "pid=" << location.pid << ", "
<< "line=" << location.line << ", "
stream.noquote().nospace() << "Location{" << "address=0x" << Qt::hex << location.address << Qt::dec << ", "
<< "relAddr=" << location.relAddr << ", " << "file=" << location.file << ", "
<< "pid=" << location.pid << ", " << "line=" << location.line << ", "
<< "column=" << location.column << ", "
<< "parentLocationId=" << location.parentLocationId << "}";
return stream;
Expand All @@ -213,8 +210,7 @@ QDataStream& operator>>(QDataStream& stream, LocationDefinition& locationDefinit

QDebug operator<<(QDebug stream, const LocationDefinition& locationDefinition)
{
stream.noquote().nospace() << "LocationDefinition{"
<< "id=" << locationDefinition.id << ", "
stream.noquote().nospace() << "LocationDefinition{" << "id=" << locationDefinition.id << ", "
<< "location=" << locationDefinition.location << "}";
return stream;
}
Expand All @@ -239,15 +235,10 @@ QDataStream& operator>>(QDataStream& stream, Symbol& symbol)

QDebug operator<<(QDebug stream, const Symbol& symbol)
{
stream.noquote().nospace() << "Symbol{"
<< "name=" << symbol.name << ", "
<< "relAddr=" << symbol.relAddr << ", "
<< "size=" << symbol.size << ", "
<< "binary=" << symbol.binary << ", "
<< "path=" << symbol.path << ", "
<< "actualPath=" << symbol.actualPath << ", "
<< "isKernel=" << symbol.isKernel << ", "
<< "isInline=" << symbol.isInline << "}";
stream.noquote().nospace() << "Symbol{" << "name=" << symbol.name << ", " << "relAddr=" << symbol.relAddr << ", "
<< "size=" << symbol.size << ", " << "binary=" << symbol.binary << ", "
<< "path=" << symbol.path << ", " << "actualPath=" << symbol.actualPath << ", "
<< "isKernel=" << symbol.isKernel << ", " << "isInline=" << symbol.isInline << "}";
return stream;
}

Expand All @@ -264,8 +255,7 @@ QDataStream& operator>>(QDataStream& stream, SymbolDefinition& symbolDefinition)

QDebug operator<<(QDebug stream, const SymbolDefinition& symbolDefinition)
{
stream.noquote().nospace() << "SymbolDefinition{"
<< "id=" << symbolDefinition.id << ", "
stream.noquote().nospace() << "SymbolDefinition{" << "id=" << symbolDefinition.id << ", "
<< "symbol=" << symbolDefinition.symbol << "}";
return stream;
}
Expand All @@ -283,8 +273,7 @@ QDataStream& operator>>(QDataStream& stream, SampleCost& sampleCost)

QDebug operator<<(QDebug stream, SampleCost sampleCost)
{
stream.noquote().nospace() << "SampleCost{"
<< "attributeId=" << sampleCost.attributeId << ", "
stream.noquote().nospace() << "SampleCost{" << "attributeId=" << sampleCost.attributeId << ", "
<< "cost=" << sampleCost.cost << "}";
return stream;
}
Expand All @@ -303,10 +292,9 @@ QDataStream& operator>>(QDataStream& stream, Sample& sample)

QDebug operator<<(QDebug stream, const Sample& sample)
{
stream.noquote().nospace() << "Sample{" << static_cast<const Record&>(sample) << ", "
<< "frames=" << sample.frames << ", "
<< "guessedFrames=" << sample.guessedFrames << ", "
<< "costs=" << sample.costs << "}";
stream.noquote().nospace() << "Sample{" << static_cast<const Record&>(sample) << ", " << "frames=" << sample.frames
<< ", " << "guessedFrames=" << sample.guessedFrames << ", " << "costs=" << sample.costs
<< "}";
return stream;
}

Expand Down Expand Up @@ -340,8 +328,7 @@ QDataStream& operator>>(QDataStream& stream, StringDefinition& stringDefinition)

QDebug operator<<(QDebug stream, const StringDefinition& stringDefinition)
{
stream.noquote().nospace() << "StringDefinition{"
<< "id=" << stringDefinition.id << ", "
stream.noquote().nospace() << "StringDefinition{" << "id=" << stringDefinition.id << ", "
<< "string=" << stringDefinition.string << "}";
return stream;
}
Expand Down Expand Up @@ -377,9 +364,7 @@ QDataStream& operator>>(QDataStream& stream, BuildId& buildId)

QDebug operator<<(QDebug stream, const BuildId& buildId)
{
stream.noquote().nospace() << "BuildId{"
<< "pid=" << buildId.pid << ", "
<< "id=" << buildId.id.toHex() << ", "
stream.noquote().nospace() << "BuildId{" << "pid=" << buildId.pid << ", " << "id=" << buildId.id.toHex() << ", "
<< "fileName=" << buildId.fileName << "}";
return stream;
}
Expand All @@ -399,10 +384,8 @@ QDataStream& operator>>(QDataStream& stream, NumaNode& numaNode)

QDebug operator<<(QDebug stream, const NumaNode& numaNode)
{
stream.noquote().nospace() << "NumaNode{"
<< "nodeId=" << numaNode.nodeId << ", "
<< "memTotal=" << numaNode.memTotal << ", "
<< "memFree=" << numaNode.memFree << ", "
stream.noquote().nospace() << "NumaNode{" << "nodeId=" << numaNode.nodeId << ", "
<< "memTotal=" << numaNode.memTotal << ", " << "memFree=" << numaNode.memFree << ", "
<< "topology=" << numaNode.topology << "}";
return stream;
}
Expand All @@ -420,9 +403,7 @@ QDataStream& operator>>(QDataStream& stream, Pmu& pmu)

QDebug operator<<(QDebug stream, const Pmu& pmu)
{
stream.noquote().nospace() << "Pmu{"
<< "type=" << pmu.type << ", "
<< "name=" << pmu.name << "}";
stream.noquote().nospace() << "Pmu{" << "type=" << pmu.type << ", " << "name=" << pmu.name << "}";
return stream;
}

Expand All @@ -440,8 +421,7 @@ QDataStream& operator>>(QDataStream& stream, GroupDesc& groupDesc)

QDebug operator<<(QDebug stream, const GroupDesc& groupDesc)
{
stream.noquote().nospace() << "GroupDesc{"
<< "name=" << groupDesc.name << ", "
stream.noquote().nospace() << "GroupDesc{" << "name=" << groupDesc.name << ", "
<< "leaderIndex=" << groupDesc.leaderIndex << ", "
<< "numMembers=" << groupDesc.numMembers << "}";
return stream;
Expand Down Expand Up @@ -481,12 +461,10 @@ QDataStream& operator>>(QDataStream& stream, FeaturesDefinition& featuresDefinit

QDebug operator<<(QDebug stream, const FeaturesDefinition& featuresDefinition)
{
stream.noquote().nospace() << "FeaturesDefinition{"
<< "hostName=" << featuresDefinition.hostName << ", "
stream.noquote().nospace() << "FeaturesDefinition{" << "hostName=" << featuresDefinition.hostName << ", "
<< "osRelease=" << featuresDefinition.osRelease << ", "
<< "version=" << featuresDefinition.version << ", "
<< "arch=" << featuresDefinition.arch << ", "
<< "nrCpusOnline=" << featuresDefinition.nrCpusOnline << ", "
<< "version=" << featuresDefinition.version << ", " << "arch=" << featuresDefinition.arch
<< ", " << "nrCpusOnline=" << featuresDefinition.nrCpusOnline << ", "
<< "nrCpusAvailable=" << featuresDefinition.nrCpusAvailable << ", "
<< "cpuDesc=" << featuresDefinition.cpuDesc << ", "
<< "cpuId=" << featuresDefinition.cpuId << ", "
Expand Down Expand Up @@ -528,9 +506,7 @@ QDataStream& operator>>(QDataStream& stream, Error& error)

QDebug operator<<(QDebug stream, const Error& error)
{
stream.noquote().nospace() << "Error{"
<< "code=" << error.code << ", "
<< "message=" << error.message << "}";
stream.noquote().nospace() << "Error{" << "code=" << error.code << ", " << "message=" << error.message << "}";
return stream;
}

Expand Down Expand Up @@ -1991,19 +1967,112 @@ void PerfParser::exportResults(const QUrl& url)
});
}

// helper for extracting files from archives
namespace {
// create archive reader from mimetype
auto getArchiveFromMime = [](const QString& filename, const QMimeType& mimeType) -> std::unique_ptr<KArchive> {
if (mimeType.name() == QLatin1String("application/x-tar")) {
return std::make_unique<KTar>(filename);
} else if (mimeType.name() == QLatin1String("application/zip")) {
return std::make_unique<KZip>(filename);
} else if (mimeType.name() == QLatin1String("application/x-7z-compressed")) {
return std::make_unique<K7Zip>(filename);
}
return {};
};

auto extractFromArchive = [](const std::unique_ptr<KArchive>& archive) -> std::unique_ptr<QTemporaryFile> {
auto extractFile = [](const KArchiveDirectory* directory,
const QString& filename) -> std::unique_ptr<QTemporaryFile> {
auto extracted = std::make_unique<QTemporaryFile>();
extracted->open();

auto fileToExtract = directory->file(filename);
if (!fileToExtract) {
return {};
}

auto fileToExtractHandle = fileToExtract->createDevice();

const int chunkSize = 1024 * 100;

QByteArray buffer;
buffer.resize(chunkSize);

while (!fileToExtractHandle->atEnd()) {
const auto size = fileToExtractHandle->read(buffer.data(), buffer.size());
extracted->write(buffer.data(), size);
}
extracted->flush();

return extracted;
};

if (!archive->open(QIODevice::ReadOnly)) {
qWarning() << "Failed to open archive:" << archive->errorString();
return {};
}

auto dir = archive->directory();
auto entries = dir->entries();

if (entries.size() == 1) {
return extractFile(dir, entries[0]);
}

for (const auto& file : {QStringLiteral("perf.data"), QStringLiteral("perf.data.perfparser")}) {
if (entries.contains(file)) {
return extractFile(dir, file);
}
}

return {};
};
}

QString PerfParser::decompressIfNeeded(const QString& path)
{
#if KFArchive_FOUND
m_decompressed = std::make_unique<QTemporaryFile>(this);

KCompressionDevice compressedFile(path);

QMimeDatabase mimedb;

// detect archive type and extract perf.data file
auto extractArchive = [this, &mimedb](const QString& path) {
const auto mimetype = mimedb.mimeTypeForFile(path);
auto archive = getArchiveFromMime(path, mimetype);

if (!archive) {
// we don't have and archive -> return original file
return path;
}

auto extracted = extractFromArchive(archive);
if (extracted) {
m_decompressed = std::move(extracted);
}
return m_decompressed->fileName();
};

// uncompressed file -> check if it is an archive (tar for example)
// extractArchive returns the original path if it couldn't open the archive
if (compressedFile.compressionType() == KCompressionDevice::None) {
return extractArchive(path);
}

// we now have a compressed file that could be an archive
// we first extract it and then check if the mimetype to see if it is an archive

if (!compressedFile.open(QIODevice::ReadOnly)) {
// we failed to open the compressed file
qWarning() << "Failed to open:" << path;
return path;
}

if (compressedFile.open(QIODevice::ReadOnly)) {
m_decompressed->open();
// decompress compressed file
{
auto decompressed = std::make_unique<QTemporaryFile>();
decompressed->open();

const int chunkSize = 1024 * 100;

Expand All @@ -2012,13 +2081,16 @@ QString PerfParser::decompressIfNeeded(const QString& path)

while (!compressedFile.atEnd()) {
const auto size = compressedFile.read(buffer.data(), buffer.size());
m_decompressed->write(buffer.data(), size);
decompressed->write(buffer.data(), size);
}
m_decompressed->flush();
decompressed->flush();

compressedFile.close();
return m_decompressed->fileName();
m_decompressed = std::move(decompressed);
}

// if m_decompressed is not an archive, this will return m_decompressed
return extractArchive(m_decompressed->fileName());
#endif
// fallback
return path;
Expand Down

0 comments on commit 7eb77b7

Please sign in to comment.