calamares/src/modules/partition/core/PartitionCoreModule.cpp

652 lines
19 KiB
C++
Raw Normal View History

/* === This file is part of Calamares - <http://github.com/calamares> ===
*
* Copyright 2014, Aurélien Gâteau <agateau@kde.org>
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
*
* 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/>.
*/
2015-07-03 17:03:33 +02:00
#include "core/PartitionCoreModule.h"
#include "core/BootLoaderModel.h"
#include "core/DeviceModel.h"
#include "core/PartitionInfo.h"
#include "core/PartitionIterator.h"
#include "core/PartitionModel.h"
#include "core/KPMHelpers.h"
#include "core/PartUtils.h"
2015-07-03 17:03:33 +02:00
#include "jobs/ClearMountsJob.h"
#include "jobs/ClearTempMountsJob.h"
#include "jobs/CreatePartitionJob.h"
#include "jobs/CreatePartitionTableJob.h"
#include "jobs/DeletePartitionJob.h"
#include "jobs/FillGlobalStorageJob.h"
#include "jobs/FormatPartitionJob.h"
#include "jobs/ResizePartitionJob.h"
#include "jobs/SetPartitionFlagsJob.h"
2015-07-03 17:03:33 +02:00
#include "Typedefs.h"
#include "utils/Logger.h"
// KPMcore
#include <kpmcore/core/device.h>
#include <kpmcore/core/partition.h>
#include <kpmcore/backend/corebackend.h>
#include <kpmcore/backend/corebackendmanager.h>
#include <kpmcore/fs/filesystemfactory.h>
// Qt
#include <QStandardItemModel>
#include <QDir>
#include <QProcess>
#include <QFutureWatcher>
#include <QtConcurrent/QtConcurrent>
2014-07-30 23:21:06 +02:00
static bool
hasRootPartition( Device* device )
{
for ( auto it = PartitionIterator::begin( device ); it != PartitionIterator::end( device ); ++it )
2014-08-05 09:54:30 +02:00
if ( ( *it )->mountPoint() == "/" )
2014-07-30 23:21:06 +02:00
return true;
return false;
}
//- DeviceInfo ---------------------------------------------
PartitionCoreModule::DeviceInfo::DeviceInfo( Device* _device )
: device( _device )
, partitionModel( new PartitionModel )
{}
PartitionCoreModule::DeviceInfo::~DeviceInfo()
{
}
void
PartitionCoreModule::DeviceInfo::forgetChanges()
{
jobs.clear();
2014-07-17 09:49:24 +02:00
for ( auto it = PartitionIterator::begin( device.data() ); it != PartitionIterator::end( device.data() ); ++it )
PartitionInfo::reset( *it );
2015-12-18 15:02:55 +01:00
partitionModel->revert();
}
2014-07-24 19:28:53 +02:00
bool
PartitionCoreModule::DeviceInfo::isDirty() const
{
if ( !jobs.isEmpty() )
return true;
for ( auto it = PartitionIterator::begin( device.data() ); it != PartitionIterator::end( device.data() ); ++it )
if ( PartitionInfo::isDirty( *it ) )
return true;
return false;
}
//- PartitionCoreModule ------------------------------------
PartitionCoreModule::PartitionCoreModule( QObject* parent )
: QObject( parent )
, m_deviceModel( new DeviceModel( this ) )
, m_bootLoaderModel( new BootLoaderModel( this ) )
{
if ( !KPMHelpers::initKPMcore() )
2015-07-03 17:03:33 +02:00
qFatal( "Failed to initialize KPMcore backend" );
FileSystemFactory::init();
2014-07-24 19:28:53 +02:00
init();
}
void
PartitionCoreModule::init()
{
CoreBackend* backend = CoreBackendManager::self()->backend();
2016-05-09 16:17:35 +02:00
QList< Device* > devices = backend->scanDevices( true );
2014-07-30 23:21:06 +02:00
// Remove the device which contains / from the list
2016-05-09 16:17:35 +02:00
for ( QList< Device* >::iterator it = devices.begin(); it != devices.end(); )
if ( hasRootPartition( *it ) ||
(*it)->deviceNode().startsWith( "/dev/zram") )
2014-07-30 23:21:06 +02:00
it = devices.erase( it );
else
++it;
cDebug() << "LIST OF DETECTED DEVICES:\nnode\tcapacity\tname\tprettyName";
for ( auto device : devices )
{
auto deviceInfo = new DeviceInfo( device );
m_deviceInfos << deviceInfo;
cDebug() << device->deviceNode() << device->capacity() << device->name() << device->prettyName();
}
m_deviceModel->init( devices );
// The following PartUtils::runOsprober call in turn calls PartUtils::canBeResized,
// which relies on a working DeviceModel.
m_osproberLines = PartUtils::runOsprober( this );
for ( auto deviceInfo : m_deviceInfos )
{
deviceInfo->partitionModel->init( deviceInfo->device.data(), m_osproberLines );
}
m_bootLoaderModel->init( devices );
if ( QDir( "/sys/firmware/efi/efivars" ).exists() )
scanForEfiSystemPartitions(); //FIXME: this should be removed in favor of
// proper KPM support for EFI
}
PartitionCoreModule::~PartitionCoreModule()
{
qDeleteAll( m_deviceInfos );
}
DeviceModel*
PartitionCoreModule::deviceModel() const
{
return m_deviceModel;
}
QAbstractItemModel*
PartitionCoreModule::bootLoaderModel() const
{
return m_bootLoaderModel;
}
PartitionModel*
PartitionCoreModule::partitionModelForDevice( Device* device ) const
{
DeviceInfo* info = infoForDevice( device );
Q_ASSERT( info );
return info->partitionModel.data();
}
Device*
PartitionCoreModule::createImmutableDeviceCopy( Device* device )
{
CoreBackend* backend = CoreBackendManager::self()->backend();
2016-05-12 13:57:17 +02:00
QString node = device->deviceNode();
cDebug() << "Creating immutable copy for node:" << node;
Device* deviceBefore = backend->scanDevice( node );
return deviceBefore;
}
void
PartitionCoreModule::createPartitionTable( Device* device, PartitionTable::TableType type )
{
DeviceInfo* info = infoForDevice( device );
2015-06-21 01:27:02 +02:00
if ( info )
{
// Creating a partition table wipes all the disk, so there is no need to
// keep previous changes
info->forgetChanges();
PartitionModel::ResetHelper helper( partitionModelForDevice( device ) );
CreatePartitionTableJob* job = new CreatePartitionTableJob( device, type );
job->updatePreview();
info->jobs << Calamares::job_ptr( job );
}
refresh();
}
void
PartitionCoreModule::createPartition( Device *device,
Partition *partition,
PartitionTable::Flags flags )
{
auto deviceInfo = infoForDevice( device );
Q_ASSERT( deviceInfo );
PartitionModel::ResetHelper helper( partitionModelForDevice( device ) );
CreatePartitionJob* job = new CreatePartitionJob( device, partition );
job->updatePreview();
deviceInfo->jobs << Calamares::job_ptr( job );
2014-07-02 15:49:35 +02:00
if ( flags != PartitionTable::FlagNone )
{
SetPartFlagsJob* fJob = new SetPartFlagsJob( device, partition, flags );
deviceInfo->jobs << Calamares::job_ptr( fJob );
}
refresh();
}
2014-07-02 15:49:35 +02:00
void
PartitionCoreModule::deletePartition( Device* device, Partition* partition )
{
auto deviceInfo = infoForDevice( device );
Q_ASSERT( deviceInfo );
PartitionModel::ResetHelper helper( partitionModelForDevice( device ) );
if ( partition->roles().has( PartitionRole::Extended ) )
{
// Delete all logical partitions first
// I am not sure if we can iterate on Partition::children() while
// deleting them, so let's play it safe and keep our own list.
QList< Partition* > lst;
for ( auto childPartition : partition->children() )
if ( !KPMHelpers::isPartitionFreeSpace( childPartition ) )
lst << childPartition;
for ( auto partition : lst )
deletePartition( device, partition );
}
QList< Calamares::job_ptr >& jobs = deviceInfo->jobs;
2014-07-02 15:49:35 +02:00
if ( partition->state() == Partition::StateNew )
{
// First remove matching SetPartFlagsJobs
for ( auto it = jobs.begin(); it != jobs.end(); )
{
SetPartFlagsJob* job = qobject_cast< SetPartFlagsJob* >( it->data() );
if ( job && job->partition() == partition )
it = jobs.erase( it );
else
++it;
}
2014-07-02 15:49:35 +02:00
// Find matching CreatePartitionJob
auto it = std::find_if( jobs.begin(), jobs.end(), [ partition ]( Calamares::job_ptr job )
2014-07-02 15:49:35 +02:00
{
CreatePartitionJob* createJob = qobject_cast< CreatePartitionJob* >( job.data() );
return createJob && createJob->partition() == partition;
} );
if ( it == jobs.end() )
2014-07-02 15:49:35 +02:00
{
cDebug() << "Failed to find a CreatePartitionJob matching the partition to remove";
return;
}
// Remove it
if ( ! partition->parent()->remove( partition ) )
{
cDebug() << "Failed to remove partition from preview";
return;
}
2014-07-02 15:49:35 +02:00
device->partitionTable()->updateUnallocated( *device );
jobs.erase( it );
2014-07-02 15:49:35 +02:00
// The partition is no longer referenced by either a job or the device
// partition list, so we have to delete it
delete partition;
}
else
{
2014-07-18 14:29:27 +02:00
// Remove any PartitionJob on this partition
for ( auto it = jobs.begin(); it != jobs.end(); )
{
PartitionJob* job = qobject_cast< PartitionJob* >( it->data() );
if ( job && job->partition() == partition )
it = jobs.erase( it );
else
++it;
}
2014-07-02 15:49:35 +02:00
DeletePartitionJob* job = new DeletePartitionJob( device, partition );
job->updatePreview();
jobs << Calamares::job_ptr( job );
2014-07-02 15:49:35 +02:00
}
refresh();
2014-07-02 15:49:35 +02:00
}
void
PartitionCoreModule::formatPartition( Device* device, Partition* partition )
{
auto deviceInfo = infoForDevice( device );
Q_ASSERT( deviceInfo );
PartitionModel::ResetHelper helper( partitionModelForDevice( device ) );
FormatPartitionJob* job = new FormatPartitionJob( device, partition );
deviceInfo->jobs << Calamares::job_ptr( job );
refresh();
}
void
PartitionCoreModule::resizePartition( Device* device,
Partition* partition,
qint64 first,
qint64 last )
{
auto deviceInfo = infoForDevice( device );
Q_ASSERT( deviceInfo );
PartitionModel::ResetHelper helper( partitionModelForDevice( device ) );
ResizePartitionJob* job = new ResizePartitionJob( device, partition, first, last );
job->updatePreview();
deviceInfo->jobs << Calamares::job_ptr( job );
refresh();
}
void
PartitionCoreModule::setPartitionFlags( Device* device,
Partition* partition,
PartitionTable::Flags flags )
{
auto deviceInfo = infoForDevice( device );
Q_ASSERT( deviceInfo );
PartitionModel::ResetHelper( partitionModelForDevice( device ) );
SetPartFlagsJob* job = new SetPartFlagsJob( device, partition, flags );
deviceInfo->jobs << Calamares::job_ptr( job );
refresh();
}
QList< Calamares::job_ptr >
PartitionCoreModule::jobs() const
{
QList< Calamares::job_ptr > lst;
2014-07-22 17:32:26 +02:00
QList< Device* > devices;
lst << Calamares::job_ptr( new ClearTempMountsJob() );
for ( auto info : m_deviceInfos )
{
if ( info->isDirty() )
lst << Calamares::job_ptr( new ClearMountsJob( info->device.data() ) );
}
for ( auto info : m_deviceInfos )
2014-07-22 17:32:26 +02:00
{
lst << info->jobs;
2014-07-22 17:32:26 +02:00
devices << info->device.data();
}
lst << Calamares::job_ptr( new FillGlobalStorageJob( devices, m_bootLoaderInstallPath ) );
QStringList jobsDebug;
foreach ( auto job, lst )
{
jobsDebug.append( job->prettyName() );
}
cDebug() << "PartitionCodeModule has been asked for jobs. About to return:"
<< jobsDebug.join( "\n" );
return lst;
}
2015-05-28 18:32:59 +02:00
bool
PartitionCoreModule::hasRootMountPoint() const
{
return m_hasRootMountPoint;
}
QList< Partition* >
PartitionCoreModule::efiSystemPartitions() const
{
return m_efiSystemPartitions;
}
2014-07-02 15:49:35 +02:00
void
PartitionCoreModule::dumpQueue() const
{
2014-07-16 11:15:22 +02:00
cDebug() << "# Queue:";
for ( auto info : m_deviceInfos )
2014-07-02 15:49:35 +02:00
{
2014-07-16 11:15:22 +02:00
cDebug() << "## Device:" << info->device->name();
for ( auto job : info->jobs )
2014-07-16 11:15:22 +02:00
cDebug() << "-" << job->prettyName();
2014-07-02 15:49:35 +02:00
}
}
2014-07-10 18:55:19 +02:00
OsproberEntryList
PartitionCoreModule::osproberEntries() const
{
return m_osproberLines;
}
2014-07-10 18:55:19 +02:00
void
PartitionCoreModule::refreshPartition( Device* device, Partition* partition )
2014-07-10 18:55:19 +02:00
{
// Keep it simple for now: reset the model. This can be improved to cause
// the model to emit dataChanged() for the affected row instead, avoiding
// the loss of the current selection.
auto model = partitionModelForDevice( device );
2014-07-10 18:55:19 +02:00
Q_ASSERT( model );
PartitionModel::ResetHelper helper( model );
refresh();
}
void
PartitionCoreModule::refresh()
{
updateHasRootMountPoint();
2014-07-24 19:28:53 +02:00
updateIsDirty();
m_bootLoaderModel->update();
if ( QDir( "/sys/firmware/efi/efivars" ).exists() )
scanForEfiSystemPartitions(); //FIXME: this should be removed in favor of
// proper KPM support for EFI
2014-07-10 18:55:19 +02:00
}
void PartitionCoreModule::updateHasRootMountPoint()
{
bool oldValue = m_hasRootMountPoint;
m_hasRootMountPoint = findPartitionByMountPoint( "/" );
if ( oldValue != m_hasRootMountPoint )
hasRootMountPointChanged( m_hasRootMountPoint );
}
2014-07-24 19:28:53 +02:00
void
PartitionCoreModule::updateIsDirty()
{
bool oldValue = m_isDirty;
m_isDirty = false;
for ( auto info : m_deviceInfos )
if ( info->isDirty() )
{
m_isDirty = true;
break;
}
if ( oldValue != m_isDirty )
isDirtyChanged( m_isDirty );
}
void
PartitionCoreModule::scanForEfiSystemPartitions()
{
m_efiSystemPartitions.clear();
QList< Device* > devices;
for ( int row = 0; row < deviceModel()->rowCount(); ++row )
{
Device* device = deviceModel()->deviceForIndex(
deviceModel()->index( row ) );
devices.append( device );
}
//FIXME: Unfortunately right now we have to call sgdisk manually because
// the KPM submodule does not expose the ESP flag from libparted.
// The following findPartitions call and lambda should be scrapped and
// rewritten based on libKPM. -- Teo 5/2015
QList< Partition* > efiSystemPartitions =
KPMHelpers::findPartitions( devices,
[]( Partition* partition ) -> bool
{
QProcess process;
process.setProgram( "sgdisk" );
process.setArguments( { "-i",
QString::number( partition->number() ),
partition->devicePath() } );
process.setProcessChannelMode( QProcess::MergedChannels );
process.start();
if ( process.waitForFinished() )
{
if ( process.readAllStandardOutput()
.contains( "C12A7328-F81F-11D2-BA4B-00A0C93EC93B" ) )
{
cDebug() << "Found EFI system partition at" << partition->partitionPath();
return true;
}
}
return false;
} );
if ( efiSystemPartitions.isEmpty() )
cDebug() << "WARNING: system is EFI but no EFI system partitions found.";
m_efiSystemPartitions = efiSystemPartitions;
}
PartitionCoreModule::DeviceInfo*
PartitionCoreModule::infoForDevice( Device* device ) const
{
for ( auto deviceInfo : m_deviceInfos )
{
if ( deviceInfo->device.data() == device )
return deviceInfo;
}
return nullptr;
}
Partition*
PartitionCoreModule::findPartitionByMountPoint( const QString& mountPoint ) const
{
for ( auto deviceInfo : m_deviceInfos )
{
Device* device = deviceInfo->device.data();
for ( auto it = PartitionIterator::begin( device ); it != PartitionIterator::end( device ); ++it )
if ( PartitionInfo::mountPoint( *it ) == mountPoint )
return *it;
}
return nullptr;
}
void
PartitionCoreModule::setBootLoaderInstallPath( const QString& path )
{
m_bootLoaderInstallPath = path;
}
2014-07-24 19:28:53 +02:00
void
PartitionCoreModule::revert()
{
2015-12-30 16:37:04 +01:00
QMutexLocker locker( &m_revertMutex );
2014-07-24 19:28:53 +02:00
qDeleteAll( m_deviceInfos );
m_deviceInfos.clear();
init();
updateIsDirty();
2015-10-30 17:28:31 +01:00
emit reverted();
2014-07-24 19:28:53 +02:00
}
2015-12-30 16:37:04 +01:00
void
PartitionCoreModule::revertAllDevices()
{
foreach ( DeviceInfo* devInfo, m_deviceInfos )
{
revertDevice( devInfo->device.data() );
}
refresh();
2015-12-30 16:37:04 +01:00
}
2015-12-17 18:00:00 +01:00
void
PartitionCoreModule::revertDevice( Device* dev )
{
2015-12-23 19:14:33 +01:00
QMutexLocker locker( &m_revertMutex );
2015-12-17 18:00:00 +01:00
DeviceInfo* devInfo = infoForDevice( dev );
if ( !devInfo )
return;
devInfo->forgetChanges();
CoreBackend* backend = CoreBackendManager::self()->backend();
Device *newDev = backend->scanDevice( devInfo->device->deviceNode() );
devInfo->device.reset( newDev );
devInfo->partitionModel->init( newDev, m_osproberLines );
2015-12-17 18:00:00 +01:00
m_deviceModel->swapDevice( dev, newDev );
QList< Device* > devices;
foreach ( auto info, m_deviceInfos )
devices.append( info->device.data() );
m_bootLoaderModel->init( devices );
refresh();
emit deviceReverted( newDev );
2015-12-17 18:00:00 +01:00
}
2015-12-23 19:14:33 +01:00
void
PartitionCoreModule::asyncRevertDevice( Device* dev, std::function< void() > callback )
{
QFutureWatcher< void >* watcher = new QFutureWatcher< void >();
connect( watcher, &QFutureWatcher< void >::finished,
this, [ watcher, callback ]
2015-12-23 19:21:08 +01:00
{
callback();
watcher->deleteLater();
2015-12-23 19:21:08 +01:00
} );
QFuture< void > future = QtConcurrent::run( this, &PartitionCoreModule::revertDevice, dev );
watcher->setFuture( future );
2015-12-23 19:14:33 +01:00
}
2014-09-16 18:11:19 +02:00
void
PartitionCoreModule::clearJobs()
{
foreach ( DeviceInfo* deviceInfo, m_deviceInfos )
{
deviceInfo->forgetChanges();
}
updateIsDirty();
}
bool
PartitionCoreModule::isDirty()
{
return m_isDirty;
}
QList< PartitionCoreModule::SummaryInfo >
PartitionCoreModule::createSummaryInfo() const
{
QList< SummaryInfo > lst;
CoreBackend* backend = CoreBackendManager::self()->backend();
for ( auto deviceInfo : m_deviceInfos )
{
if ( !deviceInfo->isDirty() )
continue;
SummaryInfo summaryInfo;
summaryInfo.deviceName = deviceInfo->device->name();
summaryInfo.deviceNode = deviceInfo->device->deviceNode();
Device* deviceBefore = backend->scanDevice( deviceInfo->device->deviceNode() );
summaryInfo.partitionModelBefore = new PartitionModel;
summaryInfo.partitionModelBefore->init( deviceBefore, m_osproberLines );
// Make deviceBefore a child of partitionModelBefore so that it is not
// leaked (as long as partitionModelBefore is deleted)
deviceBefore->setParent( summaryInfo.partitionModelBefore );
summaryInfo.partitionModelAfter = new PartitionModel;
summaryInfo.partitionModelAfter->init( deviceInfo->device.data(), m_osproberLines );
lst << summaryInfo;
}
return lst;
}