diff --git a/src/plugins/coreplugin/documentmanager.cpp b/src/plugins/coreplugin/documentmanager.cpp
index 5607341ec24413f1593f716528a453fc9a0f7546..f2c70385bc17fec756d6983b648788addf0e24e1 100644
--- a/src/plugins/coreplugin/documentmanager.cpp
+++ b/src/plugins/coreplugin/documentmanager.cpp
@@ -104,11 +104,11 @@ namespace Core {
 
 static void readSettings();
 
-static QList<IDocument *> saveModifiedFilesHelper(const QList<IDocument *> &documents,
-                               bool *cancelled, bool silently,
-                               const QString &message,
-                               const QString &alwaysSaveMessage = QString(),
-                               bool *alwaysSave = 0);
+static bool saveModifiedFilesHelper(const QList<IDocument *> &documents,
+                                    const QString &message,
+                                    bool *cancelled, bool silently,
+                                    const QString &alwaysSaveMessage,
+                                    bool *alwaysSave, QList<IDocument *> *failedToSave);
 
 namespace Internal {
 
@@ -549,47 +549,10 @@ void DocumentManager::unexpectFileChange(const QString &fileName)
         updateExpectedState(fixedResolvedName);
 }
 
-
-/*!
-    Tries to save the files listed in \a documents. The \a cancelled argument is set to true
-    if the user cancelled the dialog. Returns the files that could not be saved. If the files
-    listed in documents have no write permissions, an additional dialog will be
-    displayed to
-    query the user for these permissions.
-*/
-QList<IDocument *> DocumentManager::saveModifiedDocumentsSilently(const QList<IDocument *> &documents, bool *cancelled)
-{
-    return saveModifiedFilesHelper(documents, cancelled, true, QString());
-}
-
-/*!
-    Asks the user whether to save the files listed in \a documents. Opens a
-    dialog that displays the \a message, and additional text to ask the users
-    if they want to enable automatic saving
-    of modified files (in this context).
-
-    The \a cancelled argument is set to true if the user cancels the dialog.
-    \a alwaysSave is set to match the selection of the user if files should
-    always automatically be saved. If the files listed in documents have no write
-    permissions, an additional dialog will be displayed to query the user for
-    these permissions.
-
-    Returns the files that have not been saved.
-*/
-QList<IDocument *> DocumentManager::saveModifiedDocuments(const QList<IDocument *> &documents,
-                                              bool *cancelled, const QString &message,
-                                              const QString &alwaysSaveMessage,
-                                              bool *alwaysSave)
-{
-    return saveModifiedFilesHelper(documents, cancelled, false, message, alwaysSaveMessage, alwaysSave);
-}
-
-static QList<IDocument *> saveModifiedFilesHelper(const QList<IDocument *> &documents,
-                                              bool *cancelled,
-                                              bool silently,
-                                              const QString &message,
-                                              const QString &alwaysSaveMessage,
-                                              bool *alwaysSave)
+static bool saveModifiedFilesHelper(const QList<IDocument *> &documents,
+                                    const QString &message, bool *cancelled, bool silently,
+                                    const QString &alwaysSaveMessage, bool *alwaysSave,
+                                    QList<IDocument *> *failedToSave)
 {
     if (cancelled)
         (*cancelled) = false;
@@ -626,9 +589,10 @@ static QList<IDocument *> saveModifiedFilesHelper(const QList<IDocument *> &docu
                 if (cancelled)
                     (*cancelled) = true;
                 if (alwaysSave)
-                    *alwaysSave = dia.alwaysSaveChecked();
-                notSaved = modifiedDocuments;
-                return notSaved;
+                    (*alwaysSave) = dia.alwaysSaveChecked();
+                if (failedToSave)
+                    (*failedToSave) = modifiedDocuments;
+                return false;
             }
             if (alwaysSave)
                 *alwaysSave = dia.alwaysSaveChecked();
@@ -648,8 +612,9 @@ static QList<IDocument *> saveModifiedFilesHelper(const QList<IDocument *> &docu
             if (roDialog.exec() == Core::Internal::ReadOnlyFilesDialog::RO_Cancel) {
                 if (cancelled)
                     (*cancelled) = true;
-                notSaved = modifiedDocuments;
-                return notSaved;
+                if (failedToSave)
+                    (*failedToSave) = modifiedDocuments;
+                return false;
             }
         }
         foreach (IDocument *document, documentsToSave) {
@@ -660,7 +625,9 @@ static QList<IDocument *> saveModifiedFilesHelper(const QList<IDocument *> &docu
             }
         }
     }
-    return notSaved;
+    if (failedToSave)
+        (*failedToSave) = notSaved;
+    return notSaved.isEmpty();
 }
 
 bool DocumentManager::saveDocument(IDocument *document, const QString &fileName, bool *isReadOnly)
