diff --git a/src/plugins/projectexplorer/userfileaccessor.cpp b/src/plugins/projectexplorer/userfileaccessor.cpp
index 60d0be8e2df2182246bf2a4ca62679a915a0a3be..2ac2798c6b818740371461b89c1b2dfefbd797da 100644
--- a/src/plugins/projectexplorer/userfileaccessor.cpp
+++ b/src/plugins/projectexplorer/userfileaccessor.cpp
@@ -167,6 +167,23 @@ public:
     QVariantMap update(Project *project, const QVariantMap &map);
 };
 
+// Version 5 reflects the introduction of new deploy steps for Symbian/Maemo
+class Version5Handler : public UserFileVersionHandler
+{
+public:
+    int userFileVersion() const
+    {
+        return 5;
+    }
+
+    QString displayUserFileVersion() const
+    {
+        return QLatin1String("2.2pre2");
+    }
+
+    QVariantMap update(Project *project, const QVariantMap &map);
+};
+
 //
 // Helper functions:
 //
@@ -238,6 +255,7 @@ UserFileAccessor::UserFileAccessor() :
     addVersionHandler(new Version2Handler);
     addVersionHandler(new Version3Handler);
     addVersionHandler(new Version4Handler);
+    addVersionHandler(new Version5Handler);
 }
 
 UserFileAccessor::~UserFileAccessor()
@@ -1108,3 +1126,65 @@ QVariantMap Version4Handler::update(Project *, const QVariantMap &map)
     }
     return result;
 }
