s60devicerunconfiguration.cpp 34.3 KB
Newer Older
con's avatar
con committed
1
2
3
4
/**************************************************************************
**
** This file is part of Qt Creator
**
hjk's avatar
hjk committed
5
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
con's avatar
con committed
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
**
** Contact: Nokia Corporation (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
hjk's avatar
hjk committed
26
** contact the sales department at http://qt.nokia.com/contact.
con's avatar
con committed
27
28
29
**
**************************************************************************/

30
#include "s60devicerunconfiguration.h"
31
#include "s60devicerunconfigurationwidget.h"
32
#include "qt4project.h"
Tobias Hunger's avatar
Tobias Hunger committed
33
#include "qt4target.h"
34
35
36
37
#include "qtversionmanager.h"
#include "profilereader.h"
#include "s60manager.h"
#include "s60devices.h"
38
39
#include "s60runconfigbluetoothstarter.h"
#include "bluetoothlistener_gui.h"
40
#include "symbiandevicemanager.h"
41
#include "qt4buildconfiguration.h"
42
#include "qt4projectmanagerconstants.h"
43
44
45

#include <coreplugin/icore.h>
#include <coreplugin/messagemanager.h>
46
#include <coreplugin/progressmanager/progressmanager.h>
47
#include <utils/qtcassert.h>
con's avatar
con committed
48
#include <utils/pathchooser.h>
49
50
#include <projectexplorer/projectexplorerconstants.h>
#include <projectexplorer/project.h>
con's avatar
con committed
51
#include <projectexplorer/buildconfiguration.h>
52

53
54
#include <debugger/debuggermanager.h>

55
56
#include <QtGui/QMessageBox>
#include <QtGui/QMainWindow>
57
#include <QtCore/QCoreApplication>
58

59
using namespace ProjectExplorer;
dt's avatar
dt committed
60
using namespace Qt4ProjectManager;
61
62
using namespace Qt4ProjectManager::Internal;

63
64
65
66
67
68
69
70
71
namespace {
const char * const S60_DEVICE_RC_ID("Qt4ProjectManager.S60DeviceRunConfiguration");
const char * const S60_DEVICE_RC_PREFIX("Qt4ProjectManager.S60DeviceRunConfiguration.");

const char * const PRO_FILE_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.ProFile");
const char * const SERIAL_PORT_NAME_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.SerialPortName");
const char * const COMMUNICATION_TYPE_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.CommunicationType");
const char * const COMMAND_LINE_ARGUMENTS_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.CommandLineArguments");

72
73
74
75
const int    PROGRESS_DEPLOYBASE = 0;
const int    PROGRESS_PACKAGEDEPLOYED = 100;
const int    PROGRESS_PACKAGEINSTALLED = 200;
const int    PROGRESS_MAX = 200;
76

77
enum { debug = 0 };
78

79
// Format information about a file
80
static inline QString msgListFile(const QString &f)
81
82
83
84
{
    QString rc;
    const QFileInfo fi(f);
    QTextStream str(&rc);
85
86
87
88
89
    if (fi.exists()) {
        str << fi.size() << ' ' << fi.lastModified().toString(Qt::ISODate) << ' ' << QDir::toNativeSeparators(fi.absoluteFilePath());
    } else {
        str << "<non-existent> " << QDir::toNativeSeparators(fi.absoluteFilePath());
    }
90
91
    return rc;
}
92

93
94
95
96
97
98
99
100
101
102
103
104
105
106
QString pathFromId(const QString &id)
{
    if (!id.startsWith(QLatin1String(S60_DEVICE_RC_PREFIX)))
        return QString();
    return id.mid(QString::fromLatin1(S60_DEVICE_RC_PREFIX).size());
}

QString pathToId(const QString &path)
{
    return QString::fromLatin1(S60_DEVICE_RC_PREFIX) + path;
}

}

107
// ======== S60DeviceRunConfiguration
108

Tobias Hunger's avatar
Tobias Hunger committed
109
110
S60DeviceRunConfiguration::S60DeviceRunConfiguration(Target *parent, const QString &proFilePath) :
    RunConfiguration(parent,  QLatin1String(S60_DEVICE_RC_ID)),
111
    m_proFilePath(proFilePath),
112
#ifdef Q_OS_WIN
113
    m_serialPortName(QLatin1String("COM5"))
114
#else
115
    m_serialPortName(QLatin1String(SymbianUtils::SymbianDeviceManager::linuxBlueToothDeviceRootC) + QLatin1Char('0'))
116
#endif
117
118
119
120
{
    ctor();
}

Tobias Hunger's avatar
Tobias Hunger committed
121
122
S60DeviceRunConfiguration::S60DeviceRunConfiguration(Target *target, S60DeviceRunConfiguration *source) :
    RunConfiguration(target, source),
123
    m_proFilePath(source->m_proFilePath),
124
    m_serialPortName(source->m_serialPortName)
125
126
127
128
129
{
    ctor();
}

void S60DeviceRunConfiguration::ctor()
130
131
{
    if (!m_proFilePath.isEmpty())
132
        setDisplayName(tr("%1 on Symbian Device").arg(QFileInfo(m_proFilePath).completeBaseName()));
133
    else
134
        setDisplayName(tr("QtS60DeviceRunConfiguration"));
135

Tobias Hunger's avatar
Tobias Hunger committed
136
    connect(qt4Target()->qt4Project(), SIGNAL(proFileUpdated(Qt4ProjectManager::Internal::Qt4ProFileNode*)),
137
            this, SLOT(proFileUpdate(Qt4ProjectManager::Internal::Qt4ProFileNode*)));
dt's avatar
dt committed
138
139
140
141
142
143
}

