File indexing completed on 2025-10-19 05:29:27

0001 /*
0002    This file is part of Massif Visualizer
0003 
0004    Copyright 2010 Milian Wolff <mail@milianw.de>
0005 
0006    This library is free software; you can redistribute it and/or
0007    modify it under the terms of the GNU Lesser General Public
0008    License as published by the Free Software Foundation; either
0009    version 2.1 of the License, or (at your option) version 3, or any
0010    later version accepted by the membership of KDE e.V. (or its
0011    successor approved by the membership of KDE e.V.), which shall
0012    act as a proxy defined in Section 6 of version 3 of the license.
0013 
0014    This library is distributed in the hope that it will be useful,
0015    but WITHOUT ANY WARRANTY; without even the implied warranty of
0016    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
0017    Lesser General Public License for more details.
0018 
0019    You should have received a copy of the GNU Lesser General Public
0020    License along with this library.  If not, see <http://www.gnu.org/licenses/>.
0021 */
0022 
0023 #include "dotgraphgenerator.h"
0024 
0025 #include "massifdata/filedata.h"
0026 #include "massifdata/snapshotitem.h"
0027 #include "massifdata/treeleafitem.h"
0028 #include "massifdata/util.h"
0029 
0030 #include <QTextStream>
0031 #include <QFile>
0032 #include <QColor>
0033 #include <QDebug>
0034 
0035 #include <KLocalizedString>
0036 
0037 
0038 namespace Massif {
0039 
0040 struct GraphNode {
0041     const TreeLeafItem* item;
0042     // incoming calls + cost
0043     QHash<GraphNode*, quint64> children;
0044     // outgoing calls
0045     QVector<GraphNode*> parents;
0046     quint64 accumulatedCost;
0047     bool visited;
0048     quint32 belowThresholdCount;
0049     quint64 belowThresholdCost;
0050 };
0051 
0052 }
0053 
0054 Q_DECLARE_TYPEINFO(Massif::GraphNode, Q_MOVABLE_TYPE);
0055 
0056 using namespace Massif;
0057 
0058 DotGraphGenerator::DotGraphGenerator(const SnapshotItem* snapshot, const QString& timeUnit, QObject* parent)
0059     : QThread(parent)
0060     , m_snapshot(snapshot)
0061     , m_node(snapshot->heapTree())
0062     , m_canceled(false)
0063     , m_timeUnit(timeUnit)
0064     , m_highestCost(0)
0065 {
0066     m_file.open();
0067 }
0068 
0069 DotGraphGenerator::DotGraphGenerator(const TreeLeafItem* node, const QString& timeUnit, QObject* parent)
0070     : QThread(parent)
0071     , m_snapshot(nullptr)
0072     , m_node(node)
0073     , m_canceled(false)
0074     , m_timeUnit(timeUnit)
0075     , m_highestCost(0)
0076 {
0077     m_file.open();
0078 }
0079 
0080 DotGraphGenerator::~DotGraphGenerator()
0081 {
0082     qDebug() << "closing generator, file will get removed";
0083 }
0084 
0085 void DotGraphGenerator::cancel()
0086 {
0087     m_canceled = true;
0088 }
0089 
0090 QString getLabel(const TreeLeafItem* node)
0091 {
0092     QByteArray label = prettyLabel(node->label());
0093     const int lineWidth = 40;
0094     if (label.length() > lineWidth) {
0095         int lastPos = 0;
0096         int lastBreak = 0;
0097         while (true) {
0098             lastPos = label.indexOf(',', lastPos);
0099             if (lastPos == -1) {
0100                 break;
0101             } else if (lastPos - lastBreak > lineWidth) {
0102                 label.insert(lastPos, "\\n\\ \\ ");
0103                 lastPos = lastPos + 4;
0104                 lastBreak = lastPos;
0105                 continue;
0106             } else {
0107                 lastPos++;
0108             }
0109         }
0110     }
0111     return QString::fromUtf8(label);
0112 }
0113 
0114 QString getColor(quint64 cost, quint64 maxCost)
0115 {
0116     Q_ASSERT(cost <= maxCost);
0117     const double ratio = (double(cost) / maxCost);
0118     Q_ASSERT(ratio <= 1.0);
0119     return QColor::fromHsv(120 - ratio * 120, (-((ratio-1) * (ratio-1))) * 255 + 255, 255, 255).name();
0120 //     return QColor::fromHsv(120 - ratio * 120, 255, 255).name();
0121 }
0122 
0123 GraphNode* buildGraph(const TreeLeafItem* item, QMultiHash<QByteArray, GraphNode*>& knownNodes,
0124                       quint64& maxCost, GraphNode* parent = nullptr)
0125 {
0126     // merge below-threshold items
0127     if (parent && item->children().isEmpty()) {
0128         static QRegExp matchBT(QStringLiteral("in ([0-9]+) places, all below massif's threshold"),
0129                                                 Qt::CaseSensitive, QRegExp::RegExp2);
0130         if (matchBT.indexIn(QString::fromLatin1(item->label())) != -1) {
0131             parent->belowThresholdCost += item->cost();
0132             parent->belowThresholdCount += matchBT.cap(1).toInt();
0133         }
0134         return nullptr;
0135     }
0136     GraphNode* node = knownNodes.value(item->label(), nullptr);
0137     if (!node) {
0138         node = new GraphNode;
0139         knownNodes.insert(item->label(), node);
0140         node->item = item;
0141         node->accumulatedCost = 0;
0142         node->visited = false;
0143         node->belowThresholdCost = 0;
0144         node->belowThresholdCount = 0;
0145     }
0146 
0147     if (parent && !node->parents.contains(parent)) {
0148         node->parents << parent;
0149     }
0150 
0151     node->accumulatedCost += item->cost();
0152 
0153     if (node->accumulatedCost > maxCost) {
0154         maxCost = node->accumulatedCost;
0155     }
0156 
0157     foreach(TreeLeafItem* child, item->children()) {
0158         GraphNode* childNode = buildGraph(child, knownNodes, maxCost, node);
0159         if (!childNode) {
0160             // was below-threshold item
0161             continue;
0162         }
0163         QMultiHash< GraphNode*, quint64 >::iterator it = node->children.find(childNode);
0164         if (it != node->children.end()) {
0165             it.value() += child->cost();
0166         } else {
0167             node->children.insert(childNode, child->cost());
0168         }
0169     }
0170 
0171     return node;
0172 }
0173 
0174 void DotGraphGenerator::run()
0175 {
0176     if (!m_file.isOpen()) {
0177         qWarning() << "could not create temp file for writing Dot-graph";
0178         return;
0179     }
0180 
0181     if (m_canceled) {
0182         return;
0183     }
0184 
0185     qDebug() << "creating new dot file in" << m_file.fileName();
0186     QTextStream out(&m_file);
0187 
0188     out << "digraph callgraph {\n"
0189            "rankdir = BT;\n";
0190     if (m_canceled) {
0191         return;
0192     }
0193 
0194     QString parentId;
0195     if (m_snapshot) {
0196         // also show some info about the selected snapshot
0197         parentId = QString::number((qint64) m_snapshot);
0198         const QString label = i18n("snapshot #%1 (taken at %2%4)\\nheap cost: %3",
0199                                 m_snapshot->number(), m_snapshot->time(), prettyCost(m_snapshot->cost()),
0200                                 m_timeUnit);
0201         out << '"' << parentId << "\" [shape=box,label=\"" << label << "\",fillcolor=white];\n";
0202 
0203         m_maxCost = m_snapshot->cost();
0204     } else if (m_node) {
0205         const TreeLeafItem* topMost = m_node;
0206         while (topMost->parent()) {
0207             topMost = topMost->parent();
0208         }
0209         m_maxCost = topMost->cost();
0210     }
0211 
0212     if (m_node) {
0213         QMultiHash<QByteArray, GraphNode*> nodes;
0214         GraphNode* root = buildGraph(m_node, nodes, m_maxCost);
0215         m_highestCost = 0;
0216         nodeToDot(root, out, parentId, 0);
0217         qDeleteAll(nodes);
0218     }
0219     out << "}\n";
0220     m_file.flush();
0221 }
0222 
0223 void DotGraphGenerator::nodeToDot(GraphNode* node, QTextStream& out, const QString& parentId, quint64 cost)
0224 {
0225     if (m_canceled) {
0226         return;
0227     }
0228 
0229     const QString nodeId = QString::number((qint64) node);
0230     // write edge with annotated cost
0231     if (!parentId.isEmpty()) {
0232         // edge
0233         out << '"' << nodeId << "\" -> \"" << parentId << '"';
0234         if (cost) {
0235             out << " [label = \"" << prettyCost(cost) << "\"]";
0236         }
0237         out << ";\n";
0238     }
0239 
0240     if (node->visited) {
0241         // don't visit children again - the edge is all we need
0242         return;
0243     }
0244     node->visited = true;
0245 
0246     const bool isRoot = m_snapshot && m_snapshot->heapTree() == node->item;
0247 
0248     // first item we find will be the most cost-intensive one
0249     ///TODO this should take accumulated cost into account!
0250     if (m_highestCost < node->accumulatedCost && !isRoot) {
0251         m_costlyGraphvizId = nodeId;
0252         m_highestCost = node->accumulatedCost;
0253     }
0254 
0255 
0256     QString label = getLabel(node->item);
0257     // group nodes with same cost but different label
0258     // but only if they don't have any other outgoing calls, i.e. parents.size() = 1
0259     bool wasGrouped = false;
0260     while (node && node->children.count() == 1)
0261     {
0262         GraphNode* child = node->children.begin().key();
0263         if (child->accumulatedCost != node->accumulatedCost || node->parents.size() != 1 || child->belowThresholdCount ) {
0264             break;
0265         }
0266         if (m_canceled) {
0267             return;
0268         }
0269         node = child;
0270 
0271         label += QLatin1String(" | ") + QString::fromUtf8(prettyLabel(node->item->label()));
0272         wasGrouped = true;
0273     }
0274 
0275     QString shape;
0276     if (wasGrouped) {
0277         label = QLatin1Char('{') + label + QLatin1Char('}');
0278         // <...> would be an id, escape it
0279         label = label.replace(QLatin1Char('<'), QLatin1String("\\<"));
0280         label = label.replace(QLatin1Char('>'), QLatin1String("\\>"));
0281         shape = QStringLiteral("record");
0282     } else {
0283         shape = QStringLiteral("box");
0284     }
0285 
0286     const QString color = isRoot ? QStringLiteral("white") : getColor(node->accumulatedCost, m_maxCost);
0287     out << '"' << nodeId << "\" [shape=" << shape << ",label=\"" << label << "\",fillcolor=\"" << color << "\"];\n";
0288     if (!node) {
0289         return;
0290     }
0291 
0292     QMultiHash< GraphNode*, quint64 >::const_iterator it = node->children.constBegin();
0293     while(it != node->children.constEnd()) {
0294         if (m_canceled) {
0295             return;
0296         }
0297         nodeToDot(it.key(), out, nodeId, it.value());
0298         ++it;
0299     }
0300     // handle below-threshold
0301     if (node->belowThresholdCount) {
0302         // node
0303         const QString btLabel = i18np("in one place below threshold", "in %1 places, all below threshold", node->belowThresholdCount);
0304         out << '"' << nodeId << "-bt\" [shape=box,label=\"" << btLabel
0305             << "\",fillcolor=\"" << getColor(node->belowThresholdCost, m_maxCost) << "\"];\n";
0306         // edge
0307         out << '"' << nodeId << "-bt\" -> \"" << nodeId << "\" [label =\"" << prettyCost(node->belowThresholdCost) << "\"];\n";
0308     }
0309 }
0310 
0311 QString DotGraphGenerator::outputFile() const
0312 {
0313     return m_file.fileName();
0314 }
0315 
0316 QString DotGraphGenerator::mostCostIntensiveGraphvizId() const
0317 {
0318     return m_costlyGraphvizId;
0319 }
0320 
0321 #include "moc_dotgraphgenerator.cpp"