+
+// -------------------------------------------------------------------------
+// Version5Handler
+// -------------------------------------------------------------------------
+
+// Move packaging steps from build steps into deploy steps
+QVariantMap Version5Handler::update(Project *, const QVariantMap &map)
+{
+    QVariantMap result;
+    QMapIterator<QString, QVariant> it(map);
+    while (it.hasNext()) {
+        it.next();
+        const QString &globalKey = it.key();
+        // check for target info
+        if (!globalKey.startsWith(QLatin1String("ProjectExplorer.Project.Target."))) {
+            result.insert(globalKey, it.value());
+            continue;
+        }
+        const QVariantMap &originalTarget = it.value().toMap();
+        // check for symbian and maemo device target
+        if (originalTarget.value(QLatin1String("ProjectExplorer.ProjectConfiguration.Id"))
+            != QLatin1String("Qt4ProjectManager.Target.S60DeviceTarget")
+            && originalTarget.value(QLatin1String("ProjectExplorer.ProjectConfiguration.Id"))
+            != QLatin1String("Qt4ProjectManager.Target.MaemoDeviceTarget")) {
+            result.insert(globalKey, originalTarget);
+            continue;
+        }
+
+        QVariantMap newTarget;
+        QMapIterator<QString, QVariant> targetIt(originalTarget);
+        while (targetIt.hasNext()) {
+            targetIt.next();
+            const QString &targetKey = targetIt.key();
+            if (!targetKey.startsWith(QLatin1String("ProjectExplorer.Target.BuildConfiguration."))) {
+                newTarget.insert(targetKey, targetIt.value());
+                continue;
+            }
+
+            const QVariantMap &originalBc = targetIt.value().toMap();
+            QVariantMap newBc = originalBc;
+            QVariantMap newDeployStep;
+
+            if (originalTarget.value(QLatin1String("ProjectExplorer.ProjectConfiguration.Id"))
+                == QLatin1String("Qt4ProjectManager.Target.S60DeviceTarget")) {
+                newDeployStep.insert(QLatin1String("ProjectExplorer.ProjectConfiguration.Id"),
+                                     QLatin1String("Qt4ProjectManager.S60DeployStep"));
+            } else {
+                newDeployStep.insert(QLatin1String("ProjectExplorer.ProjectConfiguration.Id"),
+                                     QLatin1String("Qt4ProjectManager.MaemoDeployStep"));
+            }
+
+            int deployCount = newBc.value(QLatin1String("ProjectExplorer.BuildConfiguration.DeployStepsCount"), 0).toInt();
+            newBc.insert(QString::fromLatin1("ProjectExplorer.BuildConfiguration.DeployStep.") + QString::number(deployCount),
+                         newDeployStep);
+            newBc.insert(QLatin1String("ProjectExplorer.BuildConfiguration.DeployStepsCount"), deployCount + 1);
+
+            newTarget.insert(targetKey, newBc);
+        }
+        result.insert(globalKey, newTarget);
+    }
+    return result;
+}
diff --git a/src/plugins/qt4projectmanager/qt-maemo/maemodeploystep.cpp b/src/plugins/qt4projectmanager/qt-maemo/maemodeploystep.cpp
index d61b8f89ad611f2669e019b266f88f13b85d5bf3..bfd5895ac4b8c3c051f0ccf3813347b92085caf0 100644
--- a/src/plugins/qt4projectmanager/qt-maemo/maemodeploystep.cpp
+++ b/src/plugins/qt4projectmanager/qt-maemo/maemodeploystep.cpp
@@ -58,7 +58,6 @@ namespace Internal {
 
 const QLatin1String MaemoDeployStep::Id("Qt4ProjectManager.MaemoDeployStep");
 
-
 MaemoDeployStep::MaemoDeployStep(ProjectExplorer::BuildConfiguration *bc)
     : BuildStep(bc, Id), m_deployables(new MaemoDeployables(this))
 {
diff --git a/src/plugins/qt4projectmanager/qt-s60/qt-s60.pri b/src/plugins/qt4projectmanager/qt-s60/qt-s60.pri
index 60b1f5283a46ce80c710c4bd80252316d1697155..2fc08a98b3f2be31639a8a000ef28b9c58c44cc0 100644
--- a/src/plugins/qt4projectmanager/qt-s60/qt-s60.pri
+++ b/src/plugins/qt4projectmanager/qt-s60/qt-s60.pri
@@ -15,7 +15,8 @@ SOURCES += $$PWD/s60devices.cpp \
     $$PWD/abldparser.cpp \
     $$PWD/rvctparser.cpp \
     $$PWD/winscwparser.cpp \
-    qt-s60/s60createpackagestep.cpp
+    qt-s60/s60createpackagestep.cpp \
+    qt-s60/s60deploystep.cpp
 HEADERS += $$PWD/s60devices.h \
     $$PWD/s60devicespreferencepane.h \
     $$PWD/s60manager.h \
@@ -30,6 +31,7 @@ HEADERS += $$PWD/s60devices.h \
     $$PWD/abldparser.h \
     $$PWD/rvctparser.h \
     $$PWD/winscwparser.h \
-    qt-s60/s60createpackagestep.h
+    qt-s60/s60createpackagestep.h \
+    qt-s60/s60deploystep.h
 FORMS += $$PWD/s60devicespreferencepane.ui \
     qt-s60/s60createpackagestep.ui
diff --git a/src/plugins/qt4projectmanager/qt-s60/s60deploystep.cpp b/src/plugins/qt4projectmanager/qt-s60/s60deploystep.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..6c78d6adbcb203ccff01c7b6c47143fef10599da
--- /dev/null
+++ b/src/plugins/qt4projectmanager/qt-s60/s60deploystep.cpp
@@ -0,0 +1,489 @@
+/**************************************************************************
+**
+** This file is part of Qt Creator
+**
+** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
+**
+** 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
+** contact the sales department at http://qt.nokia.com/contact.
+**
+**************************************************************************/
+
+
+#include "S60deployStep.h"
+
+#include "qt4buildconfiguration.h"
+#include "s60devicerunconfiguration.h"
+#include "symbiandevicemanager.h"
+#include "s60runconfigbluetoothstarter.h"
+
+#include <QtGui/QMessageBox>
+#include <QtGui/QMainWindow>
+#include <QtCore/QCoreApplication>
+#include <QtCore/QTimer>
+#include <QDateTime>
+#include <QFile>
+#include <QFileInfo>
+#include <QDir>
+
+#include <coreplugin/icore.h>
+#include <projectexplorer/buildconfiguration.h>
+#include <projectexplorer/target.h>
+#include <projectexplorer/ioutputparser.h>
+#include <qt4projectmanagerconstants.h>
+
+using namespace ProjectExplorer;
+using namespace Qt4ProjectManager::Internal;
+
+namespace {
+    const char * const S60_DEPLOY_STEP_ID = "Qt4ProjectManager.S60DeployStep";
+}
+
+static inline bool ensureDeleteFile(const QString &fileName, QString *errorMessage)
+{
+    QFile file(fileName);
+    if (file.exists() && !file.remove()) {
+        *errorMessage = S60DeployStep::tr("Unable to remove existing file '%1': %2").arg(fileName, file.errorString());
+        return false;
+    }
+    return true;
+}
+
+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 = S60DeployStep::tr("Unable to rename file '%1' to '%2': %3")
+                        .arg(sourceName, targetName, source.errorString());
+        return false;
+    }
+    return true;
+}
+
+// #pragma mark -- S60DeployStep
+
+S60DeployStep::S60DeployStep(ProjectExplorer::BuildConfiguration *bc, const QString &id) :
+        BuildStep(bc, id), m_timer(0), m_launcher(0),
+        m_releaseDeviceAfterLauncherFinish(true), m_handleDeviceRemoval(true),
+        m_eventLoop(0)
+{
+}
+
+S60DeployStep::S60DeployStep(ProjectExplorer::BuildConfiguration *bc,
+                             S60DeployStep *bs):
+        BuildStep(bc, bs), m_timer(0), m_launcher(0),
+        m_releaseDeviceAfterLauncherFinish(bs->m_releaseDeviceAfterLauncherFinish),
+        m_handleDeviceRemoval(bs->m_handleDeviceRemoval),
+        m_eventLoop(0)
+{
+}
+
+S60DeployStep::S60DeployStep(ProjectExplorer::BuildConfiguration *bc):
+        BuildStep(bc, QLatin1String(S60_DEPLOY_STEP_ID)), m_timer(0),
+        m_launcher(0), m_releaseDeviceAfterLauncherFinish(true),
+        m_handleDeviceRemoval(true), m_eventLoop(0)
+{
+}
+
+S60DeployStep::~S60DeployStep()
+{
+    delete m_timer;
+    delete m_launcher;
+    delete m_eventLoop;
+}
+
+
+bool S60DeployStep::init()
+{
+    Qt4BuildConfiguration *bc = static_cast<Qt4BuildConfiguration *>(buildConfiguration());
+    S60DeviceRunConfiguration* runConfiguration = static_cast<S60DeviceRunConfiguration *>(bc->target()->activeRunConfiguration());
+
+    if(!runConfiguration) {
+        return false;
+    }
+    m_serialPortName = runConfiguration->serialPortName();
+    m_serialPortFriendlyName = SymbianUtils::SymbianDeviceManager::instance()->friendlyNameForPort(m_serialPortName);
+    m_packageFileNameWithTarget = runConfiguration->packageFileNameWithTargetInfo();
+    m_signedPackage = runConfiguration->signedPackage();
+
+    setDisplayName(tr("Deploy", "Qt4 DeployStep display name."));
+    QString message;
+    m_launcher = trk::Launcher::acquireFromDeviceManager(m_serialPortName, this, &message);
+    if (!message.isEmpty()) {
+        trk::Launcher::releaseToDeviceManager(m_launcher);
+        delete m_launcher;
+        m_launcher = 0;
+        appendMessage(message, true);
+        return false;
+    }
+    return true;
+}
+
+QVariantMap S60DeployStep::toMap() const
+{
+    return BuildStep::toMap();
+}
+
+bool S60DeployStep::fromMap(const QVariantMap &map)
+{
+    return BuildStep::fromMap(map);
+}
+
+void S60DeployStep::appendMessage(const QString &error, bool isError)
+{
+    emit addOutput(error, isError?ProjectExplorer::BuildStep::ErrorMessageOutput:
+                                  ProjectExplorer::BuildStep::MessageOutput);
+}
+
+bool S60DeployStep::processPackageName(QString &errorMessage)
+{
+    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()) { //TODO change the QtCore
+                // the 'targetname_armX_udeb.sis' crap exists and is new, rename it
+                appendMessage(tr("Renaming new package '%1' to '%2'")
+                              .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget),
+                                   QDir::toNativeSeparators(m_signedPackage)), false);
+                return renameFile(m_packageFileNameWithTarget, m_signedPackage, &errorMessage);
+            } else {
+                // the 'targetname_armX_udeb.sis' crap exists but is old, remove it
+                appendMessage(tr("Removing old package '%1'")
+                              .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget)),
+                              false);
+                ensureDeleteFile(m_packageFileNameWithTarget, &errorMessage);
+            }
+        }
+    }
+    if (!packageInfo.exists() || !packageInfo.isFile()) {
+        errorMessage = tr("Package file not found");
+        return false;
+    }
+    return true;
+}
+
+void S60DeployStep::start()
+{
+    if (m_serialPortName.isEmpty()) {
+        appendMessage(tr("There is no device plugged in."), true);
+        emit finished();
+        return;
+    }
+
+    QString errorMessage;
+
+    // make sure we have the right name of the sis package
+    if (processPackageName(errorMessage)) {
+        startDeployment();
+    } else {
+        errorMessage = tr("Failed to find package '%1': %2").arg(m_signedPackage, errorMessage);
+        appendMessage(errorMessage, true);
+        stop();
+        emit finished();
+    }
+}
+
+void S60DeployStep::stop()
+{
+    if (m_launcher)
+        m_launcher->terminate();
+    emit finished();
+}
+
+void S60DeployStep::setupConnections()
+{
+    connect(SymbianUtils::SymbianDeviceManager::instance(), SIGNAL(deviceRemoved(SymbianUtils::SymbianDevice)),
+            this, SLOT(deviceRemoved(SymbianUtils::SymbianDevice)));
+    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(stateChanged(int)), this, SLOT(slotLauncherStateChanged(int)));
+}
+
+void S60DeployStep::startDeployment()
+{
+    Q_ASSERT(m_launcher);
+
+    QString errorMessage;
+    bool success = false;
+    do {
+        setupConnections();
+        //TODO sisx destination and file path user definable
+        const QString copyDst = QString::fromLatin1("C:\\Data\\%1").arg(QFileInfo(m_signedPackage).fileName());
+
+        m_launcher->setCopyFileName(m_signedPackage, copyDst);
+        m_launcher->setInstallFileName(copyDst);
+        m_launcher->addStartupActions(trk::Launcher::ActionCopyInstall);
+
+        appendMessage(tr("Package: %1\nDeploying application to '%2'...").arg(m_signedPackage, m_serialPortFriendlyName), false);
+
+        // Prompt the user to start up the Blue tooth connection
+        const trk::PromptStartCommunicationResult src =
+                S60RunConfigBluetoothStarter::startCommunication(m_launcher->trkDevice(),
+                                                                 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);
+
+    if (!success) {
+        if (!errorMessage.isEmpty())
+            appendMessage(errorMessage, true);
+        stop();
+        emit finished();
+    }
+}
+
+void S60DeployStep::run(QFutureInterface<bool> &fi)
+{
+    m_futureInterface = &fi;
+    m_deployResult = false;
+    connect(this, SIGNAL(finished()),
+            this, SLOT(launcherFinished()));
+    connect(this, SIGNAL(finishNow()),
+            this, SLOT(launcherFinished()), Qt::DirectConnection);
+
+    start();
+    m_timer = new QTimer();
+    connect(m_timer, SIGNAL(timeout()), this, SLOT(checkForCancel()), Qt::DirectConnection);
+    m_timer->start(500);
+    m_eventLoop = new QEventLoop();
+    m_eventLoop->exec();
+    m_timer->stop();
+    delete m_timer;
+    m_timer = 0;
+
+    delete m_eventLoop;
+    m_eventLoop = 0;
+    fi.reportResult(m_deployResult);
+    m_futureInterface = 0;
+}
+
+void S60DeployStep::setReleaseDeviceAfterLauncherFinish(bool v)
+{
+    m_releaseDeviceAfterLauncherFinish = v;
+}
+
+void S60DeployStep::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 S60DeployStep::slotWaitingForTrkClosed()
+{
+    if (m_launcher && m_launcher->state() == trk::Launcher::WaitingForTrk) {
+        stop();
+        appendMessage(tr("Canceled."), true);
+        emit finished();
+    }
+}
+
+void S60DeployStep::printCreateFileFailed(const QString &filename, const QString &errorMessage)
+{
+    appendMessage(tr("Could not create file %1 on device: %2").arg(filename, errorMessage), true);
+}
+
+void S60DeployStep::printWriteFileFailed(const QString &filename, const QString &errorMessage)
+{
+    appendMessage(tr("Could not write to file %1 on device: %2").arg(filename, errorMessage), true);
+}
+
+void S60DeployStep::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.");
+    appendMessage( msg.arg(filename, errorMessage), true);
+}
+
+void S60DeployStep::printConnectFailed(const QString &errorMessage)
+{
+    appendMessage(tr("Could not connect to App TRK on device: %1. Restarting App TRK might help.").arg(errorMessage), true);
+}
+
+void S60DeployStep::printCopyingNotice()
+{
+    appendMessage(tr("Copying installation file..."), false);
+}
+
+void S60DeployStep::printInstallingNotice()
+{
+    appendMessage(tr("Installing application..."), false);
+}
+
+void S60DeployStep::printInstallingFinished()
+{
+    appendMessage(tr("Installation has finished"), false);
+}
+
+void S60DeployStep::printInstallFailed(const QString &filename, const QString &errorMessage)
+{
+    appendMessage(tr("Could not install from package %1 on device: %2").arg(filename, errorMessage), true);
+}
+
+void S60DeployStep::checkForCancel()
+{
+    if (m_futureInterface->isCanceled() && m_timer->isActive()) {
+        m_timer->stop();
+        stop();
+        appendMessage(tr("Canceled."), true);
+        emit finishNow();
+    }
+}
+
+void S60DeployStep::launcherFinished()
+{
+    if (m_releaseDeviceAfterLauncherFinish && m_launcher) {
+        m_handleDeviceRemoval = false;
+        trk::Launcher::releaseToDeviceManager(m_launcher);
+    }
+    m_deployResult = true;
+    if(m_launcher)
+        m_launcher->deleteLater();
+    m_launcher = 0;
+    if(m_eventLoop)
+        m_eventLoop->exit(0);
+}
+
+void S60DeployStep::deviceRemoved(const SymbianUtils::SymbianDevice &d)
+{
+    if (m_handleDeviceRemoval && d.portName() == m_serialPortName) {
+        appendMessage(tr("The device '%1' has been disconnected").arg(d.friendlyName()), true);
+        emit finished();
+    }
+}
+
+// #pragma mark -- S60DeployStepWidget
+
+BuildStepConfigWidget *S60DeployStep::createConfigWidget()
+{
+    return new S60DeployStepWidget();
+}
+
+void S60DeployStepWidget::init()
+{
+}
+
+QString S60DeployStepWidget::summaryText() const
+{
+    return tr("<b>Deploy SIS Package</b>");
+}
+
+QString S60DeployStepWidget::displayName() const
+{
+    return QString("S60DeployStepWidget::displayName");
+}
+
+// #pragma mark -- S60DeployStepFactory
+
+S60DeployStepFactory::S60DeployStepFactory(QObject *parent) :
+        ProjectExplorer::IBuildStepFactory(parent)
+{
+}
+
+S60DeployStepFactory::~S60DeployStepFactory()
+{
+}
+
+bool S60DeployStepFactory::canCreate(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QString &id) const
+{
+    if (type != ProjectExplorer::BuildStep::Deploy)
+        return false;
+    if (parent->target()->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID))
+        return false;
+    return (id == QLatin1String(S60_DEPLOY_STEP_ID));
+}
+
+ProjectExplorer::BuildStep *S60DeployStepFactory::create(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QString &id)
+{
+    if (!canCreate(parent, type, id))
+        return 0;
+    return new S60DeployStep(parent, id);
+}
+
+bool S60DeployStepFactory::canClone(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, ProjectExplorer::BuildStep *source) const
+{
+    return canCreate(parent, type, source->id());
+}
+
+ProjectExplorer::BuildStep *S60DeployStepFactory::clone(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, ProjectExplorer::BuildStep *source)
+{
+    if (!canClone(parent, type, source))
+        return 0;
+    return new S60DeployStep(parent, static_cast<S60DeployStep *>(source));
+}
+
+bool S60DeployStepFactory::canRestore(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QVariantMap &map) const
+{
+    QString id(ProjectExplorer::idFromMap(map));
+    return canCreate(parent, type, id);
+}
+
+ProjectExplorer::BuildStep *S60DeployStepFactory::restore(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QVariantMap &map)
+{
+    if (!canRestore(parent, type, map))
+        return 0;
+    S60DeployStep *bs = new S60DeployStep(parent);
+    if (bs->fromMap(map))
+        return bs;
+    delete bs;
+    return 0;
+}
+
+QStringList S60DeployStepFactory::availableCreationIds(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type) const
+{
+    if (type != ProjectExplorer::BuildStep::Deploy)
+        return QStringList();
+    if (parent->target()->id() == QLatin1String(Constants::S60_DEVICE_TARGET_ID))
+        return QStringList() << QLatin1String(S60_DEPLOY_STEP_ID);
+    return QStringList();
+}
+
+QString S60DeployStepFactory::displayNameForId(const QString &id) const
+{
+    if (id == QLatin1String(S60_DEPLOY_STEP_ID))
+        return tr("Deploy SIS Package");
+    return QString();
+}
diff --git a/src/plugins/qt4projectmanager/qt-s60/s60deploystep.h b/src/plugins/qt4projectmanager/qt-s60/s60deploystep.h
new file mode 100644
index 0000000000000000000000000000000000000000..579ebdde73048d0b1b37ffe0dd18ee3668736811
--- /dev/null
+++ b/src/plugins/qt4projectmanager/qt-s60/s60deploystep.h
@@ -0,0 +1,165 @@
+/**************************************************************************
+**
+** This file is part of Qt Creator
+**
+** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
+**
+** 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
+** contact the sales department at http://qt.nokia.com/contact.
+**
+**************************************************************************/
+
+
+#ifndef S60DeployStep_H
+#define S60DeployStep_H
+
+#include <projectexplorer/buildstep.h>
+
+#include "launcher.h"
+
+#include <QtCore/QString>
+#include <QtCore/QEventLoop>
+
+namespace SymbianUtils {
+class SymbianDevice;
+}
+
+namespace ProjectExplorer {
+class IOutputParser;
+}
+
+namespace Qt4ProjectManager {
+namespace Internal {
+
+class BuildConfiguration;
+class S60DeviceRunConfiguration;
+
+class S60DeployStepFactory : public ProjectExplorer::IBuildStepFactory
+{
+    Q_OBJECT
+public:
+    explicit S60DeployStepFactory(QObject *parent = 0);
+    ~S60DeployStepFactory();
+
+    // used to show the list of possible additons to a target, returns a list of types
+    QStringList availableCreationIds(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type) const;
+    // used to translate the types to names to display to the user
+    QString displayNameForId(const QString &id) const;
+
+    bool canCreate(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QString &id) const;
+    ProjectExplorer::BuildStep *create(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QString &id);
+    // used to recreate the runConfigurations when restoring settings
+    bool canRestore(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QVariantMap &map) const;
+    ProjectExplorer::BuildStep *restore(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, const QVariantMap &map);
+    bool canClone(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, ProjectExplorer::BuildStep *product) const;
+    ProjectExplorer::BuildStep *clone(ProjectExplorer::BuildConfiguration *parent, ProjectExplorer::BuildStep::Type type, ProjectExplorer::BuildStep *product);
+};
+
+class S60DeployStep : public ProjectExplorer::BuildStep
+{
+    Q_OBJECT
+public:
+    friend class S60DeployStepFactory;
+
+    S60DeployStep(ProjectExplorer::BuildConfiguration *bc, const QString &id);
+    S60DeployStep(ProjectExplorer::BuildConfiguration *bc,
+                  S60DeployStep *bs);
+    explicit S60DeployStep(ProjectExplorer::BuildConfiguration *bc);
+
+    virtual ~S60DeployStep();
+
+    virtual bool init();
+    virtual void run(QFutureInterface<bool> &fi);
+    virtual ProjectExplorer::BuildStepConfigWidget *createConfigWidget();
+
+    void setReleaseDeviceAfterLauncherFinish(bool);
+
+    virtual QVariantMap toMap() const;
+
+protected:
+    virtual bool fromMap(const QVariantMap &map);
+
+private:
+    void start();
+    void stop();
+    void startDeployment();
+    bool processPackageName(QString &errorMessage);
+    void setupConnections();
+    void appendMessage(const QString &error, bool isError);
+
+protected slots:
+    void deviceRemoved(const SymbianUtils::SymbianDevice &);
+
+private slots:
+    void printConnectFailed(const QString &errorMessage);
+    void printCopyingNotice();
+    void printCreateFileFailed(const QString &filename, const QString &errorMessage);
+    void printWriteFileFailed(const QString &filename, const QString &errorMessage);
+    void printCloseFileFailed(const QString &filename, const QString &errorMessage);
+    void printInstallingNotice();
+    void printInstallFailed(const QString &filename, const QString &errorMessage);
+    void printInstallingFinished();
+    void launcherFinished();
+    void slotLauncherStateChanged(int);
+    void slotWaitingForTrkClosed();
+    void checkForCancel();
+
+signals:
+    void finished();
+    void finishNow();
+
+private:
+    QString m_serialPortName;
+    QString m_serialPortFriendlyName;
+    QString m_packageFileNameWithTarget; // Support for 4.6.1
+    QString m_signedPackage;
+
+    QTimer *m_timer;
+
+    bool m_releaseDeviceAfterLauncherFinish;
+    bool m_handleDeviceRemoval;
+
+    QFutureInterface<bool> *m_futureInterface; //not owned
+
+    trk::Launcher *m_launcher;
+
+    QEventLoop *m_eventLoop;
+    bool m_deployResult;
+
+};
+
+class S60DeployStepWidget : public ProjectExplorer::BuildStepConfigWidget
+{
+    Q_OBJECT
+public:
+    S60DeployStepWidget()
+        : ProjectExplorer::BuildStepConfigWidget()
+        {}
+    void init();
+    QString summaryText() const;
+    QString displayName() const;
+
+};
+
+} // Internal
+} // Qt4ProjectManager
+
+#endif // S60DeployStep_H
diff --git a/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.cpp b/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.cpp
index 3070b8a2124f79aac38129d8417b11c8ace43f46..1acabce8ab371802a7909e4ec06c42429ee23620 100644
--- a/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.cpp
+++ b/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.cpp
@@ -509,10 +509,7 @@ S60DeviceRunControlBase::S60DeviceRunControlBase(RunConfiguration *runConfigurat
     m_serialPortName = s60runConfig->serialPortName();
     m_serialPortFriendlyName = SymbianUtils::SymbianDeviceManager::instance()->friendlyNameForPort(m_serialPortName);
     m_targetName = s60runConfig->targetName();
-    m_packageFileNameWithTarget = s60runConfig->packageFileNameWithTargetInfo();
-    m_signedPackage = s60runConfig->signedPackage();
     m_commandLineArguments = s60runConfig->commandLineArguments();
-    m_workingDirectory = QFileInfo(m_signedPackage).absolutePath();
     m_qtDir = activeBuildConf->qtVersion()->versionInfo().value("QT_INSTALL_DATA");
     if (const QtVersion *qtv = s60runConfig->qtVersion())
         m_qtBinPath = qtv->versionInfo().value(QLatin1String("QT_INSTALL_BINS"));
@@ -520,7 +517,7 @@ S60DeviceRunControlBase::S60DeviceRunControlBase(RunConfiguration *runConfigurat
     m_executableFileName = s60runConfig->localExecutableFileName();
     if (debug)
         qDebug() << "S60DeviceRunControlBase::CT" << m_targetName << ProjectExplorer::ToolChain::toolChainName(m_toolChain)
-                 << m_serialPortName << m_workingDirectory;
+                 << m_serialPortName;
 }
 
 S60DeviceRunControlBase::~S60DeviceRunControlBase()
@@ -536,32 +533,6 @@ void S60DeviceRunControlBase::setReleaseDeviceAfterLauncherFinish(bool v)
     m_releaseDeviceAfterLauncherFinish = v;
 }
 
