/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Qt Software Information (qt-info@nokia.com) ** ** Commercial Usage ** ** Licensees holding valid Qt Commercial licenses may use this file in ** accordance with the Qt Commercial License Agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Nokia. ** ** GNU Lesser General Public License Usage ** ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** If you are unsure which license is appropriate for your use, please ** contact the sales department at qt-sales@nokia.com. ** **************************************************************************/ #include "debuggerdialogs.h" #include "ui_attachcoredialog.h" #include "ui_attachexternaldialog.h" #include "ui_attachremotedialog.h" #include "ui_startexternaldialog.h" #ifdef Q_OS_WIN # include "dbgwinutils.h" #endif #include #include #include #include #include #include #include #include #include #include namespace Debugger { namespace Internal { bool operator<(const ProcData &p1, const ProcData &p2) { return p1.name < p2.name; } // A filterable process list model class ProcessListFilterModel : public QSortFilterProxyModel { public: explicit ProcessListFilterModel(QObject *parent); QString processIdAt(const QModelIndex &index) const; void populate(QList processes, const QString &excludePid = QString()); private: QStandardItemModel *m_model; }; ProcessListFilterModel::ProcessListFilterModel(QObject *parent) : QSortFilterProxyModel(parent), m_model(new QStandardItemModel(this)) { QStringList columns; columns << AttachExternalDialog::tr("Process ID") << AttachExternalDialog::tr("Name") << AttachExternalDialog::tr("State"); m_model->setHorizontalHeaderLabels(columns); setSourceModel(m_model); setFilterCaseSensitivity(Qt::CaseInsensitive); setFilterKeyColumn(1); } QString ProcessListFilterModel::processIdAt(const QModelIndex &index) const { if (index.isValid()) { const QModelIndex index0 = mapToSource(index); QModelIndex index = index0.sibling(index0.row(), 0); if (const QStandardItem *item = m_model->itemFromIndex(index)) return item->text(); } return QString(); } void ProcessListFilterModel::populate(QList processes, const QString &excludePid) { qStableSort(processes); if (const int rowCount = m_model->rowCount()) m_model->removeRows(0, rowCount); QStandardItem *root = m_model->invisibleRootItem(); foreach(const ProcData &proc, processes) { QList row; row.append(new QStandardItem(proc.ppid)); row.append(new QStandardItem(proc.name)); if (!proc.image.isEmpty()) row.back()->setToolTip(proc.image); row.append(new QStandardItem(proc.state)); if (proc.ppid == excludePid) foreach(QStandardItem *i, row) i->setEnabled(false); root->appendRow(row); } } /////////////////////////////////////////////////////////////////////// // // AttachCoreDialog // /////////////////////////////////////////////////////////////////////// AttachCoreDialog::AttachCoreDialog(QWidget *parent) : QDialog(parent), m_ui(new Ui::AttachCoreDialog) { m_ui->setupUi(this); m_ui->execFileName->setExpectedKind(Core::Utils::PathChooser::File); m_ui->execFileName->setPromptDialogTitle(tr("Select Executable")); m_ui->coreFileName->setExpectedKind(Core::Utils::PathChooser::File); m_ui->coreFileName->setPromptDialogTitle(tr("Select Executable")); m_ui->buttonBox->button(QDialogButtonBox::Ok)->setDefault(true); connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept())); connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); } AttachCoreDialog::~AttachCoreDialog() { delete m_ui; } QString AttachCoreDialog::executableFile() const { return m_ui->execFileName->path(); } void AttachCoreDialog::setExecutableFile(const QString &fileName) { m_ui->execFileName->setPath(fileName); } QString AttachCoreDialog::coreFile() const { return m_ui->coreFileName->path(); } void AttachCoreDialog::setCoreFile(const QString &fileName) { m_ui->coreFileName->setPath(fileName); } /////////////////////////////////////////////////////////////////////// // // process model helpers // /////////////////////////////////////////////////////////////////////// static bool isUnixProcessId(const QString &procname) { for (int i = 0; i != procname.size(); ++i) if (!procname.at(i).isDigit()) return false; return true; } // Determine UNIX processes by reading "/proc" static QList unixProcessList() { QList rc; const QStringList procIds = QDir(QLatin1String("/proc/")).entryList(); if (procIds.isEmpty()) return rc; foreach (const QString &procId, procIds) { if (!isUnixProcessId(procId)) continue; QString filename = QLatin1String("/proc/"); filename += procId; filename += QLatin1String("/stat"); QFile file(filename); file.open(QIODevice::ReadOnly); const QStringList data = QString::fromLocal8Bit(file.readAll()).split(' '); ProcData proc; proc.ppid = procId; proc.name = data.at(1); if (proc.name.startsWith(QLatin1Char('(')) && proc.name.endsWith(QLatin1Char(')'))) { proc.name.truncate(proc.name.size() - 1); proc.name.remove(0, 1); } proc.state = data.at(2); // PPID is element 3 rc.push_back(proc); } return rc; } static QList processList() { #ifdef Q_OS_WIN return winProcessList(); #else return unixProcessList(); #endif } /////////////////////////////////////////////////////////////////////// // // AttachExternalDialog // /////////////////////////////////////////////////////////////////////// AttachExternalDialog::AttachExternalDialog(QWidget *parent) : QDialog(parent), m_selfPid(QString::number(QCoreApplication::applicationPid())), m_ui(new Ui::AttachExternalDialog), m_model(new ProcessListFilterModel(this)) { m_ui->setupUi(this); okButton()->setDefault(true); okButton()->setEnabled(false); m_ui->procView->setModel(m_model); m_ui->procView->setSortingEnabled(true); connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept())); connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); QPushButton *refreshButton = new QPushButton(tr("Refresh")); connect(refreshButton, SIGNAL(clicked()), this, SLOT(rebuildProcessList())); m_ui->buttonBox->addButton(refreshButton, QDialogButtonBox::ActionRole); // Do not use activated, will be single click in Oxygen connect(m_ui->procView, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(procSelected(QModelIndex))); connect(m_ui->pidLineEdit, SIGNAL(textChanged(QString)), this, SLOT(pidChanged(QString))); connect(m_ui->filterClearToolButton, SIGNAL(clicked()), m_ui->filterLineEdit, SLOT(clear())); connect(m_ui->filterLineEdit, SIGNAL(textChanged(QString)), m_model, SLOT(setFilterFixedString(QString))); rebuildProcessList(); } AttachExternalDialog::~AttachExternalDialog() { delete m_ui; } QPushButton *AttachExternalDialog::okButton() const { return m_ui->buttonBox->button(QDialogButtonBox::Ok); } void AttachExternalDialog::rebuildProcessList() { m_model->populate(processList(), m_selfPid); m_ui->procView->expandAll(); m_ui->procView->resizeColumnToContents(0); m_ui->procView->resizeColumnToContents(1); } void AttachExternalDialog::procSelected(const QModelIndex &proxyIndex) { const QString proccessId = m_model->processIdAt(proxyIndex); if (!proccessId.isEmpty()) { m_ui->pidLineEdit->setText(proccessId); if (okButton()->isEnabled()) okButton()->animateClick(); } } int AttachExternalDialog::attachPID() const { return m_ui->pidLineEdit->text().toInt(); } void AttachExternalDialog::pidChanged(const QString &pid) { okButton()->setEnabled(!pid.isEmpty() && pid != QLatin1String("0") && pid != m_selfPid); } /////////////////////////////////////////////////////////////////////// // // AttachRemoteDialog // /////////////////////////////////////////////////////////////////////// AttachRemoteDialog::AttachRemoteDialog(QWidget *parent, const QString &pid) : QDialog(parent), m_ui(new Ui::AttachRemoteDialog), m_model(new ProcessListFilterModel(this)) { m_ui->setupUi(this); m_ui->buttonBox->button(QDialogButtonBox::Ok)->setDefault(true); m_defaultPID = pid; m_ui->procView->setModel(m_model); m_ui->procView->setSortingEnabled(true); connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept())); connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); // Do not use activated, will be single click in Oxygen connect(m_ui->procView, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(procSelected(QModelIndex))); QPushButton *refreshButton = new QPushButton(tr("Refresh")); connect(refreshButton, SIGNAL(clicked()), this, SLOT(rebuildProcessList())); m_ui->buttonBox->addButton(refreshButton, QDialogButtonBox::ActionRole); connect(m_ui->pidLineEdit, SIGNAL(textChanged(QString)), this, SLOT(pidChanged(QString))); connect(m_ui->filterClearToolButton, SIGNAL(clicked()), m_ui->filterLineEdit, SLOT(clear())); connect(m_ui->filterLineEdit, SIGNAL(textChanged(QString)), m_model, SLOT(setFilterFixedString(QString))); m_ui->pidLineEdit->setText(m_defaultPID); rebuildProcessList(); } AttachRemoteDialog::~AttachRemoteDialog() { delete m_ui; } QPushButton *AttachRemoteDialog::okButton() const { return m_ui->buttonBox->button(QDialogButtonBox::Ok); } void AttachRemoteDialog::rebuildProcessList() { m_model->populate(processList()); m_ui->procView->expandAll(); m_ui->procView->resizeColumnToContents(0); m_ui->procView->resizeColumnToContents(1); } void AttachRemoteDialog::procSelected(const QModelIndex &index0) { const QString proccessId = m_model->processIdAt(index0); if (!proccessId.isEmpty()) { m_ui->pidLineEdit->setText(proccessId); if (okButton()->isEnabled()) okButton()->animateClick(); } } int AttachRemoteDialog::attachPID() const { return m_ui->pidLineEdit->text().toInt(); } void AttachRemoteDialog::pidChanged(const QString &pid) { okButton()->setEnabled(!pid.isEmpty() && pid != QLatin1String("0")); } /////////////////////////////////////////////////////////////////////// // // StartExternalDialog // /////////////////////////////////////////////////////////////////////// StartExternalDialog::StartExternalDialog(QWidget *parent) : QDialog(parent), m_ui(new Ui::StartExternalDialog) { m_ui->setupUi(this); m_ui->execFile->setExpectedKind(Core::Utils::PathChooser::File); m_ui->execFile->setPromptDialogTitle(tr("Select Executable")); m_ui->buttonBox->button(QDialogButtonBox::Ok)->setDefault(true); //execLabel->setHidden(false); //execEdit->setHidden(false); //browseButton->setHidden(false); m_ui->execLabel->setText(tr("Executable:")); m_ui->argLabel->setText(tr("Arguments:")); connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept())); connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); } StartExternalDialog::~StartExternalDialog() { delete m_ui; } void StartExternalDialog::setExecutableFile(const QString &str) { m_ui->execFile->setPath(str); } void StartExternalDialog::setExecutableArguments(const QString &str) { m_ui->argsEdit->setText(str); } QString StartExternalDialog::executableFile() const { return m_ui->execFile->path(); } QString StartExternalDialog::executableArguments() const { return m_ui->argsEdit->text(); } } }