void S60DeviceRunConfiguration::proFileUpdate(Qt4ProjectManager::Internal::Qt4ProFileNode *pro)
{
    if (m_proFilePath == pro->path())
        emit targetInformationChanged();
144
145
146
147
148
149
}

S60DeviceRunConfiguration::~S60DeviceRunConfiguration()
{
}

Tobias Hunger's avatar
Tobias Hunger committed
150
Qt4Target *S60DeviceRunConfiguration::qt4Target() const
dt's avatar
dt committed
151
{
Tobias Hunger's avatar
Tobias Hunger committed
152
    return static_cast<Qt4Target *>(target());
dt's avatar
dt committed
153
154
}

155
156
157
ProjectExplorer::ToolChain::ToolChainType S60DeviceRunConfiguration::toolChainType(
        ProjectExplorer::BuildConfiguration *configuration) const
{
158
159
    if (Qt4BuildConfiguration *bc = qobject_cast<Qt4BuildConfiguration *>(configuration))
        return bc->toolChainType();
160
161
162
    return ProjectExplorer::ToolChain::INVALID;
}

163
164
ProjectExplorer::ToolChain::ToolChainType S60DeviceRunConfiguration::toolChainType() const
{
Tobias Hunger's avatar
Tobias Hunger committed
165
    if (Qt4BuildConfiguration *bc = qobject_cast<Qt4BuildConfiguration *>(target()->activeBuildConfiguration()))
166
        return bc->toolChainType();
167
    return ProjectExplorer::ToolChain::INVALID;
168
169
}

170
bool S60DeviceRunConfiguration::isEnabled(ProjectExplorer::BuildConfiguration *configuration) const
171
{
172
173
174
175
176
177
178
179
180
181
182
183
    const Qt4BuildConfiguration *qt4bc = static_cast<const Qt4BuildConfiguration *>(configuration);
    switch (qt4bc->toolChainType()) {
    case ToolChain::GCCE:
    case ToolChain::RVCT_ARMV5:
    case ToolChain::RVCT_ARMV6:
    case ToolChain::GCCE_GNUPOC:
    case ToolChain::RVCT_ARMV5_GNUPOC:
        return true;
    default:
        break;
    }
    return false;
184
185
}

186
QWidget *S60DeviceRunConfiguration::createConfigurationWidget()
187
188
189
190
{
    return new S60DeviceRunConfigurationWidget(this);
}

191
QVariantMap S60DeviceRunConfiguration::toMap() const
192
{
193
    QVariantMap map(ProjectExplorer::RunConfiguration::toMap());
194
    const QDir projectDir = QDir(target()->project()->projectDirectory());
195
196
197
198
199
200

    map.insert(QLatin1String(PRO_FILE_KEY), projectDir.relativeFilePath(m_proFilePath));
    map.insert(QLatin1String(SERIAL_PORT_NAME_KEY), m_serialPortName);
    map.insert(QLatin1String(COMMAND_LINE_ARGUMENTS_KEY), m_commandLineArguments);

    return map;
201
202
}

203
bool S60DeviceRunConfiguration::fromMap(const QVariantMap &map)
204
{
205
    const QDir projectDir = QDir(target()->project()->projectDirectory());
206
207
208
209
210
211

    m_proFilePath = projectDir.filePath(map.value(QLatin1String(PRO_FILE_KEY)).toString());
    m_serialPortName = map.value(QLatin1String(SERIAL_PORT_NAME_KEY)).toString().trimmed();
    m_commandLineArguments = map.value(QLatin1String(COMMAND_LINE_ARGUMENTS_KEY)).toStringList();

    return RunConfiguration::fromMap(map);
con's avatar
con committed
212
213
214
215
216
217
218
219
220
}

QString S60DeviceRunConfiguration::serialPortName() const
{
    return m_serialPortName;
}

void S60DeviceRunConfiguration::setSerialPortName(const QString &name)
{
con's avatar
con committed
221
222
223
224
225
    const QString &candidate = name.trimmed();
    if (m_serialPortName == candidate)
        return;
    m_serialPortName = candidate;
    emit serialPortNameChanged();
226
227
}

228
229
QString S60DeviceRunConfiguration::targetName() const
{
dt's avatar
dt committed
230
231
232
233
234
235
236
237
238
239
240
241
242
    TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(m_proFilePath);
    if (!ti.valid)
        return QString();
    return ti.target;
}

static inline QString fixBaseNameTarget(const QString &in)
{
    if (in == QLatin1String("udeb"))
        return QLatin1String("debug");
    if (in == QLatin1String("urel"))
        return QLatin1String("release");
    return in;
243
244
}

con's avatar
con committed
245
QString S60DeviceRunConfiguration::packageFileNameWithTargetInfo() const
246
{
dt's avatar
dt committed
247
248
249
250
    TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(m_proFilePath);
    if (!ti.valid)
        return QString();
    QString baseFileName = ti.workingDir + QLatin1Char('/') + ti.target;
Friedemann Kleint's avatar
Friedemann Kleint committed
251
252
    baseFileName += QLatin1Char('_')
                    + (isDebug() ? QLatin1String("debug") : QLatin1String("release"))
con's avatar
con committed
253
                    + QLatin1Char('-') + symbianPlatform() + QLatin1String(".sis");
dt's avatar
dt committed
254
    return baseFileName;
255
256
}