-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;
-}
-
-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;
-}
-
 void S60DeviceRunControlBase::start()
 {
     m_deployProgress = new QFutureInterface<void>;
@@ -594,46 +565,7 @@ void S60DeviceRunControlBase::start()
         return;
     }
 
-    // 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
-                emit appendMessage(this, tr("Renaming new package '%1' to '%2'")
-                                   .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget),
-                                        QDir::toNativeSeparators(m_signedPackage)), false);
-                ok = renameFile(m_packageFileNameWithTarget, m_signedPackage, &errorMessage);
-            } else {
-                // the 'targetname_armX_udeb.sis' crap exists but is old, remove it
-                emit appendMessage(this, tr("Removing old package '%1'")
-                                   .arg(QDir::toNativeSeparators(m_packageFileNameWithTarget)),
-                                   false);
-                QFile::remove(m_packageFileNameWithTarget);
-            }
-        }
-    }
-    if (ok) {
-        if (!packageInfo.exists() || !packageInfo.isFile()) {
-            errorMessage = tr("Package file not found");
-            ok = false;
-        }
-    }
-    if (ok) {
-        startDeployment();
-    } else {
-        m_deployProgress->reportCanceled();
-        errorMessage = tr("Failed to find package '%1': %2").arg(m_signedPackage, errorMessage);
-        appendMessage(this, errorMessage, true);
-        stop();
-        emit finished();
-    }
+    startDeployment();
 }
 
 static inline void stopProcess(QProcess *p)
