2017-12-20 14:39:09 +01:00
|
|
|
/* === This file is part of Calamares - <https://github.com/calamares> ===
|
2014-11-28 15:50:27 +01:00
|
|
|
*
|
2015-02-13 14:02:42 +01:00
|
|
|
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
2014-11-28 15:50:27 +01:00
|
|
|
*
|
|
|
|
* Calamares is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Calamares is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ClearMountsJob.h"
|
|
|
|
|
2015-07-02 13:49:21 +02:00
|
|
|
#include "core/PartitionInfo.h"
|
|
|
|
#include "core/PartitionIterator.h"
|
|
|
|
#include "utils/Logger.h"
|
|
|
|
|
|
|
|
// KPMcore
|
|
|
|
#include <kpmcore/core/device.h>
|
|
|
|
#include <kpmcore/core/partition.h>
|
|
|
|
#include <kpmcore/util/report.h>
|
2014-11-28 15:50:27 +01:00
|
|
|
|
2016-05-13 17:11:13 +02:00
|
|
|
#include <QDir>
|
2015-02-13 14:02:42 +01:00
|
|
|
#include <QProcess>
|
2014-11-28 15:50:27 +01:00
|
|
|
#include <QStringList>
|
|
|
|
|
|
|
|
|
|
|
|
ClearMountsJob::ClearMountsJob( Device* device )
|
|
|
|
: Calamares::Job()
|
|
|
|
, m_device( device )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QString
|
|
|
|
ClearMountsJob::prettyName() const
|
|
|
|
{
|
|
|
|
return tr( "Clear mounts for partitioning operations on %1" )
|
|
|
|
.arg( m_device->deviceNode() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-13 02:24:58 +02:00
|
|
|
QString
|
|
|
|
ClearMountsJob::prettyStatusMessage() const
|
|
|
|
{
|
|
|
|
return tr( "Clearing mounts for partitioning operations on %1." )
|
|
|
|
.arg( m_device->deviceNode() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-28 15:50:27 +01:00
|
|
|
Calamares::JobResult
|
|
|
|
ClearMountsJob::exec()
|
|
|
|
{
|
|
|
|
QStringList goodNews;
|
2015-02-13 14:02:42 +01:00
|
|
|
|
2015-06-11 03:17:52 +02:00
|
|
|
QString deviceName = m_device->deviceNode().split( '/' ).last();
|
|
|
|
|
2015-02-13 14:02:42 +01:00
|
|
|
QProcess process;
|
|
|
|
process.setProgram( "sh" );
|
|
|
|
process.setArguments( {
|
|
|
|
"-c",
|
|
|
|
QString( "echo $(awk '{print $4}' /proc/partitions | sed -e '/name/d' -e '/^$/d' -e '/[1-9]/!d' | grep %1)" )
|
2015-06-11 03:17:52 +02:00
|
|
|
.arg( deviceName )
|
2015-02-13 14:02:42 +01:00
|
|
|
} );
|
|
|
|
process.start();
|
|
|
|
process.waitForFinished();
|
|
|
|
|
2016-09-01 14:21:05 +02:00
|
|
|
const QString partitions = process.readAllStandardOutput();
|
|
|
|
const QStringList partitionsList = partitions.simplified().split( ' ' );
|
2015-02-13 14:02:42 +01:00
|
|
|
|
2015-06-20 20:38:40 +02:00
|
|
|
// Build a list of partitions of type 82 (Linux swap / Solaris).
|
|
|
|
// We then need to clear them just in case they contain something resumable from a
|
|
|
|
// previous suspend-to-disk.
|
|
|
|
QStringList swapPartitions;
|
|
|
|
process.start( "sfdisk", { "-d", m_device->deviceNode() } );
|
|
|
|
process.waitForFinished();
|
|
|
|
// Sample output:
|
|
|
|
// % sudo sfdisk -d /dev/sda
|
|
|
|
// label: dos
|
|
|
|
// label-id: 0x000ced89
|
|
|
|
// device: /dev/sda
|
|
|
|
// unit: sectors
|
|
|
|
|
|
|
|
// /dev/sda1 : start= 63, size= 29329345, type=83, bootable
|
|
|
|
// /dev/sda2 : start= 29331456, size= 2125824, type=82
|
|
|
|
|
|
|
|
swapPartitions = QString::fromLocal8Bit( process.readAllStandardOutput() )
|
|
|
|
.split( '\n' );
|
|
|
|
swapPartitions = swapPartitions.filter( "type=82" );
|
|
|
|
for ( QStringList::iterator it = swapPartitions.begin();
|
|
|
|
it != swapPartitions.end(); ++it )
|
|
|
|
{
|
|
|
|
*it = (*it).simplified().split( ' ' ).first();
|
|
|
|
}
|
|
|
|
|
2016-09-01 14:21:05 +02:00
|
|
|
const QStringList cryptoDevices = getCryptoDevices();
|
|
|
|
for ( const QString &mapperPath : cryptoDevices )
|
2016-05-13 17:11:13 +02:00
|
|
|
{
|
|
|
|
tryUmount( mapperPath );
|
|
|
|
QString news = tryCryptoClose( mapperPath );
|
|
|
|
if ( !news.isEmpty() )
|
|
|
|
goodNews.append( news );
|
|
|
|
}
|
|
|
|
|
2015-06-11 03:17:52 +02:00
|
|
|
// First we umount all LVM logical volumes we can find
|
|
|
|
process.start( "lvscan", { "-a" } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 ) //means LVM2 tools are installed
|
|
|
|
{
|
2016-09-01 14:21:05 +02:00
|
|
|
const QStringList lvscanLines = QString::fromLocal8Bit( process.readAllStandardOutput() ).split( '\n' );
|
|
|
|
for ( const QString& lvscanLine : lvscanLines )
|
2015-06-11 03:17:52 +02:00
|
|
|
{
|
|
|
|
QString lvPath = lvscanLine.simplified().split( ' ' ).value( 1 ); //second column
|
|
|
|
lvPath = lvPath.replace( '\'', "" );
|
|
|
|
|
|
|
|
QString news = tryUmount( lvPath );
|
|
|
|
if ( !news.isEmpty() )
|
|
|
|
goodNews.append( news );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2018-02-13 11:07:12 +01:00
|
|
|
cWarning() << "this system does not seem to have LVM2 tools.";
|
2015-06-11 03:17:52 +02:00
|
|
|
|
|
|
|
// Then we go looking for volume groups that use this device for physical volumes
|
|
|
|
process.start( "pvdisplay", { "-C", "--noheadings" } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 ) //means LVM2 tools are installed
|
|
|
|
{
|
|
|
|
QString pvdisplayOutput = process.readAllStandardOutput();
|
|
|
|
if ( !pvdisplayOutput.simplified().isEmpty() ) //means there is at least one LVM PV
|
|
|
|
{
|
|
|
|
QSet< QString > vgSet;
|
|
|
|
|
2016-09-01 14:21:05 +02:00
|
|
|
const QStringList pvdisplayLines = pvdisplayOutput.split( '\n' );
|
|
|
|
for ( const QString& pvdisplayLine : pvdisplayLines )
|
2015-06-11 03:17:52 +02:00
|
|
|
{
|
|
|
|
QString pvPath = pvdisplayLine.simplified().split( ' ' ).value( 0 );
|
|
|
|
QString vgName = pvdisplayLine.simplified().split( ' ' ).value( 1 );
|
|
|
|
if ( !pvPath.contains( deviceName ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vgSet.insert( vgName );
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ( const QString& vgName, vgSet )
|
|
|
|
{
|
|
|
|
process.start( "vgchange", { "-an", vgName } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 )
|
|
|
|
goodNews.append( QString( "Successfully disabled volume group %1." ).arg( vgName ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2018-02-13 11:07:12 +01:00
|
|
|
cWarning() << "this system does not seem to have LVM2 tools.";
|
2015-06-11 03:17:52 +02:00
|
|
|
|
2016-09-01 14:21:05 +02:00
|
|
|
const QStringList cryptoDevices2 = getCryptoDevices();
|
|
|
|
for ( const QString &mapperPath : cryptoDevices2 )
|
2016-05-13 17:11:13 +02:00
|
|
|
{
|
|
|
|
tryUmount( mapperPath );
|
|
|
|
QString news = tryCryptoClose( mapperPath );
|
|
|
|
if ( !news.isEmpty() )
|
|
|
|
goodNews.append( news );
|
|
|
|
}
|
|
|
|
|
2016-09-01 14:21:05 +02:00
|
|
|
for ( const QString &p : partitionsList )
|
2014-11-28 15:50:27 +01:00
|
|
|
{
|
2015-02-13 14:02:42 +01:00
|
|
|
QString partPath = QString( "/dev/%1" ).arg( p );
|
|
|
|
|
2015-06-11 03:17:52 +02:00
|
|
|
QString news = tryUmount( partPath );
|
|
|
|
if ( !news.isEmpty() )
|
|
|
|
goodNews.append( news );
|
2014-11-28 15:50:27 +01:00
|
|
|
}
|
2015-02-13 14:02:42 +01:00
|
|
|
|
2015-06-20 20:38:40 +02:00
|
|
|
foreach ( QString p, swapPartitions )
|
|
|
|
{
|
|
|
|
QString news = tryClearSwap( p );
|
|
|
|
if ( !news.isEmpty() )
|
|
|
|
goodNews.append( news );
|
|
|
|
}
|
|
|
|
|
2014-11-28 15:50:27 +01:00
|
|
|
Calamares::JobResult ok = Calamares::JobResult::ok();
|
|
|
|
ok.setMessage( tr( "Cleared all mounts for %1" )
|
|
|
|
.arg( m_device->deviceNode() ) );
|
|
|
|
ok.setDetails( goodNews.join( "\n" ) );
|
2015-02-13 11:56:10 +01:00
|
|
|
|
2015-02-19 20:24:14 +01:00
|
|
|
cDebug() << "ClearMountsJob finished. Here's what was done:\n" << goodNews.join( "\n" );
|
|
|
|
|
2014-11-28 15:50:27 +01:00
|
|
|
return ok;
|
|
|
|
}
|
2015-06-11 03:17:52 +02:00
|
|
|
|
|
|
|
|
|
|
|
QString
|
|
|
|
ClearMountsJob::tryUmount( const QString& partPath )
|
|
|
|
{
|
|
|
|
QProcess process;
|
|
|
|
process.start( "umount", { partPath } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 )
|
|
|
|
return QString( "Successfully unmounted %1." ).arg( partPath );
|
|
|
|
|
|
|
|
process.start( "swapoff", { partPath } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 )
|
2015-06-20 20:38:40 +02:00
|
|
|
return QString( "Successfully disabled swap %1." ).arg( partPath );
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QString
|
|
|
|
ClearMountsJob::tryClearSwap( const QString& partPath )
|
|
|
|
{
|
|
|
|
QProcess process;
|
2015-09-17 15:14:18 +02:00
|
|
|
process.start( "blkid", { "-s", "UUID", "-o", "value", partPath } );
|
2015-06-20 20:38:40 +02:00
|
|
|
process.waitForFinished();
|
2015-09-17 15:14:18 +02:00
|
|
|
QString swapPartUuid = QString::fromLocal8Bit( process.readAllStandardOutput() ).simplified();
|
|
|
|
if ( process.exitCode() != 0 ||
|
|
|
|
swapPartUuid.isEmpty() )
|
|
|
|
return QString();
|
2015-06-11 03:17:52 +02:00
|
|
|
|
2015-09-17 15:14:18 +02:00
|
|
|
process.start( "mkswap", { "-U", swapPartUuid, partPath } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() != 0 )
|
|
|
|
return QString();
|
|
|
|
|
|
|
|
return QString( "Successfully cleared swap %1." ).arg( partPath );
|
2015-06-11 03:17:52 +02:00
|
|
|
}
|
2016-05-13 17:11:13 +02:00
|
|
|
|
|
|
|
|
|
|
|
QString
|
|
|
|
ClearMountsJob::tryCryptoClose( const QString& mapperPath )
|
|
|
|
{
|
|
|
|
QProcess process;
|
|
|
|
process.start( "cryptsetup", { "close", mapperPath } );
|
|
|
|
process.waitForFinished();
|
|
|
|
if ( process.exitCode() == 0 )
|
|
|
|
return QString( "Successfully closed mapper device %1." ).arg( mapperPath );
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QStringList
|
2016-09-01 14:21:05 +02:00
|
|
|
ClearMountsJob::getCryptoDevices() const
|
2016-05-13 17:11:13 +02:00
|
|
|
{
|
|
|
|
QDir mapperDir( "/dev/mapper" );
|
2016-09-01 14:21:05 +02:00
|
|
|
const QFileInfoList fiList = mapperDir.entryInfoList( QDir::Files );
|
2016-05-13 17:11:13 +02:00
|
|
|
QStringList list;
|
|
|
|
QProcess process;
|
2016-09-01 14:21:05 +02:00
|
|
|
for ( const QFileInfo &fi : fiList )
|
2016-05-13 17:11:13 +02:00
|
|
|
{
|
|
|
|
if ( fi.baseName() == "control" )
|
|
|
|
continue;
|
|
|
|
list.append( fi.absoluteFilePath() );
|
|
|
|
}
|
|
|
|
return list;
|
|
|
|
}
|