Files
qt-creator/src/plugins/debugger/debuggerdialogs.cpp

654 lines
19 KiB
C++
Raw Normal View History

2009-03-02 14:04:03 +01:00
/**************************************************************************
**
** This file is part of Qt Creator
**
2010-03-05 11:25:49 +01:00
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
2009-03-02 14:04:03 +01:00
**
** Contact: Nokia Corporation (qt-info@nokia.com)
2009-03-02 14:04:03 +01:00
**
** 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
2009-08-14 09:30:56 +02:00
** contact the sales department at http://qt.nokia.com/contact.
2009-03-02 14:04:03 +01:00
**
**************************************************************************/
#include "debuggerdialogs.h"
#include "debuggerconstants.h"
2009-03-02 14:04:03 +01:00
#include "ui_attachcoredialog.h"
#include "ui_attachexternaldialog.h"
#include "ui_startexternaldialog.h"
#include "ui_startremotedialog.h"
2009-03-04 16:00:43 +01:00
#ifdef Q_OS_WIN
# include "shared/dbgwinutils.h"
2009-03-04 16:00:43 +01:00
#endif
#include <coreplugin/icore.h>
#include <utils/synchronousprocess.h>
2009-03-02 14:04:03 +01:00
#include <QtCore/QDebug>
#include <QtCore/QProcess>
2009-03-02 14:04:03 +01:00
#include <QtCore/QDir>
#include <QtCore/QFile>
#include <QtCore/QCoreApplication>
2009-03-02 14:04:03 +01:00
#include <QtGui/QStandardItemModel>
#include <QtGui/QHeaderView>
#include <QtGui/QFileDialog>
#include <QtGui/QPushButton>
2009-03-04 16:00:43 +01:00
#include <QtGui/QProxyModel>
#include <QtGui/QSortFilterProxyModel>
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;
QString executableForPid(const QString& pid) const;
void populate(QList<ProcData> processes, const QString &excludePid = QString());
private:
enum { processImageRole = Qt::UserRole };
bool lessThan(const QModelIndex &left, const QModelIndex &right) const;
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);
}
bool ProcessListFilterModel::lessThan(const QModelIndex &left,
const QModelIndex &right) const
{
const QString l = sourceModel()->data(left).toString();
const QString r = sourceModel()->data(right).toString();
if (left.column() == 0)
return l.toInt() < r.toInt();
return l < r;
}
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();
}
QString ProcessListFilterModel::executableForPid(const QString &pid) const
{
const int rowCount = m_model->rowCount();
for (int r = 0; r < rowCount; r++) {
const QStandardItem *item = m_model->item(r, 0);
if (item->text() == pid)
return item->data(processImageRole).toString();
}
return QString();
}
void ProcessListFilterModel::populate(QList<ProcData> 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<QStandardItem *> row;
row.append(new QStandardItem(proc.ppid));
row.front()->setData(QVariant(proc.image), processImageRole);
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);
}
}
2009-03-02 14:04:03 +01:00
2009-03-02 14:04:03 +01:00
///////////////////////////////////////////////////////////////////////
//
// AttachCoreDialog
//
///////////////////////////////////////////////////////////////////////
AttachCoreDialog::AttachCoreDialog(QWidget *parent)
: QDialog(parent), m_ui(new Ui::AttachCoreDialog)
2009-03-02 14:04:03 +01:00
{
m_ui->setupUi(this);
m_ui->execFileName->setExpectedKind(Utils::PathChooser::File);
m_ui->execFileName->setPromptDialogTitle(tr("Select Executable"));
m_ui->coreFileName->setExpectedKind(Utils::PathChooser::File);
m_ui->coreFileName->setPromptDialogTitle(tr("Select Core File"));
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setDefault(true);
2009-03-02 14:04:03 +01:00
connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept()));
connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject()));
}
AttachCoreDialog::~AttachCoreDialog()
{
delete m_ui;
}
2009-03-02 14:04:03 +01:00
QString AttachCoreDialog::executableFile() const
{
return m_ui->execFileName->path();
2009-03-02 14:04:03 +01:00
}
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);
}
2009-03-02 14:04:03 +01:00
2009-03-02 14:04:03 +01:00
///////////////////////////////////////////////////////////////////////
//
// Process model helpers
2009-03-02 14:04:03 +01:00
//
///////////////////////////////////////////////////////////////////////
static bool isUnixProcessId(const QString &procname)
2009-03-02 14:04:03 +01:00
{
for (int i = 0; i != procname.size(); ++i)
if (!procname.at(i).isDigit())
return false;
return true;
}
// Determine UNIX processes by running ps
static QList<ProcData> unixProcessListPS()
{
#ifdef Q_OS_MAC
static const char formatC[] = "pid state command";
#else
static const char formatC[] = "pid,state,cmd";
#endif
QList<ProcData> rc;
QProcess psProcess;
QStringList args;
args << QLatin1String("-e") << QLatin1String("-o") << QLatin1String(formatC);
psProcess.start(QLatin1String("ps"), args);
if (!psProcess.waitForStarted())
return rc;
QByteArray output;
if (!Utils::SynchronousProcess::readDataFromProcess(psProcess, 30000, &output))
return rc;
// Split "457 S+ /Users/foo.app"
const QStringList lines = QString::fromLocal8Bit(output).split(QLatin1Char('\n'));
const int lineCount = lines.size();
const QChar blank = QLatin1Char(' ');
for (int l = 1; l < lineCount; l++) { // Skip header
const QString line = lines.at(l).simplified();
const int pidSep = line.indexOf(blank);
const int cmdSep = pidSep != -1 ? line.indexOf(blank, pidSep + 1) : -1;
if (cmdSep > 0) {
ProcData procData;
procData.ppid = line.left(pidSep);
procData.state = line.mid(pidSep + 1, cmdSep - pidSep - 1);
procData.name = line.mid(cmdSep + 1);
rc.push_back(procData);
}
}
return rc;
}
// Determine UNIX processes by reading "/proc". Default to ps if
// it does not exist
2009-03-04 16:00:43 +01:00
static QList<ProcData> unixProcessList()
2009-03-02 14:04:03 +01:00
{
const QDir procDir(QLatin1String("/proc/"));
if (!procDir.exists())
return unixProcessListPS();
2009-03-04 16:00:43 +01:00
QList<ProcData> rc;
const QStringList procIds = procDir.entryList();
if (procIds.isEmpty())
2009-03-04 16:00:43 +01:00
return rc;
foreach (const QString &procId, procIds) {
if (!isUnixProcessId(procId))
2009-03-02 14:04:03 +01:00
continue;
2009-03-04 16:00:43 +01:00
QString filename = QLatin1String("/proc/");
filename += procId;
2009-03-04 16:00:43 +01:00
filename += QLatin1String("/stat");
2009-03-02 14:04:03 +01:00
QFile file(filename);
if (!file.open(QIODevice::ReadOnly))
continue; // process may have exited
2009-03-04 16:00:43 +01:00
const QStringList data = QString::fromLocal8Bit(file.readAll()).split(' ');
2009-03-02 14:04:03 +01:00
ProcData proc;
proc.ppid = procId;
2009-03-02 14:04:03 +01:00
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);
}
2009-03-02 14:04:03 +01:00
proc.state = data.at(2);
// PPID is element 3
2009-03-04 16:00:43 +01:00
rc.push_back(proc);
2009-03-02 14:04:03 +01:00
}
2009-03-04 16:00:43 +01:00
return rc;
}
2009-03-02 14:04:03 +01:00
static QList<ProcData> processList()
2009-03-04 16:00:43 +01:00
{
#ifdef Q_OS_WIN
return winProcessList();
2009-03-04 16:00:43 +01:00
#else
return unixProcessList();
2009-03-04 16:00:43 +01:00
#endif
}
2009-03-04 16:00:43 +01:00
///////////////////////////////////////////////////////////////////////
//
// AttachExternalDialog
//
///////////////////////////////////////////////////////////////////////
AttachExternalDialog::AttachExternalDialog(QWidget *parent)
: QDialog(parent),
m_selfPid(QString::number(QCoreApplication::applicationPid())),
m_ui(new Ui::AttachExternalDialog),
m_model(new ProcessListFilterModel(this))
2009-03-04 16:00:43 +01:00
{
m_ui->setupUi(this);
okButton()->setDefault(true);
okButton()->setEnabled(false);
2009-03-04 16:00:43 +01:00
m_ui->procView->setModel(m_model);
2009-03-04 16:00:43 +01:00
m_ui->procView->setSortingEnabled(true);
m_ui->procView->sortByColumn(1, Qt::AscendingOrder);
2009-03-04 16:00:43 +01:00
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);
m_ui->filterWidget->setFocus(Qt::TabFocusReason);
2009-03-04 16:00:43 +01:00
m_ui->procView->setAlternatingRowColors(true);
m_ui->procView->setRootIsDecorated(false);
m_ui->procView->setUniformRowHeights(true);
// Do not use activated, will be single click in Oxygen
connect(m_ui->procView, SIGNAL(doubleClicked(QModelIndex)),
2009-03-04 16:00:43 +01:00
this, SLOT(procSelected(QModelIndex)));
connect(m_ui->procView, SIGNAL(clicked(QModelIndex)),
this, SLOT(procClicked(QModelIndex)));
connect(m_ui->pidLineEdit, SIGNAL(textChanged(QString)),
this, SLOT(pidChanged(QString)));
connect(m_ui->filterWidget, SIGNAL(filterChanged(QString)),
this, SLOT(setFilterString(QString)));
2009-03-04 16:00:43 +01:00
setMinimumHeight(500);
2009-03-04 16:00:43 +01:00
rebuildProcessList();
}
AttachExternalDialog::~AttachExternalDialog()
{
delete m_ui;
}
void AttachExternalDialog::setFilterString(const QString &filter)
{
m_model->setFilterFixedString(filter);
// Activate the line edit if there's a unique filtered process.
QString processId;
if (m_model->rowCount(QModelIndex()) == 1)
processId = m_model->processIdAt(m_model->index(0, 0, QModelIndex()));
m_ui->pidLineEdit->setText(processId);
pidChanged(processId);
}
QPushButton *AttachExternalDialog::okButton() const
{
return m_ui->buttonBox->button(QDialogButtonBox::Ok);
}
2009-03-04 16:00:43 +01:00
void AttachExternalDialog::rebuildProcessList()
{
m_model->populate(processList(), m_selfPid);
m_ui->procView->expandAll();
m_ui->procView->resizeColumnToContents(0);
m_ui->procView->resizeColumnToContents(1);
2009-03-02 14:04:03 +01:00
}
2009-03-04 16:00:43 +01:00
void AttachExternalDialog::procSelected(const QModelIndex &proxyIndex)
2009-03-02 14:04:03 +01:00
{
const QString processId = m_model->processIdAt(proxyIndex);
if (!processId.isEmpty()) {
m_ui->pidLineEdit->setText(processId);
if (okButton()->isEnabled())
okButton()->animateClick();
2009-03-04 16:00:43 +01:00
}
2009-03-02 14:04:03 +01:00
}
void AttachExternalDialog::procClicked(const QModelIndex &proxyIndex)
{
const QString processId = m_model->processIdAt(proxyIndex);
if (!processId.isEmpty())
m_ui->pidLineEdit->setText(processId);
}
QString AttachExternalDialog::attachPIDText() const
{
return m_ui->pidLineEdit->text().trimmed();
}
qint64 AttachExternalDialog::attachPID() const
2009-03-02 14:04:03 +01:00
{
return attachPIDText().toLongLong();
}
QString AttachExternalDialog::executable() const
{
// Search pid in model in case the user typed in the PID.
return m_model->executableForPid(attachPIDText());
2009-03-02 14:04:03 +01:00
}
void AttachExternalDialog::pidChanged(const QString &pid)
{
bool enabled = !pid.isEmpty() && pid != QLatin1String("0") && pid != m_selfPid;;
okButton()->setEnabled(enabled);
}
///////////////////////////////////////////////////////////////////////
//
// StartExternalDialog
//
///////////////////////////////////////////////////////////////////////
StartExternalDialog::StartExternalDialog(QWidget *parent)
: QDialog(parent), m_ui(new Ui::StartExternalDialog)
{
m_ui->setupUi(this);
m_ui->execFile->setExpectedKind(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);
}
QString StartExternalDialog::executableFile() const
{
return m_ui->execFile->path();
}
void StartExternalDialog::setExecutableArguments(const QString &str)
{
m_ui->argsEdit->setText(str);
}
QString StartExternalDialog::executableArguments() const
{
return m_ui->argsEdit->text();
}
bool StartExternalDialog::breakAtMain() const
{
return m_ui->checkBoxBreakAtMain->isChecked();
}
2009-03-02 14:04:03 +01:00
///////////////////////////////////////////////////////////////////////
//
// StartRemoteDialog
2009-03-02 14:04:03 +01:00
//
///////////////////////////////////////////////////////////////////////
StartRemoteDialog::StartRemoteDialog(QWidget *parent)
: QDialog(parent),
m_ui(new Ui::StartRemoteDialog)
2009-03-02 14:04:03 +01:00
{
m_ui->setupUi(this);
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setDefault(true);
m_ui->debuggerPathChooser->setExpectedKind(Utils::PathChooser::File);
m_ui->debuggerPathChooser->setPromptDialogTitle(tr("Select Debugger"));
m_ui->executablePathChooser->setExpectedKind(Utils::PathChooser::File);
m_ui->executablePathChooser->setPromptDialogTitle(tr("Select Executable"));
m_ui->sysrootPathChooser->setPromptDialogTitle(tr("Select Sysroot"));
m_ui->serverStartScript->setExpectedKind(Utils::PathChooser::File);
m_ui->serverStartScript->setPromptDialogTitle(tr("Select Start Script"));
connect(m_ui->useServerStartScriptCheckBox, SIGNAL(toggled(bool)),
this, SLOT(updateState()));
connect(m_ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept()));
connect(m_ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject()));
updateState();
2009-03-02 14:04:03 +01:00
}
StartRemoteDialog::~StartRemoteDialog()
{
delete m_ui;
}
void StartRemoteDialog::setRemoteChannel(const QString &channel)
{
m_ui->channelLineEdit->setText(channel);
}
QString StartRemoteDialog::remoteChannel() const
2009-03-02 14:04:03 +01:00
{
return m_ui->channelLineEdit->text();
2009-03-02 14:04:03 +01:00
}
void StartRemoteDialog::setLocalExecutable(const QString &executable)
{
m_ui->executablePathChooser->setPath(executable);
}
QString StartRemoteDialog::localExecutable() const
{
return m_ui->executablePathChooser->path();
}
void StartRemoteDialog::setDebugger(const QString &debugger)
{
m_ui->debuggerPathChooser->setPath(debugger);
}
QString StartRemoteDialog::debugger() const
{
return m_ui->debuggerPathChooser->path();
}
void StartRemoteDialog::setRemoteArchitectures(const QStringList &list)
{
m_ui->architectureComboBox->clear();
if (!list.isEmpty()) {
m_ui->architectureComboBox->insertItems(0, list);
m_ui->architectureComboBox->setCurrentIndex(0);
}
2009-03-02 14:04:03 +01:00
}
void StartRemoteDialog::setRemoteArchitecture(const QString &arch)
2009-03-02 14:04:03 +01:00
{
int index = m_ui->architectureComboBox->findText(arch);
if (index != -1)
m_ui->architectureComboBox->setCurrentIndex(index);
2009-03-02 14:04:03 +01:00
}
QString StartRemoteDialog::remoteArchitecture() const
{
return m_ui->architectureComboBox->currentText();
}
void StartRemoteDialog::setServerStartScript(const QString &scriptName)
{
m_ui->serverStartScript->setPath(scriptName);
}
QString StartRemoteDialog::serverStartScript() const
{
return m_ui->serverStartScript->path();
}
void StartRemoteDialog::setUseServerStartScript(bool on)
2009-03-02 14:04:03 +01:00
{
m_ui->useServerStartScriptCheckBox->setChecked(on);
2009-03-02 14:04:03 +01:00
}
bool StartRemoteDialog::useServerStartScript() const
2009-03-02 14:04:03 +01:00
{
return m_ui->useServerStartScriptCheckBox->isChecked();
2009-03-02 14:04:03 +01:00
}
void StartRemoteDialog::setSysRoot(const QString &sysroot)
{
m_ui->sysrootPathChooser->setPath(sysroot);
}
QString StartRemoteDialog::sysRoot() const
{
return m_ui->sysrootPathChooser->path();
}
void StartRemoteDialog::updateState()
2009-03-02 14:04:03 +01:00
{
bool enabled = m_ui->useServerStartScriptCheckBox->isChecked();
m_ui->serverStartScriptLabel->setEnabled(enabled);
m_ui->serverStartScript->setEnabled(enabled);
2009-03-02 14:04:03 +01:00
}
AddressDialog::AddressDialog(QWidget *parent) :
QDialog(parent),
m_lineEdit(new QLineEdit),
m_box(new QDialogButtonBox(QDialogButtonBox::Ok|QDialogButtonBox::Cancel))
{
setWindowTitle(tr("Select start address"));
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
QHBoxLayout *hLayout = new QHBoxLayout;
hLayout->addWidget(new QLabel(tr("Enter an address: ")));
hLayout->addWidget(m_lineEdit);
QVBoxLayout *vLayout = new QVBoxLayout;
vLayout->addLayout(hLayout);
vLayout->addWidget(m_box);
setLayout(vLayout);
connect(m_box, SIGNAL(accepted()), this, SLOT(accept()));
connect(m_box, SIGNAL(rejected()), this, SLOT(reject()));
connect(m_lineEdit, SIGNAL(returnPressed()), this, SLOT(accept()));
connect(m_lineEdit, SIGNAL(textChanged(QString)), this, SLOT(textChanged()));
setOkButtonEnabled(false);
}
void AddressDialog::setOkButtonEnabled(bool v)
{
m_box->button(QDialogButtonBox::Ok)->setEnabled(v);
}
bool AddressDialog::isOkButtonEnabled() const
{
return m_box->button(QDialogButtonBox::Ok)->isEnabled();
}
quint64 AddressDialog::address() const
{
return m_lineEdit->text().toULongLong(0, 16);
}
void AddressDialog::accept()
{
if (isOkButtonEnabled())
QDialog::accept();
}
void AddressDialog::textChanged()
{
setOkButtonEnabled(isValid());
}
bool AddressDialog::isValid() const
{
const QString text = m_lineEdit->text();
bool ok = false;
text.toULongLong(&ok, 16);
return ok;
}
} // namespace Internal
} // namespace Debugger