@@ -672,14 +604,6 @@ void S60DeviceRunControlBase::startDeployment()
 
         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)));
@@ -687,16 +611,11 @@ void S60DeviceRunControlBase::startDeployment()
         //TODO sisx destination and file path user definable
         if (!m_commandLineArguments.isEmpty())
             m_launcher->setCommandLineArgs(m_commandLineArguments);
-        const QString copyDst = QString::fromLatin1("C:\\Data\\%1").arg(QFileInfo(m_signedPackage).fileName());
         const QString runFileName = QString::fromLatin1("C:\\sys\\bin\\%1.exe").arg(m_targetName);
-        m_launcher->setCopyFileName(m_signedPackage, copyDst);
-        m_launcher->setInstallFileName(copyDst);
         initLauncher(runFileName, m_launcher);
-        emit appendMessage(this, tr("Package: %1\nDeploying application to '%2'...").arg(msgListFile(m_signedPackage), m_serialPortFriendlyName), false);
-        // Prompt the user to start up the Blue tooth connection
         const trk::PromptStartCommunicationResult src =
-            S60RunConfigBluetoothStarter::startCommunication(m_launcher->trkDevice(),
-                                                             0, &errorMessage);
+                S60RunConfigBluetoothStarter::startCommunication(m_launcher->trkDevice(),
+                                                                 0, &errorMessage);
         if (src != trk::PromptStartCommunicationConnected)
             break;
         if (!m_launcher->startServer(&errorMessage)) {
@@ -715,59 +634,11 @@ void S60DeviceRunControlBase::startDeployment()
     }
 }
 