257
258
QString S60DeviceRunConfiguration::symbianPlatform() const
{
dt's avatar
dt committed
259
260
261
262
263
264
265
266
267
268
    Qt4BuildConfiguration *qt4bc = qt4Target()->qt4Project()->activeTarget()->activeBuildConfiguration();
    switch (qt4bc->toolChainType()) {
    case ToolChain::GCCE:
    case ToolChain::GCCE_GNUPOC:
        return QLatin1String("gcce");
    case ToolChain::RVCT_ARMV5:
        return QLatin1String("armv5");
    default: // including ToolChain::RVCT_ARMV6_GNUPOC:
        return QLatin1String("armv6");
    }
269
270
}

Friedemann Kleint's avatar
Friedemann Kleint committed
271
272
273
274
275
276
bool S60DeviceRunConfiguration::isDebug() const
{
    const Qt4BuildConfiguration *qt4bc = qt4Target()->qt4Project()->activeTarget()->activeBuildConfiguration();
    return (qt4bc->qmakeBuildConfiguration() & QtVersion::DebugBuild);
}

277
278
QString S60DeviceRunConfiguration::symbianTarget() const
{
Friedemann Kleint's avatar
Friedemann Kleint committed
279
    return isDebug() ? QLatin1String("udeb") : QLatin1String("urel");
280
281
282
283
}

QString S60DeviceRunConfiguration::packageTemplateFileName() const
{
dt's avatar
dt committed
284
285
286
287
    TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(m_proFilePath);
    if (!ti.valid)
        return QString();
    return ti.workingDir + QLatin1Char('/') + ti.target + QLatin1String("_template.pkg");
288
289
}

290
291
292
293
294
295
296
297
/* Grep a package file for the '.exe' file. Curently for use on Linux only
 * as the '.pkg'-files on Windows do not contain drive letters, which is not
 * handled here. \code
; Executable and default resource files
"./foo.exe"    - "!:\sys\bin\foo.exe"
\endcode  */

static inline QString executableFromPackageUnix(const QString &packageFileName)
298
{
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
    QFile packageFile(packageFileName);
    if (!packageFile.open(QIODevice::ReadOnly|QIODevice::Text))
        return QString();
    QRegExp pattern(QLatin1String("^\"(.*.exe)\" *- \"!:.*.exe\"$"));
    QTC_ASSERT(pattern.isValid(), return QString());
    foreach(const QString &line, QString::fromLocal8Bit(packageFile.readAll()).split(QLatin1Char('\n')))
        if (pattern.exactMatch(line)) {
            // Expand relative paths by package file paths
            QString rc = pattern.cap(1);
            if (rc.startsWith(QLatin1String("./")))
                rc.remove(0, 2);
            const QFileInfo fi(rc);
            if (fi.isAbsolute())
                return rc;
            return QFileInfo(packageFileName).absolutePath() + QLatin1Char('/') + rc;
        }
    return QString();
}
317

318
319
320
321
322
323
324
325
const QtVersion *S60DeviceRunConfiguration::qtVersion() const
{
    if (const BuildConfiguration *bc = target()->activeBuildConfiguration())
        if (const Qt4BuildConfiguration *qt4bc = qobject_cast<const Qt4BuildConfiguration *>(bc))
            return qt4bc->qtVersion();
    return 0;
}

326
327
328
329
330
331
332
333
334
QString S60DeviceRunConfiguration::localExecutableFileName() const
{
    QString localExecutable;
    switch (toolChainType()) {
    case ToolChain::GCCE_GNUPOC:
    case ToolChain::RVCT_ARMV5_GNUPOC:
        localExecutable = executableFromPackageUnix(packageTemplateFileName());
        break;
    default: {
335
336
337
            const QtVersion *qtv = qtVersion();
            QTC_ASSERT(qtv, return QString());
            const S60Devices::Device device = S60Manager::instance()->deviceForQtVersion(qtv);
338
339
340
341
342
343
344
345
            QTextStream(&localExecutable) << device.epocRoot << "/epoc32/release/"
                    << symbianPlatform() << '/' << symbianTarget() << '/' << targetName()
                    << ".exe";
        }
        break;
    }
    if (debug)
        qDebug() << "Local executable" << localExecutable;
346
    return QDir::toNativeSeparators(localExecutable);
347
348
}

349
350
QString S60DeviceRunConfiguration::signedPackage() const
{
con's avatar
con committed
351
352
353
354
    TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(m_proFilePath);
    if (!ti.valid)
        return QString();
    return ti.workingDir + QLatin1Char('/') + ti.target + QLatin1String(".sis");
355
356
}

357
358
359
360
361
362
363
364
365
366
QStringList S60DeviceRunConfiguration::commandLineArguments() const
{
    return m_commandLineArguments;
}

void S60DeviceRunConfiguration::setCommandLineArguments(const QStringList &args)
{
    m_commandLineArguments = args;
}

367
368
// ======== S60DeviceRunConfigurationFactory

369
370
S60DeviceRunConfigurationFactory::S60DeviceRunConfigurationFactory(QObject *parent) :
    IRunConfigurationFactory(parent)
371
372
373
374
375
376
377
{
}

S60DeviceRunConfigurationFactory::~S60DeviceRunConfigurationFactory()
{
}