@@ -780,6 +747,118 @@ QString DocumentManager::getSaveAsFileName(const IDocument *document, const QStr
     return absoluteFilePath;
 }
 
+/*!
+    Silently saves all documents and will return true if all modified documents were saved
+    successfully.
+
+    This method will try to avoid showing dialogs to the user, but can do so anyway (e.g. if
+    a file is not writeable).
+
+    \a Canceled will be set if the user canceled any of the dialogs that he interacted with.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveAllModifiedDocumentsSilently(bool *canceled,
+                                                       QList<IDocument *> *failedToClose)
+{
+    return saveModifiedDocumentsSilently(modifiedDocuments(), canceled, failedToClose);
+}
+
+/*!
+    Silently saves \a documents and will return true if all of them were saved successfully.
+
+    This method will try to avoid showing dialogs to the user, but can do so anyway (e.g. if
+    a file is not writeable).
+
+    \a Canceled will be set if the user canceled any of the dialogs that he interacted with.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveModifiedDocumentsSilently(const QList<IDocument *> &documents, bool *canceled,
+                                                    QList<IDocument *> *failedToClose)
+{
+    return saveModifiedFilesHelper(documents, QString(), canceled, true, QString(), 0, failedToClose);
+}
+
+/*!
+    Silently saves a \a document and will return true if it was saved successfully.
+
+    This method will try to avoid showing dialogs to the user, but can do so anyway (e.g. if
+    a file is not writeable).
+
+    \a Canceled will be set if the user canceled any of the dialogs that he interacted with.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveModifiedDocumentSilently(IDocument *document, bool *canceled,
+                                                   QList<IDocument *> *failedToClose)
+{
+    return saveModifiedDocumentsSilently(QList<IDocument *>() << document, canceled, failedToClose);
+}
+
+/*!
+    Presents a dialog with all modified documents to the user and will ask him which of these
+    should be saved.
+
+    This method may show additional dialogs to the user, e.g. if a file is not writeable).
+
+    The dialog text can be set using \a message. \a Canceled will be set if the user canceled any
+    of the dialogs that he interacted with (the method will also return false in this case).
+    The \a alwaysSaveMessage will show an additional checkbox asking in the dialog. The state of
+    this checkbox will be written into \a alwaysSave if set.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveAllModifiedDocuments(const QString &message, bool *canceled,
+                                               const QString &alwaysSaveMessage, bool *alwaysSave,
+                                               QList<IDocument *> *failedToClose)
+{
+    return saveModifiedDocuments(modifiedDocuments(), message, canceled,
+                                 alwaysSaveMessage, alwaysSave, failedToClose);
+}
+
+/*!
+    Presents a dialog with \a documents to the user and will ask him which of these should be saved.
+
+    This method may show additional dialogs to the user, e.g. if a file is not writeable).
+
+    The dialog text can be set using \a message. \a Canceled will be set if the user canceled any
+    of the dialogs that he interacted with (the method will also return false in this case).
+    The \a alwaysSaveMessage will show an additional checkbox asking in the dialog. The state of
+    this checkbox will be written into \a alwaysSave if set.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveModifiedDocuments(const QList<IDocument *> &documents,
+                                            const QString &message, bool *canceled,
+                                            const QString &alwaysSaveMessage, bool *alwaysSave,
+                                            QList<IDocument *> *failedToClose)
+{
+    return saveModifiedFilesHelper(documents, message, canceled, false,
+                                   alwaysSaveMessage, alwaysSave, failedToClose);
+}
+
+/*!
+    Presents a dialog with the one \a document to the user and will ask him whether he wants it
+    saved.
+
+    This method may show additional dialogs to the user, e.g. if the file is not writeable).
+
+    The dialog text can be set using \a message. \a Canceled will be set if the user canceled any
+    of the dialogs that he interacted with (the method will also return false in this case).
+    The \a alwaysSaveMessage will show an additional checkbox asking in the dialog. The state of
+    this checkbox will be written into \a alwaysSave if set.
+    \a FailedToClose will contain a list of documents that could not be saved if passed into the
+    method.
+*/
+bool DocumentManager::saveModifiedDocument(IDocument *document, const QString &message, bool *canceled,
+                                           const QString &alwaysSaveMessage, bool *alwaysSave,
+                                           QList<IDocument *> *failedToClose)
+{
+    return saveModifiedDocuments(QList<IDocument *>() << document, message, canceled,
+                                 alwaysSaveMessage, alwaysSave, failedToClose);
+}
+
 /*!
     Asks the user for a set of file names to be opened. The \a filters
     and \a selectedFilter arguments are interpreted like in
diff --git a/src/plugins/coreplugin/documentmanager.h b/src/plugins/coreplugin/documentmanager.h
index 2bd7439da0b65679a08ff289f5093ba3ca58bab2..1144a15d294586932714fda2ca3da62bf1776e44 100644
--- a/src/plugins/coreplugin/documentmanager.h
+++ b/src/plugins/coreplugin/documentmanager.h
@@ -99,12 +99,27 @@ public:
     static QString getSaveAsFileName(const IDocument *document, const QString &filter = QString(),
                               QString *selectedFilter = 0);
 
-    static QList<IDocument *> saveModifiedDocumentsSilently(const QList<IDocument *> &documents, bool *cancelled = 0);
-    static QList<IDocument *> saveModifiedDocuments(const QList<IDocument *> &documents,
-                                     bool *cancelled = 0,
-                                     const QString &message = QString(),
+    static bool saveAllModifiedDocumentsSilently(bool *canceled = 0,
+                                                 QList<IDocument *> *failedToClose = 0);
+    static bool saveModifiedDocumentsSilently(const QList<IDocument *> &documents, bool *canceled = 0,
+                                              QList<IDocument *> *failedToClose = 0);
+    static bool saveModifiedDocumentSilently(IDocument *document, bool *canceled = 0,
+                                             QList<IDocument *> *failedToClose = 0);
+
+    static bool saveAllModifiedDocuments(const QString &message = QString(), bool *canceled = 0,
+                                         const QString &alwaysSaveMessage = QString(),
+                                         bool *alwaysSave = 0,
+                                         QList<IDocument *> *failedToClose = 0);
+    static bool saveModifiedDocuments(const QList<IDocument *> &documents,
+                                      const QString &message = QString(), bool *canceled = 0,
+                                      const QString &alwaysSaveMessage = QString(),
+                                      bool *alwaysSave = 0,
+                                      QList<IDocument *> *failedToClose = 0);
+    static bool saveModifiedDocument(IDocument *document,
+                                     const QString &message = QString(), bool *canceled = 0,
                                      const QString &alwaysSaveMessage = QString(),
-                                     bool *alwaysSave = 0);
+                                     bool *alwaysSave = 0,
+                                     QList<IDocument *> *failedToClose = 0);
 
     static QString fileDialogLastVisitedDirectory();
     static void setFileDialogLastVisitedDirectory(const QString &);
diff --git a/src/plugins/coreplugin/editormanager/editormanager.cpp b/src/plugins/coreplugin/editormanager/editormanager.cpp
index 4c2e9c9bbd887e071431fb51c25c47b54cb14db5..4f52d0a3e47bbf5c4287cf2133048b7a204e4ef5 100644
--- a/src/plugins/coreplugin/editormanager/editormanager.cpp
+++ b/src/plugins/coreplugin/editormanager/editormanager.cpp
@@ -1084,7 +1084,9 @@ bool EditorManager::closeEditors(const QList<IEditor*> &editorsToClose, bool ask
     //ask whether to save modified files
     if (askAboutModifiedEditors) {
         bool cancelled = false;
-        QList<IDocument*> list = DocumentManager::saveModifiedDocuments(acceptedDocuments.toList(), &cancelled);
+        QList<IDocument *> list;
+        DocumentManager::saveModifiedDocuments(acceptedDocuments.toList(), QString(), &cancelled,
+                                               QString(), 0, &list);
         if (cancelled)
             return false;
         if (!list.isEmpty()) {
diff --git a/src/plugins/coreplugin/externaltool.cpp b/src/plugins/coreplugin/externaltool.cpp
index a0d67d5d8f8eed8969f87c8b91fed21c1606394c..898b8e87b426b69658936c58e24c342d34e934c6 100644
--- a/src/plugins/coreplugin/externaltool.cpp
+++ b/src/plugins/coreplugin/externaltool.cpp
@@ -593,9 +593,7 @@ void ExternalToolRunner::run()
     if (m_tool->modifiesCurrentDocument()) {
         if (IDocument *document = EditorManager::currentDocument()) {
             m_expectedFileName = document->filePath();
-            bool cancelled = false;
-            DocumentManager::saveModifiedDocuments(QList<IDocument *>() << document, &cancelled);
-            if (cancelled) {
+            if (!DocumentManager::saveModifiedDocument(document)) {
                 deleteLater();
                 return;
             }
diff --git a/src/plugins/coreplugin/mainwindow.cpp b/src/plugins/coreplugin/mainwindow.cpp
index 5d31975ed5ecca3218ddd3b33777bd8bdd2a9448..42d657390fac2b16f24f1bc6f6fbc90d474c9ef7 100644
--- a/src/plugins/coreplugin/mainwindow.cpp
+++ b/src/plugins/coreplugin/mainwindow.cpp
@@ -360,9 +360,7 @@ void MainWindow::closeEvent(QCloseEvent *event)
     ICore::saveSettings();
 
     // Save opened files
-    bool cancelled;
-    QList<IDocument*> notSaved = DocumentManager::saveModifiedDocuments(DocumentManager::modifiedDocuments(), &cancelled);
-    if (cancelled || !notSaved.isEmpty()) {
+    if (!DocumentManager::saveAllModifiedDocuments()) {
         event->ignore();
         return;
     }
@@ -925,7 +923,7 @@ bool MainWindow::showOptionsDialog(Id category, Id page, QWidget *parent)
 
 void MainWindow::saveAll()
 {
-    DocumentManager::saveModifiedDocumentsSilently(DocumentManager::modifiedDocuments());
+    DocumentManager::saveAllModifiedDocuments();
 }
 
 void MainWindow::exit()
diff --git a/src/plugins/fakevim/fakevimplugin.cpp b/src/plugins/fakevim/fakevimplugin.cpp
index a7ed73ab840f8c5b8a6ad892ed0504beeb19a921..1cf0e8f01971bb110793a17d63c0528c3d981701 100644
--- a/src/plugins/fakevim/fakevimplugin.cpp
+++ b/src/plugins/fakevim/fakevimplugin.cpp
@@ -1945,8 +1945,9 @@ void FakeVimPluginPrivate::handleExCommand(bool *handled, const ExCommand &cmd)
     } else if (cmd.matches(_("wa"), _("wall"))) {
         // :w[all]
         QList<IDocument *> toSave = DocumentManager::modifiedDocuments();
-        QList<IDocument *> failed = DocumentManager::saveModifiedDocumentsSilently(toSave);
-        if (failed.isEmpty())
+        QList<IDocument *> failed;
+        bool success = DocumentManager::saveModifiedDocuments(toSave, QString(), 0, QString(), 0, &failed);
+        if (!success)
             handler->showMessage(MessageInfo, tr("Saving succeeded"));
         else
             handler->showMessage(MessageError, tr("%n files not saved", 0, failed.size()));
diff --git a/src/plugins/git/gitplugin.cpp b/src/plugins/git/gitplugin.cpp
index 62044430ecae18f86d6d1fbbdb65607bb54b159a..b2eda533d8136972852a85912bed0387e25f3496 100644
--- a/src/plugins/git/gitplugin.cpp
+++ b/src/plugins/git/gitplugin.cpp
@@ -719,14 +719,6 @@ void GitPlugin::submitEditorMerge(const QStringList &unmerged)
     m_gitClient->merge(m_submitRepository, unmerged);
 }
 
-static bool ensureAllDocumentsSaved()
-{
-    bool cancelled;
-    Core::DocumentManager::saveModifiedDocuments(Core::DocumentManager::modifiedDocuments(),
-                                                 &cancelled);
-    return !cancelled;
-}
-
 void GitPlugin::diffCurrentFile()
 {
     const VcsBase::VcsBasePluginState state = currentState();
@@ -788,7 +780,7 @@ void GitPlugin::reflogRepository()
 
 void GitPlugin::undoFileChanges(bool revertStaging)
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     const VcsBase::VcsBasePluginState state = currentState();
     QTC_ASSERT(state.hasFile(), return);
@@ -798,7 +790,7 @@ void GitPlugin::undoFileChanges(bool revertStaging)
 
 void GitPlugin::undoUnstagedFileChanges()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     undoFileChanges(false);
 }
@@ -832,7 +824,7 @@ protected:
 
 void GitPlugin::resetRepository()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     const VcsBase::VcsBasePluginState state = currentState();
     QTC_ASSERT(state.hasTopLevel(), return);
@@ -847,7 +839,7 @@ void GitPlugin::resetRepository()
 
 void GitPlugin::startRebase()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     const VcsBase::VcsBasePluginState state = currentState();
     QTC_ASSERT(state.hasTopLevel(), return);
@@ -889,7 +881,7 @@ void GitPlugin::startChangeRelatedAction()
         return;
     }
 
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
 
     switch (dialog.command()) {
@@ -1150,7 +1142,7 @@ void GitPlugin::fetch()
 
 void GitPlugin::pull()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     const VcsBase::VcsBasePluginState state = currentState();
     QTC_ASSERT(state.hasTopLevel(), return);
@@ -1187,7 +1179,7 @@ void GitPlugin::startMergeTool()
 
 void GitPlugin::continueOrAbortCommand()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     const VcsBase::VcsBasePluginState state = currentState();
     QTC_ASSERT(state.hasTopLevel(), return);
@@ -1286,11 +1278,7 @@ void GitPlugin::updateSubmodules()
 static bool ensureFileSaved(const QString &fileName)
 {
     Core::IDocument *document = Core::EditorManager::documentModel()->documentForFilePath(fileName);
-    if (!document || !document->isModified())
-        return true;
-    bool canceled;
-    Core::DocumentManager::saveModifiedDocuments(QList<Core::IDocument *>() << document, &canceled);
-    return !canceled;
+    return Core::DocumentManager::saveModifiedDocument(document);
 }
 
 void GitPlugin::applyCurrentFilePatch()
@@ -1342,7 +1330,7 @@ void GitPlugin::applyPatch(const QString &workingDirectory, QString file)
 
 void GitPlugin::stash()
 {
-    if (!ensureAllDocumentsSaved())
+    if (!Core::DocumentManager::saveAllModifiedDocuments())
         return;
     // Simple stash without prompt, reset repo.
     const VcsBase::VcsBasePluginState state = currentState();
diff --git a/src/plugins/projectexplorer/projectexplorer.cpp b/src/plugins/projectexplorer/projectexplorer.cpp
index 7fe089e91f6cc50906537014e678ffbe7f55d554..9c48e3eca00de80c809f6f235a09ac47096a278d 100644
--- a/src/plugins/projectexplorer/projectexplorer.cpp
+++ b/src/plugins/projectexplorer/projectexplorer.cpp
@@ -1075,15 +1075,7 @@ void ProjectExplorerPlugin::unloadProject()
     if (!document || document->filePath().isEmpty()) //nothing to save?
         return;
 
-    QList<IDocument*> documentsToSave;
-    documentsToSave << document;
-    bool success = false;
-    if (document->isFileReadOnly())
-        success = DocumentManager::saveModifiedDocuments(documentsToSave).isEmpty();
-    else
-        success = DocumentManager::saveModifiedDocumentsSilently(documentsToSave).isEmpty();
-
-    if (!success)
+    if (!DocumentManager::saveModifiedDocumentSilently(document))
         return;
 
     addToRecentProjects(document->filePath(), d->m_currentProject->displayName());
@@ -1967,13 +1959,13 @@ bool ProjectExplorerPlugin::saveModifiedFiles()
         } else {
             bool cancelled = false;
             bool alwaysSave = false;
-            DocumentManager::saveModifiedDocuments(documentsToSave, &cancelled, QString(),
-                                  tr("Always save files before build"), &alwaysSave);
-
-            if (cancelled)
-                return false;
-            if (alwaysSave)
-                d->m_projectExplorerSettings.saveBeforeBuild = true;
+            if (!DocumentManager::saveModifiedDocuments(documentsToSave, QString(), &cancelled,
+                                                        tr("Always save files before build"), &alwaysSave)) {
+                if (cancelled)
+                    return false;
+                if (alwaysSave)
+                    d->m_projectExplorerSettings.saveBeforeBuild = true;
+            }
         }
     }
     return true;
diff --git a/src/plugins/qmakeprojectmanager/qmakenodes.cpp b/src/plugins/qmakeprojectmanager/qmakenodes.cpp
index 32ad3e5adb28ded1fa55d313e25e5a4481468ab9..59a0945663cc8302e9c789a89862644554bdbb1d 100644
--- a/src/plugins/qmakeprojectmanager/qmakenodes.cpp
+++ b/src/plugins/qmakeprojectmanager/qmakenodes.cpp
@@ -1065,11 +1065,11 @@ bool QmakePriFileNode::saveModifiedEditors()
     if (!document || !document->isModified())
         return true;
 
-    bool cancelled;
-    Core::DocumentManager::saveModifiedDocuments(QList<Core::IDocument *>() << document, &cancelled,
-                                                 tr("There are unsaved changes for project file %1.").arg(m_projectFilePath));
-    if (cancelled)
+    if (!Core::DocumentManager::saveDocument(document,
+                                             tr("There are unsaved changes for project file %1.")
+                                             .arg(m_projectFilePath))) {
         return false;
+    }
     // force instant reload of ourselves
     QtSupport::ProFileCacheManager::instance()->discardFile(m_projectFilePath);
     m_project->qmakeProjectManager()->notifyChanged(m_projectFilePath);