-void S60DeviceRunControlBase::printCreateFileFailed(const QString &filename, const QString &errorMessage)
-{
-    emit appendMessage(this, tr("Could not create file %1 on device: %2").arg(filename, errorMessage), true);
-}
-
-void S60DeviceRunControlBase::printWriteFileFailed(const QString &filename, const QString &errorMessage)
-{
-    emit appendMessage(this, tr("Could not write to file %1 on device: %2").arg(filename, errorMessage), true);
-}
-
-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.");
-    emit appendMessage(this, msg.arg(filename, errorMessage), true);
-}
-
 void S60DeviceRunControlBase::printConnectFailed(const QString &errorMessage)
 {
     emit appendMessage(this, tr("Could not connect to App TRK on device: %1. Restarting App TRK might help.").arg(errorMessage), true);
 }
 
-void S60DeviceRunControlBase::printCopyingNotice()
-{
-    emit appendMessage(this, tr("Copying installation file..."), false);
-}
-
-void S60DeviceRunControlBase::printCopyProgress(int progress)
-{
-    m_deployProgress->setProgressValue(PROGRESS_DEPLOYBASE + progress);
-}
-
-void S60DeviceRunControlBase::printInstallingNotice()
-{
-    m_deployProgress->setProgressValue(PROGRESS_PACKAGEDEPLOYED);
-    emit appendMessage(this, tr("Installing application..."), false);
-}
-
-void S60DeviceRunControlBase::printInstallingFinished()
-{
-    m_deployProgress->setProgressValue(PROGRESS_PACKAGEINSTALLED);
-    m_deployProgress->reportFinished();
-    delete m_deployProgress;
-    m_deployProgress = 0;
-}
-
-void S60DeviceRunControlBase::printInstallFailed(const QString &filename, const QString &errorMessage)
-{
-    QTC_ASSERT(m_deployProgress, ;)
-    if (m_deployProgress)
-        m_deployProgress->reportCanceled();
-    emit appendMessage(this, tr("Could not install from package %1 on device: %2").arg(filename, errorMessage), true);
-}
-
 void S60DeviceRunControlBase::launcherFinished()
 {
     if (m_releaseDeviceAfterLauncherFinish) {
@@ -866,7 +737,7 @@ void S60DeviceRunControl::initLauncher(const QString &executable, trk::Launcher
      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)));