Tobias Hunger's avatar
Tobias Hunger committed
378
QStringList S60DeviceRunConfigurationFactory::availableCreationIds(Target *parent) const
379
{
Tobias Hunger's avatar
Tobias Hunger committed
380
381
    Qt4Target *target = qobject_cast<Qt4Target *>(parent);
    if (!target ||
382
        target->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID))
383
384
        return QStringList();

Tobias Hunger's avatar
Tobias Hunger committed
385
    return target->qt4Project()->applicationProFilePathes(QLatin1String(S60_DEVICE_RC_PREFIX));
386
387
}

388
QString S60DeviceRunConfigurationFactory::displayNameForId(const QString &id) const
389
{
390
391
392
    if (!pathFromId(id).isEmpty())
        return tr("%1 on Symbian Device").arg(QFileInfo(pathFromId(id)).completeBaseName());
    return QString();
393
394
}

Tobias Hunger's avatar
Tobias Hunger committed
395
bool S60DeviceRunConfigurationFactory::canCreate(Target *parent, const QString &id) const
396
{
Tobias Hunger's avatar
Tobias Hunger committed
397
398
    Qt4Target * t(qobject_cast<Qt4Target *>(parent));
    if (!t ||
399
        t->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID))
400
        return false;
Tobias Hunger's avatar
Tobias Hunger committed
401
    return t->qt4Project()->hasApplicationProFile(pathFromId(id));
402
403
}

Tobias Hunger's avatar
Tobias Hunger committed
404
RunConfiguration *S60DeviceRunConfigurationFactory::create(Target *parent, const QString &id)
405
{
406
407
408
    if (!canCreate(parent, id))
        return 0;

Tobias Hunger's avatar
Tobias Hunger committed
409
410
    Qt4Target *t(static_cast<Qt4Target *>(parent));
    return new S60DeviceRunConfiguration(t, pathFromId(id));
411
412
}

Tobias Hunger's avatar
Tobias Hunger committed
413
bool S60DeviceRunConfigurationFactory::canRestore(Target *parent, const QVariantMap &map) const
414
{
Tobias Hunger's avatar
Tobias Hunger committed
415
416
    Qt4Target * t(qobject_cast<Qt4Target *>(parent));
    if (!t ||
417
        t->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID))
418
419
420
421
422
        return false;
    QString id(ProjectExplorer::idFromMap(map));
    return id == QLatin1String(S60_DEVICE_RC_ID);
}

Tobias Hunger's avatar
Tobias Hunger committed
423
RunConfiguration *S60DeviceRunConfigurationFactory::restore(Target *parent, const QVariantMap &map)
424
425
426
{
    if (!canRestore(parent, map))
        return 0;
Tobias Hunger's avatar
Tobias Hunger committed
427
428
    Qt4Target *t(static_cast<Qt4Target *>(parent));
    S60DeviceRunConfiguration *rc(new S60DeviceRunConfiguration(t, QString()));
429
430
431
432
433
434
435
    if (rc->fromMap(map))
        return rc;

    delete rc;
    return 0;
}

Tobias Hunger's avatar
Tobias Hunger committed
436
bool S60DeviceRunConfigurationFactory::canClone(Target *parent, RunConfiguration *source) const
437
{
Tobias Hunger's avatar
Tobias Hunger committed
438
    if (!qobject_cast<Qt4Target *>(parent))
439
440
441
442
        return false;
    return source->id() == QLatin1String(S60_DEVICE_RC_ID);
}

Tobias Hunger's avatar
Tobias Hunger committed
443
RunConfiguration *S60DeviceRunConfigurationFactory::clone(Target *parent, RunConfiguration *source)
444
445
446
{
    if (!canClone(parent, source))
        return 0;
Tobias Hunger's avatar
Tobias Hunger committed
447
    Qt4Target *t = static_cast<Qt4Target *>(parent);
448
    S60DeviceRunConfiguration * old(static_cast<S60DeviceRunConfiguration *>(source));
Tobias Hunger's avatar
Tobias Hunger committed
449
    return new S60DeviceRunConfiguration(t, old);
450
451
}

452
// ======== S60DeviceRunControlBase
Friedemann Kleint's avatar
Friedemann Kleint committed
453

454
S60DeviceRunControlBase::S60DeviceRunControlBase(RunConfiguration *runConfiguration) :
455
    RunControl(runConfiguration),
456
    m_toolChain(ProjectExplorer::ToolChain::INVALID),
457
458
    m_releaseDeviceAfterLauncherFinish(false),
    m_handleDeviceRemoval(true),
459
    m_launcher(0)
