2014-12-20 19:10:02 +00:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 Denis Shienkov <denis.shienkov@gmail.com>
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2014-12-20 19:10:02 +00:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2016-01-15 14:57:40 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2014-12-20 19:10:02 +00:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2014-12-20 19:10:02 +00:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
#include "stlinkutilgdbserverprovider.h"
|
2014-12-20 19:10:02 +00:00
|
|
|
|
2019-11-04 00:51:58 +03:00
|
|
|
#include <baremetal/baremetalconstants.h>
|
|
|
|
|
#include <baremetal/debugserverprovidermanager.h>
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
#include <utils/fileutils.h>
|
2014-12-20 19:10:02 +00:00
|
|
|
#include <utils/pathchooser.h>
|
|
|
|
|
#include <utils/qtcassert.h>
|
2020-09-18 12:11:40 +02:00
|
|
|
#include <utils/variablechooser.h>
|
2014-12-20 19:10:02 +00:00
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
#include <QCheckBox>
|
|
|
|
|
#include <QComboBox>
|
2014-12-20 19:10:02 +00:00
|
|
|
#include <QFormLayout>
|
|
|
|
|
#include <QPlainTextEdit>
|
2019-05-08 13:49:19 +03:00
|
|
|
#include <QSpinBox>
|
2019-06-04 10:17:06 +02:00
|
|
|
|
|
|
|
|
using namespace Utils;
|
2014-12-20 19:10:02 +00:00
|
|
|
|
|
|
|
|
namespace BareMetal {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
const char executableFileKeyC[] = "BareMetal.StLinkUtilGdbServerProvider.ExecutableFile";
|
|
|
|
|
const char verboseLevelKeyC[] = "BareMetal.StLinkUtilGdbServerProvider.VerboseLevel";
|
|
|
|
|
const char extendedModeKeyC[] = "BareMetal.StLinkUtilGdbServerProvider.ExtendedMode";
|
|
|
|
|
const char resetBoardKeyC[] = "BareMetal.StLinkUtilGdbServerProvider.ResetBoard";
|
|
|
|
|
const char transportLayerKeyC[] = "BareMetal.StLinkUtilGdbServerProvider.TransportLayer";
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
// StLinkUtilGdbServerProvider
|
|
|
|
|
|
2014-12-20 19:10:02 +00:00
|
|
|
StLinkUtilGdbServerProvider::StLinkUtilGdbServerProvider()
|
2019-11-27 15:49:55 +03:00
|
|
|
: GdbServerProvider(Constants::GDBSERVER_STLINK_UTIL_PROVIDER_ID)
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
|
|
|
|
setInitCommands(defaultInitCommands());
|
|
|
|
|
setResetCommands(defaultResetCommands());
|
2019-11-28 16:20:38 +03:00
|
|
|
setChannel("localhost", 4242);
|
2019-07-23 16:29:33 +02:00
|
|
|
setSettingsKeyBase("BareMetal.StLinkUtilGdbServerProvider");
|
2020-01-15 18:53:49 +01:00
|
|
|
setTypeDisplayName(GdbServerProvider::tr("ST-LINK Utility"));
|
2019-12-12 17:41:43 +01:00
|
|
|
setConfigurationWidgetCreator([this] { return new StLinkUtilGdbServerProviderConfigWidget(this); });
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString StLinkUtilGdbServerProvider::defaultInitCommands()
|
|
|
|
|
{
|
2019-11-05 10:06:31 +01:00
|
|
|
return {"load\n"};
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString StLinkUtilGdbServerProvider::defaultResetCommands()
|
|
|
|
|
{
|
2019-05-08 14:42:54 +03:00
|
|
|
return {};
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-07-23 16:29:33 +02:00
|
|
|
QString StLinkUtilGdbServerProvider::channelString() const
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
|
|
|
|
switch (startupMode()) {
|
|
|
|
|
case StartupOnNetwork:
|
|
|
|
|
// Just return as "host:port" form.
|
2019-07-23 16:29:33 +02:00
|
|
|
return GdbServerProvider::channelString();
|
2014-12-20 19:10:02 +00:00
|
|
|
case StartupOnPipe:
|
|
|
|
|
// Unsupported mode
|
2019-05-08 14:42:54 +03:00
|
|
|
return {};
|
2014-12-20 19:10:02 +00:00
|
|
|
default: // wrong
|
2019-05-08 14:42:54 +03:00
|
|
|
return {};
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-04 10:17:06 +02:00
|
|
|
CommandLine StLinkUtilGdbServerProvider::command() const
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
2019-06-04 10:17:06 +02:00
|
|
|
CommandLine cmd{m_executableFile, {}};
|
2014-12-20 19:10:02 +00:00
|
|
|
|
|
|
|
|
if (m_extendedMode)
|
2019-06-04 10:17:06 +02:00
|
|
|
cmd.addArg("--multi");
|
2014-12-20 19:10:02 +00:00
|
|
|
|
|
|
|
|
if (!m_resetBoard)
|
2019-06-04 10:17:06 +02:00
|
|
|
cmd.addArg("--no-reset");
|
2014-12-20 19:10:02 +00:00
|
|
|
|
2019-06-04 10:17:06 +02:00
|
|
|
cmd.addArg("--stlink_version=" + QString::number(m_transport));
|
2019-07-23 16:29:33 +02:00
|
|
|
cmd.addArg("--listen_port=" + QString::number(channel().port()));
|
2019-06-04 10:17:06 +02:00
|
|
|
cmd.addArg("--verbose=" + QString::number(m_verboseLevel));
|
2014-12-20 19:10:02 +00:00
|
|
|
|
2019-06-04 10:17:06 +02:00
|
|
|
return cmd;
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-11-26 17:41:39 +03:00
|
|
|
QSet<GdbServerProvider::StartupMode>
|
|
|
|
|
StLinkUtilGdbServerProvider::supportedStartupModes() const
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
2019-11-26 17:41:39 +03:00
|
|
|
return {StartupOnNetwork};
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool StLinkUtilGdbServerProvider::isValid() const
|
|
|
|
|
{
|
|
|
|
|
if (!GdbServerProvider::isValid())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
const StartupMode m = startupMode();
|
|
|
|
|
|
2019-11-26 17:41:39 +03:00
|
|
|
if (m == StartupOnNetwork) {
|
2019-07-23 16:29:33 +02:00
|
|
|
if (channel().host().isEmpty())
|
2014-12-20 19:10:02 +00:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (m == StartupOnNetwork) {
|
|
|
|
|
if (m_executableFile.isEmpty())
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariantMap StLinkUtilGdbServerProvider::toMap() const
|
|
|
|
|
{
|
|
|
|
|
QVariantMap data = GdbServerProvider::toMap();
|
2019-11-04 00:51:58 +03:00
|
|
|
data.insert(executableFileKeyC, m_executableFile.toVariant());
|
|
|
|
|
data.insert(verboseLevelKeyC, m_verboseLevel);
|
|
|
|
|
data.insert(extendedModeKeyC, m_extendedMode);
|
|
|
|
|
data.insert(resetBoardKeyC, m_resetBoard);
|
|
|
|
|
data.insert(transportLayerKeyC, m_transport);
|
2014-12-20 19:10:02 +00:00
|
|
|
return data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool StLinkUtilGdbServerProvider::fromMap(const QVariantMap &data)
|
|
|
|
|
{
|
|
|
|
|
if (!GdbServerProvider::fromMap(data))
|
|
|
|
|
return false;
|
|
|
|
|
|
2019-11-04 00:51:58 +03:00
|
|
|
m_executableFile = FilePath::fromVariant(data.value(executableFileKeyC));
|
|
|
|
|
m_verboseLevel = data.value(verboseLevelKeyC).toInt();
|
|
|
|
|
m_extendedMode = data.value(extendedModeKeyC).toBool();
|
|
|
|
|
m_resetBoard = data.value(resetBoardKeyC).toBool();
|
2014-12-20 19:10:02 +00:00
|
|
|
m_transport = static_cast<TransportLayer>(
|
2019-11-04 00:51:58 +03:00
|
|
|
data.value(transportLayerKeyC).toInt());
|
2014-12-20 19:10:02 +00:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 00:51:58 +03:00
|
|
|
bool StLinkUtilGdbServerProvider::operator==(const IDebugServerProvider &other) const
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
|
|
|
|
if (!GdbServerProvider::operator==(other))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
const auto p = static_cast<const StLinkUtilGdbServerProvider *>(&other);
|
2019-07-23 16:29:33 +02:00
|
|
|
return m_executableFile == p->m_executableFile
|
2014-12-20 19:10:02 +00:00
|
|
|
&& m_verboseLevel == p->m_verboseLevel
|
|
|
|
|
&& m_extendedMode == p->m_extendedMode
|
|
|
|
|
&& m_resetBoard == p->m_resetBoard
|
|
|
|
|
&& m_transport == p->m_transport;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
// StLinkUtilGdbServerProviderFactory
|
|
|
|
|
|
2014-12-20 19:10:02 +00:00
|
|
|
StLinkUtilGdbServerProviderFactory::StLinkUtilGdbServerProviderFactory()
|
|
|
|
|
{
|
2019-11-27 15:49:55 +03:00
|
|
|
setId(Constants::GDBSERVER_STLINK_UTIL_PROVIDER_ID);
|
2020-01-15 18:53:49 +01:00
|
|
|
setDisplayName(GdbServerProvider::tr("ST-LINK Utility"));
|
2019-12-12 18:01:53 +01:00
|
|
|
setCreator([] { return new StLinkUtilGdbServerProvider; });
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
// StLinkUtilGdbServerProviderConfigWidget
|
|
|
|
|
|
2014-12-20 19:10:02 +00:00
|
|
|
StLinkUtilGdbServerProviderConfigWidget::StLinkUtilGdbServerProviderConfigWidget(
|
|
|
|
|
StLinkUtilGdbServerProvider *p)
|
|
|
|
|
: GdbServerProviderConfigWidget(p)
|
|
|
|
|
{
|
|
|
|
|
Q_ASSERT(p);
|
|
|
|
|
|
|
|
|
|
m_hostWidget = new HostWidget(this);
|
|
|
|
|
m_mainLayout->addRow(tr("Host:"), m_hostWidget);
|
|
|
|
|
|
|
|
|
|
m_executableFileChooser = new Utils::PathChooser;
|
|
|
|
|
m_executableFileChooser->setExpectedKind(Utils::PathChooser::ExistingCommand);
|
|
|
|
|
m_mainLayout->addRow(tr("Executable file:"), m_executableFileChooser);
|
|
|
|
|
|
|
|
|
|
m_verboseLevelSpinBox = new QSpinBox;
|
|
|
|
|
m_verboseLevelSpinBox->setRange(0, 99);
|
|
|
|
|
m_verboseLevelSpinBox->setToolTip(tr("Specify the verbosity level (0..99)."));
|
|
|
|
|
m_mainLayout->addRow(tr("Verbosity level:"), m_verboseLevelSpinBox);
|
|
|
|
|
|
|
|
|
|
m_extendedModeCheckBox = new QCheckBox;
|
|
|
|
|
m_extendedModeCheckBox->setToolTip(tr("Continue listening for connections "
|
|
|
|
|
"after disconnect."));
|
|
|
|
|
m_mainLayout->addRow(tr("Extended mode:"), m_extendedModeCheckBox);
|
|
|
|
|
|
|
|
|
|
m_resetBoardCheckBox = new QCheckBox;
|
|
|
|
|
m_resetBoardCheckBox->setToolTip(tr("Reset board on connection."));
|
|
|
|
|
m_mainLayout->addRow(tr("Reset on connection:"), m_resetBoardCheckBox);
|
|
|
|
|
|
|
|
|
|
m_transportLayerComboBox = new QComboBox;
|
|
|
|
|
m_transportLayerComboBox->setToolTip(tr("Transport layer type."));
|
|
|
|
|
m_mainLayout->addRow(tr("Version:"), m_transportLayerComboBox);
|
|
|
|
|
|
|
|
|
|
m_initCommandsTextEdit = new QPlainTextEdit(this);
|
|
|
|
|
m_initCommandsTextEdit->setToolTip(defaultInitCommandsTooltip());
|
|
|
|
|
m_mainLayout->addRow(tr("Init commands:"), m_initCommandsTextEdit);
|
|
|
|
|
m_resetCommandsTextEdit = new QPlainTextEdit(this);
|
|
|
|
|
m_resetCommandsTextEdit->setToolTip(defaultResetCommandsTooltip());
|
|
|
|
|
m_mainLayout->addRow(tr("Reset commands:"), m_resetCommandsTextEdit);
|
|
|
|
|
|
|
|
|
|
populateTransportLayers();
|
|
|
|
|
addErrorLabel();
|
|
|
|
|
setFromProvider();
|
|
|
|
|
|
2020-09-18 12:11:40 +02:00
|
|
|
const auto chooser = new VariableChooser(this);
|
2015-11-29 18:38:21 +03:00
|
|
|
chooser->addSupportedWidget(m_initCommandsTextEdit);
|
|
|
|
|
chooser->addSupportedWidget(m_resetCommandsTextEdit);
|
2015-02-04 08:12:01 +01:00
|
|
|
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_hostWidget, &HostWidget::dataChanged,
|
2014-12-20 19:10:02 +00:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_executableFileChooser, &Utils::PathChooser::rawPathChanged,
|
2014-12-20 19:10:02 +00:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
|
|
|
|
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_verboseLevelSpinBox,
|
2019-05-08 14:24:28 +03:00
|
|
|
QOverload<int>::of(&QSpinBox::valueChanged),
|
2015-06-09 10:46:54 +02:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_extendedModeCheckBox, &QAbstractButton::clicked,
|
2015-06-09 10:46:54 +02:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_resetBoardCheckBox, &QAbstractButton::clicked,
|
2015-06-09 10:46:54 +02:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_transportLayerComboBox,
|
2019-05-08 14:24:28 +03:00
|
|
|
QOverload<int>::of(&QComboBox::currentIndexChanged),
|
2015-06-09 10:46:54 +02:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2014-12-20 19:10:02 +00:00
|
|
|
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_initCommandsTextEdit, &QPlainTextEdit::textChanged,
|
2014-12-20 19:10:02 +00:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
2015-11-29 18:38:21 +03:00
|
|
|
connect(m_resetCommandsTextEdit, &QPlainTextEdit::textChanged,
|
2014-12-20 19:10:02 +00:00
|
|
|
this, &GdbServerProviderConfigWidget::dirty);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 00:51:58 +03:00
|
|
|
void StLinkUtilGdbServerProviderConfigWidget::apply()
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
2019-11-04 00:51:58 +03:00
|
|
|
const auto p = static_cast<StLinkUtilGdbServerProvider *>(m_provider);
|
2014-12-20 19:10:02 +00:00
|
|
|
Q_ASSERT(p);
|
|
|
|
|
|
2019-07-23 16:29:33 +02:00
|
|
|
p->setChannel(m_hostWidget->channel());
|
2020-04-09 11:05:50 +02:00
|
|
|
p->m_executableFile = m_executableFileChooser->filePath();
|
2014-12-20 19:10:02 +00:00
|
|
|
p->m_verboseLevel = m_verboseLevelSpinBox->value();
|
|
|
|
|
p->m_extendedMode = m_extendedModeCheckBox->isChecked();
|
|
|
|
|
p->m_resetBoard = m_resetBoardCheckBox->isChecked();
|
|
|
|
|
p->m_transport = transportLayer();
|
|
|
|
|
p->setInitCommands(m_initCommandsTextEdit->toPlainText());
|
|
|
|
|
p->setResetCommands(m_resetCommandsTextEdit->toPlainText());
|
2019-11-04 00:51:58 +03:00
|
|
|
GdbServerProviderConfigWidget::apply();
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-11-04 00:51:58 +03:00
|
|
|
void StLinkUtilGdbServerProviderConfigWidget::discard()
|
2014-12-20 19:10:02 +00:00
|
|
|
{
|
|
|
|
|
setFromProvider();
|
2019-11-04 00:51:58 +03:00
|
|
|
GdbServerProviderConfigWidget::discard();
|
2014-12-20 19:10:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
StLinkUtilGdbServerProvider::TransportLayer
|
|
|
|
|
StLinkUtilGdbServerProviderConfigWidget::transportLayerFromIndex(int idx) const
|
|
|
|
|
{
|
|
|
|
|
return static_cast<StLinkUtilGdbServerProvider::TransportLayer>(
|
|
|
|
|
m_transportLayerComboBox->itemData(idx).toInt());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
StLinkUtilGdbServerProvider::TransportLayer
|
|
|
|
|
StLinkUtilGdbServerProviderConfigWidget::transportLayer() const
|
|
|
|
|
{
|
|
|
|
|
const int idx = m_transportLayerComboBox->currentIndex();
|
|
|
|
|
return transportLayerFromIndex(idx);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void StLinkUtilGdbServerProviderConfigWidget::setTransportLayer(
|
|
|
|
|
StLinkUtilGdbServerProvider::TransportLayer tl)
|
|
|
|
|
{
|
|
|
|
|
for (int idx = 0; idx < m_transportLayerComboBox->count(); ++idx) {
|
|
|
|
|
if (tl == transportLayerFromIndex(idx)) {
|
|
|
|
|
m_transportLayerComboBox->setCurrentIndex(idx);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void StLinkUtilGdbServerProviderConfigWidget::populateTransportLayers()
|
|
|
|
|
{
|
|
|
|
|
m_transportLayerComboBox->insertItem(
|
2015-03-16 14:54:06 +01:00
|
|
|
m_transportLayerComboBox->count(), tr("ST-LINK/V1"),
|
2014-12-20 19:10:02 +00:00
|
|
|
StLinkUtilGdbServerProvider::ScsiOverUsb);
|
|
|
|
|
m_transportLayerComboBox->insertItem(
|
2015-03-16 14:54:06 +01:00
|
|
|
m_transportLayerComboBox->count(), tr("ST-LINK/V2"),
|
2014-12-20 19:10:02 +00:00
|
|
|
StLinkUtilGdbServerProvider::RawUsb);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void StLinkUtilGdbServerProviderConfigWidget::setFromProvider()
|
|
|
|
|
{
|
2019-11-04 00:51:58 +03:00
|
|
|
const auto p = static_cast<StLinkUtilGdbServerProvider *>(m_provider);
|
2014-12-20 19:10:02 +00:00
|
|
|
Q_ASSERT(p);
|
|
|
|
|
|
2019-05-08 16:18:16 +03:00
|
|
|
const QSignalBlocker blocker(this);
|
2019-07-23 16:29:33 +02:00
|
|
|
m_hostWidget->setChannel(p->channel());
|
2020-04-09 11:05:50 +02:00
|
|
|
m_executableFileChooser->setFilePath(p->m_executableFile);
|
2014-12-20 19:10:02 +00:00
|
|
|
m_verboseLevelSpinBox->setValue(p->m_verboseLevel);
|
|
|
|
|
m_extendedModeCheckBox->setChecked(p->m_extendedMode);
|
|
|
|
|
m_resetBoardCheckBox->setChecked(p->m_resetBoard);
|
|
|
|
|
setTransportLayer(p->m_transport);
|
|
|
|
|
m_initCommandsTextEdit->setPlainText(p->initCommands());
|
|
|
|
|
m_resetCommandsTextEdit->setPlainText(p->resetCommands());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ProjectExplorer
|