-     launcher->addStartupActions(trk::Launcher::ActionCopyInstallRun);
+     launcher->addStartupActions(trk::Launcher::ActionRun);
      launcher->setFileName(executable);
 }
 
@@ -936,10 +807,10 @@ void S60DeviceDebugRunControl::initLauncher(const QString &executable, trk::Laun
         m_startParams->symbolFileName.clear();
         emit appendMessage(this, tr("Warning: Cannot locate the symbol file belonging to %1.").arg(m_localExecutableFileName), true);
     }
-
-    launcher->addStartupActions(trk::Launcher::ActionCopyInstall);
     // Avoid close/open sequence in quick succession, which may cause crashs
     launcher->setCloseDevice(false);
+    // The S60DeviceDebugRunControl does not deploy anything anymore
+    emit finished();
 }
 
 void S60DeviceDebugRunControl::handleLauncherFinished()
diff --git a/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.h b/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.h
index 7776692d256a6f20f4e296a0c28f6685d588d75b..f5537a9ca431120c22c71a04f8fa509e0bd24273 100644
--- a/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.h
+++ b/src/plugins/qt4projectmanager/qt-s60/s60devicerunconfiguration.h
@@ -175,14 +175,6 @@ protected slots:
 private slots:
     void processStopped(uint pc, uint pid, uint tid, const QString& reason);
     void printConnectFailed(const QString &errorMessage);