460
461
462
463
{
    // connect for automatically reporting the "finished deploy" state to the progress manager
    connect(this, SIGNAL(finished()), this, SLOT(reportDeployFinished()));

464
    S60DeviceRunConfiguration *s60runConfig = qobject_cast<S60DeviceRunConfiguration *>(runConfiguration);
465
    const Qt4BuildConfiguration *activeBuildConf = s60runConfig->qt4Target()->activeBuildConfiguration();
dt's avatar
dt committed
466

467
    QTC_ASSERT(s60runConfig, return);
468
    m_toolChain = s60runConfig->toolChainType();
469
    m_serialPortName = s60runConfig->serialPortName();
470
    m_serialPortFriendlyName = SymbianUtils::SymbianDeviceManager::instance()->friendlyNameForPort(m_serialPortName);
471
    m_targetName = s60runConfig->targetName();
con's avatar
con committed
472
    m_packageFileNameWithTarget = s60runConfig->packageFileNameWithTargetInfo();
473
    m_signedPackage = s60runConfig->signedPackage();
474
    m_commandLineArguments = s60runConfig->commandLineArguments();
con's avatar
con committed
475
    m_workingDirectory = QFileInfo(m_signedPackage).absolutePath();
476
    m_qtDir = activeBuildConf->qtVersion()->versionInfo().value("QT_INSTALL_DATA");
477
478
479
    if (const QtVersion *qtv = s60runConfig->qtVersion())
        m_qtBinPath = qtv->versionInfo().value(QLatin1String("QT_INSTALL_BINS"));
    QTC_ASSERT(!m_qtBinPath.isEmpty(), return);
480
481
    m_executableFileName = s60runConfig->localExecutableFileName();
    if (debug)
482
        qDebug() << "S60DeviceRunControlBase::CT" << m_targetName << ProjectExplorer::ToolChain::toolChainName(m_toolChain)
483
                 << m_serialPortName << m_workingDirectory;
484
}
485

486
487
488
489
490
491
492
493
S60DeviceRunControlBase::~S60DeviceRunControlBase()
{
    if (m_launcher) {
        m_launcher->deleteLater();
        m_launcher = 0;
    }
}

494
495
496
497
498
void S60DeviceRunControlBase::setReleaseDeviceAfterLauncherFinish(bool v)
{
    m_releaseDeviceAfterLauncherFinish = v;
}

499
500
501
502
503
504
505
506
507
508
static inline bool ensureDeleteFile(const QString &fileName, QString *errorMessage)
{
    QFile file(fileName);
    if (file.exists() && !file.remove()) {
        *errorMessage = S60DeviceRunControlBase::tr("Unable to remove existing file '%1': %2").arg(fileName, file.errorString());
        return false;
    }
    return true;
}

509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
static inline bool renameFile(const QString &sourceName, const QString &targetName,
                              QString *errorMessage)
{
    if (sourceName == targetName)
        return true;
    if (!ensureDeleteFile(targetName, errorMessage))
        return false;
    QFile source(sourceName);
    if (!source.rename(targetName)) {
        *errorMessage = S60DeviceRunControlBase::tr("Unable to rename file '%1' to '%2': %3")
                        .arg(sourceName, targetName, source.errorString());
        return false;
    }
    return true;
}

525
void S60DeviceRunControlBase::start()
526
{
527
528
529
530
531
532
533
    m_deployProgress = new QFutureInterface<void>;
    Core::ICore::instance()->progressManager()->addTask(m_deployProgress->future(),
                                                        tr("Deploying"),
                                                        QLatin1String("Symbian.Deploy"));
    m_deployProgress->setProgressRange(0, PROGRESS_MAX);
    m_deployProgress->setProgressValue(0);
    m_deployProgress->reportStarted();
534
    emit started();
535
    if (m_serialPortName.isEmpty()) {
536
        m_deployProgress->reportCanceled();
537
        appendMessage(this, tr("There is no device plugged in."), true);
538
539
540
        emit finished();
        return;
    }
541

542
    emit appendMessage(this, tr("Executable file: %1").arg(msgListFile(m_executableFileName)), false);
con's avatar
con committed
543

544
545
546
547
    QString errorMessage;
    QString settingsCategory;
    QString settingsPage;
    if (!checkConfiguration(&errorMessage, &settingsCategory, &settingsPage)) {
548
        m_deployProgress->reportCanceled();
549
        appendMessage(this, errorMessage, true);
550
        emit finished();
551
552
        Core::ICore::instance()->showWarningWithOptions(tr("Debugger for Symbian Platform"),
                                                        errorMessage, QString(),
553
554
555
556
                                                        settingsCategory, settingsPage);
        return;
    }

con's avatar
con committed
557
558
559
560
561
562
563
564
565
566
567
568
    // make sure we have the right name of the sis package
    bool ok = true;
    QFileInfo packageInfo(m_signedPackage);
    {
        // support for 4.6.1 and pre, where make sis creates 'targetname_armX_udeb.sis' instead of 'targetname.sis'
        QFileInfo packageWithTargetInfo(m_packageFileNameWithTarget);
        // does the 4.6.1 version exist?
        if (packageWithTargetInfo.exists() && packageWithTargetInfo.isFile()) {
            // is the 4.6.1 version newer? (to guard against behavior change Qt Creator 1.3 --> 2.0)
            if (!packageInfo.exists()
                    || packageInfo.lastModified() < packageWithTargetInfo.lastModified()) {
                // the 'targetname_armX_udeb.sis' crap exists and is new, rename it
569
570
571
                emit appendMessage(this, tr("Renaming new package '%1' to '%2'")
                                   .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget),
                                        QDir::toNativeSeparators(m_signedPackage)), false);
con's avatar
con committed
572
573
574
                ok = renameFile(m_packageFileNameWithTarget, m_signedPackage, &errorMessage);
            } else {
                // the 'targetname_armX_udeb.sis' crap exists but is old, remove it
575
576
577
                emit appendMessage(this, tr("Removing old package '%1'")
                                   .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget)),
                                   false);
con's avatar
con committed
578
579
580
581
582
583
584
585
                QFile::remove(m_packageFileNameWithTarget);
            }
        }
    }
    if (ok) {
        if (!packageInfo.exists() || !packageInfo.isFile()) {
            errorMessage = tr("Package file not found");
            ok = false;
Friedemann Kleint's avatar
Friedemann Kleint committed
586
        }
con's avatar
con committed
587
    }