-    void printCopyingNotice();
-    void printCreateFileFailed(const QString &filename, const QString &errorMessage);
-    void printWriteFileFailed(const QString &filename, const QString &errorMessage);
-    void printCloseFileFailed(const QString &filename, const QString &errorMessage);
-    void printCopyProgress(int progress);
-    void printInstallingNotice();
-    void printInstallFailed(const QString &filename, const QString &errorMessage);
-    void printInstallingFinished();
     void launcherFinished();
     void slotLauncherStateChanged(int);
     void slotWaitingForTrkClosed();
@@ -199,8 +191,6 @@ private:
     QString m_workingDirectory;
     QString m_executableFileName;
     QString m_qtDir;
-    QString m_packageFileNameWithTarget; // Support for 4.6.1
-    QString m_signedPackage;
     QString m_qtBinPath;
     bool m_releaseDeviceAfterLauncherFinish;
     bool m_handleDeviceRemoval;
diff --git a/src/plugins/qt4projectmanager/qt-s60/s60manager.cpp b/src/plugins/qt4projectmanager/qt-s60/s60manager.cpp
index 77f8c1a5884cc753fc5bae3910dc6917a1ea8f13..fc97f98baf33b68d2bb95cd83d453a203cf6d098 100644
--- a/src/plugins/qt4projectmanager/qt-s60/s60manager.cpp
+++ b/src/plugins/qt4projectmanager/qt-s60/s60manager.cpp
@@ -38,6 +38,7 @@
 #include "s60emulatorrunconfiguration.h"
 #include "s60devicerunconfiguration.h"
 #include "s60createpackagestep.h"
+#include "s60deploystep.h"
 
 #include <coreplugin/icore.h>
 #include <extensionsystem/pluginmanager.h>
@@ -120,6 +121,7 @@ S60Manager::S60Manager(QObject *parent)
                                                 (QLatin1String(ProjectExplorer::Constants::RUNMODE),
                                                  tr("Run on Device"), parent));
     addAutoReleasedObject(new S60CreatePackageStepFactory);
+    addAutoReleasedObject(new S60DeployStepFactory);
 
     if (Debugger::DebuggerPlugin::instance())
         addAutoReleasedObject(new RunControlFactory<S60DeviceDebugRunControl,
diff --git a/src/plugins/qt4projectmanager/qt4target.cpp b/src/plugins/qt4projectmanager/qt4target.cpp
index 4f262dbea038c220bbd176b07250537a665d9184..6387799691ef9eed6e489fa1db5637ce39b4d25b 100644
--- a/src/plugins/qt4projectmanager/qt4target.cpp
+++ b/src/plugins/qt4projectmanager/qt4target.cpp
@@ -41,6 +41,7 @@
 #include "qt-s60/s60devicerunconfiguration.h"
 #include "qt-s60/s60emulatorrunconfiguration.h"
 #include "qt-s60/s60createpackagestep.h"
+#include "qt-s60/s60deploystep.h"
 
 #include <projectexplorer/customexecutablerunconfiguration.h>
 #include <projectexplorer/toolchain.h>
@@ -285,6 +286,8 @@ Qt4BuildConfiguration *Qt4Target::addQt4BuildConfiguration(QString displayName,
     if (id() == Constants::S60_DEVICE_TARGET_ID) {
         S60CreatePackageStep *packageStep = new S60CreatePackageStep(bc);
         bc->insertStep(ProjectExplorer::BuildStep::Deploy, 2, packageStep);
+        S60DeployStep *deployStep = new S60DeployStep(bc);
+        bc->insertStep(ProjectExplorer::BuildStep::Deploy, 3, deployStep);
     } else if (id() == Constants::MAEMO_DEVICE_TARGET_ID) {
         bc->insertStep(ProjectExplorer::BuildStep::Deploy, 2,
             new MaemoPackageCreationStep(bc));