588
589
590
    if (ok) {
        startDeployment();
    } else {
591
        m_deployProgress->reportCanceled();
con's avatar
con committed
592
        errorMessage = tr("Failed to find package '%1': %2").arg(m_signedPackage, errorMessage);
593
        appendMessage(this, errorMessage, true);
594
595
596
        stop();
        emit finished();
    }
597
598
}

599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
static inline void stopProcess(QProcess *p)
{
    const int timeOutMS = 200;
    if (p->state() != QProcess::Running)
        return;
    p->terminate();
    if (p->waitForFinished(timeOutMS))
        return;
    p->kill();
}

void S60DeviceRunControlBase::stop()
{
    if (m_launcher)
        m_launcher->terminate();
}

bool S60DeviceRunControlBase::isRunning() const
{
    //TODO !!!
    return false;
}

622
623
void S60DeviceRunControlBase::startDeployment()
{
624
    QString errorMessage;
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
    bool success = false;
    do {
        connect(SymbianUtils::SymbianDeviceManager::instance(), SIGNAL(deviceRemoved(const SymbianUtils::SymbianDevice)),
                this, SLOT(deviceRemoved(SymbianUtils::SymbianDevice)));
        m_launcher = trk::Launcher::acquireFromDeviceManager(m_serialPortName, 0, &errorMessage);
        if (!m_launcher)
            break;

        connect(m_launcher, SIGNAL(finished()), this, SLOT(launcherFinished()));
        connect(m_launcher, SIGNAL(canNotConnect(QString)), this, SLOT(printConnectFailed(QString)));
        connect(m_launcher, SIGNAL(copyingStarted()), this, SLOT(printCopyingNotice()));
        connect(m_launcher, SIGNAL(canNotCreateFile(QString,QString)), this, SLOT(printCreateFileFailed(QString,QString)));
        connect(m_launcher, SIGNAL(canNotWriteFile(QString,QString)), this, SLOT(printWriteFileFailed(QString,QString)));
        connect(m_launcher, SIGNAL(canNotCloseFile(QString,QString)), this, SLOT(printCloseFileFailed(QString,QString)));
        connect(m_launcher, SIGNAL(installingStarted()), this, SLOT(printInstallingNotice()));
        connect(m_launcher, SIGNAL(canNotInstall(QString,QString)), this, SLOT(printInstallFailed(QString,QString)));
        connect(m_launcher, SIGNAL(installingFinished()), this, SLOT(printInstallingFinished()));
        connect(m_launcher, SIGNAL(copyProgress(int)), this, SLOT(printCopyProgress(int)));
        connect(m_launcher, SIGNAL(stateChanged(int)), this, SLOT(slotLauncherStateChanged(int)));
        connect(m_launcher, SIGNAL(processStopped(uint,uint,uint,QString)),
                this, SLOT(processStopped(uint,uint,uint,QString)));

        //TODO sisx destination and file path user definable
        if (!m_commandLineArguments.isEmpty())
            m_launcher->setCommandLineArgs(m_commandLineArguments);
con's avatar
con committed
650
        const QString copyDst = QString::fromLatin1("C:\\Data\\%1").arg(QFileInfo(m_signedPackage).fileName());
651
        const QString runFileName = QString::fromLatin1("C:\\sys\\bin\\%1.exe").arg(m_targetName);
652
        m_launcher->setCopyFileName(m_signedPackage, copyDst);
653
654
        m_launcher->setInstallFileName(copyDst);
        initLauncher(runFileName, m_launcher);
655
        emit appendMessage(this, tr("Package: %1\nDeploying application to '%2'...").arg(msgListFile(m_signedPackage), m_serialPortFriendlyName), false);
656
657
        // Prompt the user to start up the Blue tooth connection
        const trk::PromptStartCommunicationResult src =
658
            S60RunConfigBluetoothStarter::startCommunication(m_launcher->trkDevice(),
659
660
661
662
663
664
665
666
667
668
                                                             0, &errorMessage);
        if (src != trk::PromptStartCommunicationConnected)
            break;
        if (!m_launcher->startServer(&errorMessage)) {
            errorMessage = tr("Could not connect to phone on port '%1': %2\n"
                              "Check if the phone is connected and App TRK is running.").arg(m_serialPortName, errorMessage);
            break;
        }
        success = true;
    } while (false);
669

670
671
    if (!success) {
        if (!errorMessage.isEmpty())
672
            appendMessage(this, errorMessage, true);
673
        stop();
con's avatar
con committed
674
675
        emit finished();
    }
676
677
}

678
void S60DeviceRunControlBase::printCreateFileFailed(const QString &filename, const QString &errorMessage)
con's avatar
con committed
679
{
680
    emit appendMessage(this, tr("Could not create file %1 on device: %2").arg(filename, errorMessage), true);
con's avatar
con committed
681
682
}

683
684
void S60DeviceRunControlBase::printWriteFileFailed(const QString &filename, const QString &errorMessage)
{
685
    emit appendMessage(this, tr("Could not write to file %1 on device: %2").arg(filename, errorMessage), true);
686
687
}

688
689
690
void S60DeviceRunControlBase::printCloseFileFailed(const QString &filename, const QString &errorMessage)
{
    const QString msg = tr("Could not close file %1 on device: %2. It will be closed when App TRK is closed.");
691
    emit appendMessage(this, msg.arg(filename, errorMessage), true);
692
693
}

694
695
void S60DeviceRunControlBase::printConnectFailed(const QString &errorMessage)
{
696
    emit appendMessage(this, tr("Could not connect to App TRK on device: %1. Restarting App TRK might help.").arg(errorMessage), true);
697
698
}

699
void S60DeviceRunControlBase::printCopyingNotice()
700
{
701
    emit appendMessage(this, tr("Copying install file..."), false);
702
703
}

704
void S60DeviceRunControlBase::printCopyProgress(int progress)
con's avatar
con committed
705
{
706
    m_deployProgress->setProgressValue(PROGRESS_DEPLOYBASE + progress);
con's avatar
con committed
707
708
}

709
void S60DeviceRunControlBase::printInstallingNotice()
con's avatar
con committed
710
{
711
    m_deployProgress->setProgressValue(PROGRESS_PACKAGEDEPLOYED);
712
    emit appendMessage(this, tr("Installing application..."), false);
con's avatar
con committed
713
714
}

715
716
717
718
719
720
721
722
void S60DeviceRunControlBase::printInstallingFinished()
{
    m_deployProgress->setProgressValue(PROGRESS_PACKAGEINSTALLED);
    m_deployProgress->reportFinished();
    delete m_deployProgress;
    m_deployProgress = 0;
}

723
724
void S60DeviceRunControlBase::printInstallFailed(const QString &filename, const QString &errorMessage)
{
725
726
727
    QTC_ASSERT(m_deployProgress, ;)
    if (m_deployProgress)
        m_deployProgress->reportCanceled();
728
    emit appendMessage(this, tr("Could not install from package %1 on device: %2").arg(filename, errorMessage), true);
729
730
}

731
732
void S60DeviceRunControlBase::launcherFinished()
{
733
734
735
736
    if (m_releaseDeviceAfterLauncherFinish) {
        m_handleDeviceRemoval = false;
        trk::Launcher::releaseToDeviceManager(m_launcher);
    }
737
738
739
740
741
    m_launcher->deleteLater();
    m_launcher = 0;
    handleLauncherFinished();
}

742
743
744
745
746
747
748
749
750
void S60DeviceRunControlBase::reportDeployFinished()
{
    if (m_deployProgress) {
        m_deployProgress->reportFinished();
        delete m_deployProgress;
        m_deployProgress = 0;
    }
}

751
752
void S60DeviceRunControlBase::processStopped(uint pc, uint pid, uint tid, const QString& reason)
{
753
    emit addToOutputWindow(this, trk::Launcher::msgStopped(pid, tid, pc, reason), false);
754
755
756
    m_launcher->terminate();
}

757
758
759
QMessageBox *S60DeviceRunControlBase::createTrkWaitingMessageBox(const QString &port, QWidget *parent)
{
    const QString title  = QCoreApplication::translate("Qt4ProjectManager::Internal::S60DeviceRunControlBase",
Robert Loehning's avatar
Robert Loehning committed
760
                                                       "Waiting for App TRK");
761
    const QString text = QCoreApplication::translate("Qt4ProjectManager::Internal::S60DeviceRunControlBase",
Robert Loehning's avatar
Robert Loehning committed
762
                                                     "Please start App TRK on %1.").arg(port);
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
    QMessageBox *rc = new QMessageBox(QMessageBox::Information, title, text,
                                      QMessageBox::Cancel, parent);
    return rc;
}

void S60DeviceRunControlBase::slotLauncherStateChanged(int s)
{
    if (s == trk::Launcher::WaitingForTrk) {
        QMessageBox *mb = S60DeviceRunControlBase::createTrkWaitingMessageBox(m_launcher->trkServerName(),
                                                     Core::ICore::instance()->mainWindow());
        connect(m_launcher, SIGNAL(stateChanged(int)), mb, SLOT(close()));
        connect(mb, SIGNAL(finished(int)), this, SLOT(slotWaitingForTrkClosed()));
        mb->open();
    }
}

void S60DeviceRunControlBase::slotWaitingForTrkClosed()
{
    if (m_launcher && m_launcher->state() == trk::Launcher::WaitingForTrk) {
        stop();
783
        appendMessage(this, tr("Canceled."), true);
784
785
786
787
        emit finished();
    }
}

788
789
790
791
792
void S60DeviceRunControlBase::printApplicationOutput(const QString &output)
{
    printApplicationOutput(output, false);
}

793
void S60DeviceRunControlBase::printApplicationOutput(const QString &output, bool onStdErr)
Friedemann Kleint's avatar
Friedemann Kleint committed
794
{
795
    emit addToOutputWindowInline(this, output, onStdErr);
Friedemann Kleint's avatar
Friedemann Kleint committed
796
797
}

798
799
800
void S60DeviceRunControlBase::deviceRemoved(const SymbianUtils::SymbianDevice &d)
{
    if (m_handleDeviceRemoval && d.portName() == m_serialPortName) {
801
        appendMessage(this, tr("The device '%1' has been disconnected").arg(d.friendlyName()), true);
802
803
804
805
        emit finished();
    }
}

806
807
808
809
810
811
812
bool S60DeviceRunControlBase::checkConfiguration(QString * /* errorMessage */,
                                                 QString * /* settingsCategory */,
                                                 QString * /* settingsPage */) const
{
    return true;
}

813
// =============== S60DeviceRunControl
814

815
S60DeviceRunControl::S60DeviceRunControl(ProjectExplorer::RunConfiguration *runConfiguration) :
816
817
818
819
820
821
822
823
824
825
    S60DeviceRunControlBase(runConfiguration)
{
}

void S60DeviceRunControl::initLauncher(const QString &executable, trk::Launcher *launcher)
{
     connect(launcher, SIGNAL(startingApplication()), this, SLOT(printStartingNotice()));
     connect(launcher, SIGNAL(applicationRunning(uint)), this, SLOT(printRunNotice(uint)));
     connect(launcher, SIGNAL(canNotRun(QString)), this, SLOT(printRunFailNotice(QString)));
     connect(launcher, SIGNAL(applicationOutputReceived(QString)), this, SLOT(printApplicationOutput(QString)));
826
     launcher->addStartupActions(trk::Launcher::ActionCopyInstallRun);
827
828
829
830
831
832
     launcher->setFileName(executable);
}

void S60DeviceRunControl::handleLauncherFinished()
{
     emit finished();
833
     emit appendMessage(this, tr("Finished."), false);
834
835
 }

con's avatar
con committed
836
837
void S60DeviceRunControl::printStartingNotice()
{
838
    emit appendMessage(this, tr("Starting application..."), false);
con's avatar
con committed
839
840
841
842
}

void S60DeviceRunControl::printRunNotice(uint pid)
{
843
    emit appendMessage(this, tr("Application running with pid %1.").arg(pid), false);
con's avatar
con committed
844
845
}

846
void S60DeviceRunControl::printRunFailNotice(const QString &errorMessage) {
847
    emit appendMessage(this, tr("Could not start application: %1").arg(errorMessage), true);
848
849
}

850
851
// ======== S60DeviceDebugRunControl

852
S60DeviceDebugRunControl::S60DeviceDebugRunControl(S60DeviceRunConfiguration *runConfiguration) :
853
854
    S60DeviceRunControlBase(runConfiguration),
    m_startParams(new Debugger::DebuggerStartParameters)
855
{
856
    setReleaseDeviceAfterLauncherFinish(true); // Debugger controls device after install
857
    Debugger::DebuggerManager *dm = Debugger::DebuggerManager::instance();
858
859
    S60DeviceRunConfiguration *rc = qobject_cast<S60DeviceRunConfiguration *>(runConfiguration);
    QTC_ASSERT(dm && rc, return);
860
861
862

    connect(dm, SIGNAL(debuggingFinished()),
            this, SLOT(debuggingFinished()), Qt::QueuedConnection);
863
864
    connect(dm, SIGNAL(applicationOutputAvailable(QString, bool)),
            this, SLOT(printApplicationOutput(QString, bool)),
865
866
867
            Qt::QueuedConnection);

    m_startParams->remoteChannel = rc->serialPortName();
868
    m_startParams->processArgs = rc->commandLineArguments();
869
870
    m_startParams->startMode = Debugger::StartInternal;
    m_startParams->toolChainType = rc->toolChainType();
871
872
873
874
875
876

    m_localExecutableFileName = rc->localExecutableFileName();
    const int lastDotPos = m_localExecutableFileName.lastIndexOf(QLatin1Char('.'));
    if (lastDotPos != -1) {
        m_startParams->symbolFileName = m_localExecutableFileName.mid(0, lastDotPos) + QLatin1String(".sym");
    }
877
878
}

879
void S60DeviceDebugRunControl::stop()
con's avatar
con committed
880
{
881
882
883
884
885
886
887
888
889
890
891
    S60DeviceRunControlBase::stop();
    Debugger::DebuggerManager *dm = Debugger::DebuggerManager::instance();
    QTC_ASSERT(dm, return)
    if (dm->state() == Debugger::DebuggerNotReady)
        dm->exitDebugger();
}

S60DeviceDebugRunControl::~S60DeviceDebugRunControl()
{
}

892
void S60DeviceDebugRunControl::initLauncher(const QString &executable, trk::Launcher *launcher)
893
894
895
{
    // No setting an executable on the launcher causes it to deploy only
    m_startParams->executable = executable;
896
897
    // Prefer the '*.sym' file over the '.exe', which should exist at the same
    // location in debug builds
898
899
900

    if (!QFileInfo(m_startParams->symbolFileName).isFile()) {
        m_startParams->symbolFileName.clear();
901
        emit appendMessage(this, tr("Warning: Cannot locate the symbol file belonging to %1.").arg(m_localExecutableFileName), true);
902
    }
903

904
    launcher->addStartupActions(trk::Launcher::ActionCopyInstall);
905
906
    // Avoid close/open sequence in quick succession, which may cause crashs
    launcher->setCloseDevice(false);
907
908
909
910
}

void S60DeviceDebugRunControl::handleLauncherFinished()
{
911
    emit appendMessage(this, tr("Launching debugger..."), false);
912
913
914
915
916
    Debugger::DebuggerManager::instance()->startNewDebugger(m_startParams);
}

void S60DeviceDebugRunControl::debuggingFinished()
{
917
    emit appendMessage(this, tr("Debugging finished."), false);
con's avatar
con committed
918
    emit finished();
919
}
920
921
922
923
924
925
926
927
928
929

bool S60DeviceDebugRunControl::checkConfiguration(QString *errorMessage,
                                                  QString *settingsCategory,
                                                  QString *settingsPage) const
{
    return Debugger::DebuggerManager::instance()->checkDebugConfiguration(m_startParams->toolChainType,
                                                                          errorMessage,
                                                                          settingsCategory,
                                                                          settingsPage);
}