Merge branch 'wip/resize-partition'
This commit is contained in:
commit
1c9265cd21
@ -35,6 +35,13 @@ JobResult::message() const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
JobResult::setMessage( const QString& message )
|
||||||
|
{
|
||||||
|
m_message = message;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
QString
|
QString
|
||||||
JobResult::details() const
|
JobResult::details() const
|
||||||
{
|
{
|
||||||
@ -42,6 +49,13 @@ JobResult::details() const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
JobResult::setDetails( const QString& details )
|
||||||
|
{
|
||||||
|
m_details = details;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
JobResult
|
JobResult
|
||||||
JobResult::ok()
|
JobResult::ok()
|
||||||
{
|
{
|
||||||
|
@ -32,8 +32,10 @@ public:
|
|||||||
operator bool() const;
|
operator bool() const;
|
||||||
|
|
||||||
QString message() const;
|
QString message() const;
|
||||||
|
void setMessage( const QString& message );
|
||||||
|
|
||||||
QString details() const;
|
QString details() const;
|
||||||
|
void setDetails( const QString& details );
|
||||||
|
|
||||||
static JobResult ok();
|
static JobResult ok();
|
||||||
|
|
||||||
|
@ -22,6 +22,8 @@ calamares_add_plugin( partition
|
|||||||
EXPORT_MACRO PLUGINDLLEXPORT_PRO
|
EXPORT_MACRO PLUGINDLLEXPORT_PRO
|
||||||
SOURCES
|
SOURCES
|
||||||
BootLoaderModel.cpp
|
BootLoaderModel.cpp
|
||||||
|
CheckFileSystemJob.cpp
|
||||||
|
ColorUtils.cpp
|
||||||
CreatePartitionDialog.cpp
|
CreatePartitionDialog.cpp
|
||||||
CreatePartitionJob.cpp
|
CreatePartitionJob.cpp
|
||||||
CreatePartitionTableJob.cpp
|
CreatePartitionTableJob.cpp
|
||||||
@ -30,6 +32,7 @@ calamares_add_plugin( partition
|
|||||||
EditExistingPartitionDialog.cpp
|
EditExistingPartitionDialog.cpp
|
||||||
FillGlobalStorageJob.cpp
|
FillGlobalStorageJob.cpp
|
||||||
FormatPartitionJob.cpp
|
FormatPartitionJob.cpp
|
||||||
|
MoveFileSystemJob.cpp
|
||||||
PartitionCoreModule.cpp
|
PartitionCoreModule.cpp
|
||||||
PartitionInfo.cpp
|
PartitionInfo.cpp
|
||||||
PartitionIterator.cpp
|
PartitionIterator.cpp
|
||||||
@ -37,9 +40,10 @@ calamares_add_plugin( partition
|
|||||||
PartitionModel.cpp
|
PartitionModel.cpp
|
||||||
PartitionPage.cpp
|
PartitionPage.cpp
|
||||||
PartitionPreview.cpp
|
PartitionPreview.cpp
|
||||||
PartitionSizeWidget.cpp
|
PartitionSizeController.cpp
|
||||||
PartitionViewStep.cpp
|
PartitionViewStep.cpp
|
||||||
PMUtils.cpp
|
PMUtils.cpp
|
||||||
|
ResizePartitionJob.cpp
|
||||||
UI
|
UI
|
||||||
CreatePartitionDialog.ui
|
CreatePartitionDialog.ui
|
||||||
CreatePartitionTableDialog.ui
|
CreatePartitionTableDialog.ui
|
||||||
|
56
src/modules/partition/CheckFileSystemJob.cpp
Normal file
56
src/modules/partition/CheckFileSystemJob.cpp
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <CheckFileSystemJob.h>
|
||||||
|
|
||||||
|
// CalaPM
|
||||||
|
#include <core/partition.h>
|
||||||
|
#include <fs/filesystem.h>
|
||||||
|
#include <util/report.h>
|
||||||
|
|
||||||
|
CheckFileSystemJob::CheckFileSystemJob( Partition* partition )
|
||||||
|
: PartitionJob( partition )
|
||||||
|
{}
|
||||||
|
|
||||||
|
QString
|
||||||
|
CheckFileSystemJob::prettyName() const
|
||||||
|
{
|
||||||
|
QString path = partition()->partitionPath();
|
||||||
|
return tr( "Checking file system on partition %1." ).arg( path );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult
|
||||||
|
CheckFileSystemJob::exec()
|
||||||
|
{
|
||||||
|
FileSystem& fs = partition()->fileSystem();
|
||||||
|
|
||||||
|
// if we cannot check, assume everything is fine
|
||||||
|
if ( fs.supportCheck() != FileSystem::cmdSupportFileSystem )
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
|
||||||
|
Report report( nullptr );
|
||||||
|
bool ok = fs.check( report, partition()->partitionPath() );
|
||||||
|
if ( !ok )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
tr( "The file system check on partition %1 failed." )
|
||||||
|
.arg( partition()->partitionPath() ),
|
||||||
|
report.toText()
|
||||||
|
);
|
||||||
|
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
33
src/modules/partition/CheckFileSystemJob.h
Normal file
33
src/modules/partition/CheckFileSystemJob.h
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef CHECKFILESYSTEMJOB_H
|
||||||
|
#define CHECKFILESYSTEMJOB_H
|
||||||
|
|
||||||
|
#include <PartitionJob.h>
|
||||||
|
|
||||||
|
class CheckFileSystemJob : public PartitionJob
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CheckFileSystemJob( Partition* partition );
|
||||||
|
|
||||||
|
QString prettyName() const override;
|
||||||
|
Calamares::JobResult exec() override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* CHECKFILESYSTEMJOB_H */
|
85
src/modules/partition/ColorUtils.cpp
Normal file
85
src/modules/partition/ColorUtils.cpp
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <ColorUtils.h>
|
||||||
|
|
||||||
|
#include <PMUtils.h>
|
||||||
|
|
||||||
|
// CalaPM
|
||||||
|
#include <core/partition.h>
|
||||||
|
|
||||||
|
// Qt
|
||||||
|
#include <QColor>
|
||||||
|
|
||||||
|
static QColor COLORS[ 4 ] =
|
||||||
|
{
|
||||||
|
"#448eca",
|
||||||
|
"#a5cc42",
|
||||||
|
"#d87e30",
|
||||||
|
"#ffbdbd",
|
||||||
|
};
|
||||||
|
static QColor FREE_SPACE_COLOR = "#777777";
|
||||||
|
static QColor EXTENDED_COLOR = "#aaaaaa";
|
||||||
|
|
||||||
|
|
||||||
|
namespace ColorUtils
|
||||||
|
{
|
||||||
|
|
||||||
|
QColor freeSpaceColor()
|
||||||
|
{
|
||||||
|
return FREE_SPACE_COLOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor colorForPartition( Partition* partition )
|
||||||
|
{
|
||||||
|
if ( PMUtils::isPartitionFreeSpace( partition ) )
|
||||||
|
return FREE_SPACE_COLOR;
|
||||||
|
if ( partition->roles().has( PartitionRole::Extended ) )
|
||||||
|
return EXTENDED_COLOR;
|
||||||
|
// No partition-specific color needed, pick one from our list, but skip
|
||||||
|
// free space: we don't want a partition to change colors if space before
|
||||||
|
// it is inserted or removed
|
||||||
|
PartitionNode* parent = partition->parent();
|
||||||
|
Q_ASSERT( parent );
|
||||||
|
int colorIdx = 0;
|
||||||
|
for ( auto child : parent->children() )
|
||||||
|
{
|
||||||
|
if ( child == partition )
|
||||||
|
break;
|
||||||
|
if ( !PMUtils::isPartitionFreeSpace( child ) )
|
||||||
|
++colorIdx;
|
||||||
|
}
|
||||||
|
return COLORS[ colorIdx % 4 ];
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor colorForPartitionInFreeSpace( Partition* partition )
|
||||||
|
{
|
||||||
|
PartitionNode* parent = partition->parent();
|
||||||
|
Q_ASSERT( parent );
|
||||||
|
int colorIdx = 0;
|
||||||
|
for ( auto child : parent->children() )
|
||||||
|
{
|
||||||
|
if ( child == partition )
|
||||||
|
break;
|
||||||
|
if ( !PMUtils::isPartitionFreeSpace( child ) )
|
||||||
|
++colorIdx;
|
||||||
|
}
|
||||||
|
return COLORS[ colorIdx % 4 ];
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
@ -15,36 +15,22 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
* along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
#ifndef COLORUTILS_H
|
||||||
|
#define COLORUTILS_H
|
||||||
|
|
||||||
#ifndef PARTITIONSIZEWIDGET_H
|
class QColor;
|
||||||
#define PARTITIONSIZEWIDGET_H
|
|
||||||
|
|
||||||
#include <QSpinBox>
|
|
||||||
|
|
||||||
class Device;
|
|
||||||
class Partition;
|
class Partition;
|
||||||
|
|
||||||
class PartitionSizeWidget : public QSpinBox
|
namespace ColorUtils
|
||||||
{
|
{
|
||||||
public:
|
|
||||||
typedef QPair< qint64, qint64 > SectorRange;
|
|
||||||
|
|
||||||
explicit PartitionSizeWidget( QWidget* parent = nullptr );
|
QColor freeSpaceColor();
|
||||||
void init( Device* device, Partition* partition );
|
|
||||||
|
|
||||||
SectorRange sectorRange() const;
|
QColor colorForPartition( Partition* partition );
|
||||||
|
|
||||||
bool isDirty() const;
|
QColor colorForPartitionInFreeSpace( Partition* partition );
|
||||||
|
|
||||||
private:
|
}
|
||||||
Device* m_device = nullptr;
|
|
||||||
Partition* m_partition = nullptr;
|
|
||||||
int m_initialValue;
|
|
||||||
|
|
||||||
qint64 mbSizeForSectorRange( qint64 first, qint64 last ) const;
|
#endif /* COLORUTILS_H */
|
||||||
|
|
||||||
qint64 computeMinSector() const;
|
|
||||||
qint64 computeMaxSector() const;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif /* PARTITIONSIZEWIDGET_H */
|
|
@ -18,7 +18,9 @@
|
|||||||
|
|
||||||
#include <CreatePartitionDialog.h>
|
#include <CreatePartitionDialog.h>
|
||||||
|
|
||||||
|
#include <ColorUtils.h>
|
||||||
#include <PartitionInfo.h>
|
#include <PartitionInfo.h>
|
||||||
|
#include <PartitionSizeController.h>
|
||||||
#include <PMUtils.h>
|
#include <PMUtils.h>
|
||||||
#include <ui_CreatePartitionDialog.h>
|
#include <ui_CreatePartitionDialog.h>
|
||||||
#include <utils/Logger.h>
|
#include <utils/Logger.h>
|
||||||
@ -47,8 +49,6 @@ CreatePartitionDialog::CreatePartitionDialog( Device* device, PartitionNode* par
|
|||||||
{
|
{
|
||||||
m_ui->setupUi( this );
|
m_ui->setupUi( this );
|
||||||
|
|
||||||
FileSystemFactory::init();
|
|
||||||
|
|
||||||
if ( device->partitionTable()->type() == PartitionTable::msdos )
|
if ( device->partitionTable()->type() == PartitionTable::msdos )
|
||||||
initMbrPartitionTypeUi();
|
initMbrPartitionTypeUi();
|
||||||
else
|
else
|
||||||
@ -118,7 +118,8 @@ CreatePartitionDialog::createPartition()
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
PartitionSizeWidget::SectorRange range = m_ui->sizeSpinBox->sectorRange();
|
qint64 first = m_partResizerWidgetPartition->firstSector();
|
||||||
|
qint64 last = m_partResizerWidgetPartition->lastSector();
|
||||||
|
|
||||||
FileSystem::Type fsType = m_role.has( PartitionRole::Extended )
|
FileSystem::Type fsType = m_role.has( PartitionRole::Extended )
|
||||||
? FileSystem::Extended
|
? FileSystem::Extended
|
||||||
@ -127,7 +128,7 @@ CreatePartitionDialog::createPartition()
|
|||||||
m_parent,
|
m_parent,
|
||||||
*m_device,
|
*m_device,
|
||||||
m_role,
|
m_role,
|
||||||
fsType, range.first, range.second );
|
fsType, first, last );
|
||||||
|
|
||||||
PartitionInfo::setMountPoint( partition, m_ui->mountPointComboBox->currentText() );
|
PartitionInfo::setMountPoint( partition, m_ui->mountPointComboBox->currentText() );
|
||||||
PartitionInfo::setFormat( partition, true );
|
PartitionInfo::setFormat( partition, true );
|
||||||
@ -147,10 +148,28 @@ CreatePartitionDialog::updateMountPointUi()
|
|||||||
m_ui->mountPointComboBox->setEnabled( enabled );
|
m_ui->mountPointComboBox->setEnabled( enabled );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
CreatePartitionDialog::initPartResizerWidget( Partition* partition )
|
||||||
|
{
|
||||||
|
PartitionSizeController* controller = new PartitionSizeController( this );
|
||||||
|
m_partResizerWidgetPartition.reset( PMUtils::clonePartition( m_device, partition ) );
|
||||||
|
|
||||||
|
qint64 minFirstSector = partition->firstSector() - m_device->partitionTable()->freeSectorsBefore( *partition );
|
||||||
|
qint64 maxLastSector = partition->lastSector() + m_device->partitionTable()->freeSectorsAfter( *partition );
|
||||||
|
m_ui->partResizerWidget->init( *m_device, *m_partResizerWidgetPartition, minFirstSector, maxLastSector );
|
||||||
|
|
||||||
|
QColor color = PMUtils::isPartitionFreeSpace( partition )
|
||||||
|
? ColorUtils::colorForPartitionInFreeSpace( partition )
|
||||||
|
: ColorUtils::colorForPartition( partition );
|
||||||
|
controller->init( m_device, m_partResizerWidgetPartition.data(), color );
|
||||||
|
controller->setPartResizerWidget( m_ui->partResizerWidget );
|
||||||
|
controller->setSpinBox( m_ui->sizeSpinBox );
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
CreatePartitionDialog::initFromFreeSpace( Partition* freeSpacePartition )
|
CreatePartitionDialog::initFromFreeSpace( Partition* freeSpacePartition )
|
||||||
{
|
{
|
||||||
m_ui->sizeSpinBox->init( m_device, freeSpacePartition );
|
initPartResizerWidget( freeSpacePartition );
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -166,7 +185,7 @@ CreatePartitionDialog::initFromPartitionToCreate( Partition* partition )
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_ui->sizeSpinBox->init( m_device, partition );
|
initPartResizerWidget( partition );
|
||||||
|
|
||||||
// File System
|
// File System
|
||||||
FileSystem::Type fsType = partition->fileSystem().type();
|
FileSystem::Type fsType = partition->fileSystem().type();
|
||||||
|
@ -49,10 +49,11 @@ private:
|
|||||||
Device* m_device;
|
Device* m_device;
|
||||||
PartitionNode* m_parent;
|
PartitionNode* m_parent;
|
||||||
PartitionRole m_role = PartitionRole( PartitionRole::None );
|
PartitionRole m_role = PartitionRole( PartitionRole::None );
|
||||||
|
QScopedPointer< Partition > m_partResizerWidgetPartition;
|
||||||
|
|
||||||
void initGptPartitionTypeUi();
|
void initGptPartitionTypeUi();
|
||||||
void initMbrPartitionTypeUi();
|
void initMbrPartitionTypeUi();
|
||||||
void initSectorRange( Partition* );
|
void initPartResizerWidget( Partition* );
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* CREATEPARTITIONDIALOG_H */
|
#endif /* CREATEPARTITIONDIALOG_H */
|
||||||
|
@ -14,9 +14,25 @@
|
|||||||
<string>Create a Partition</string>
|
<string>Create a Partition</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
|
<item>
|
||||||
|
<widget class="PartResizerWidget" name="partResizerWidget" native="true">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Fixed">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="minimumSize">
|
||||||
|
<size>
|
||||||
|
<width>0</width>
|
||||||
|
<height>59</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<layout class="QFormLayout" name="formLayout">
|
<layout class="QFormLayout" name="formLayout">
|
||||||
<item row="0" column="0">
|
<item row="1" column="0">
|
||||||
<widget class="QLabel" name="partitionTypeLabel">
|
<widget class="QLabel" name="partitionTypeLabel">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Partition &Type:</string>
|
<string>Partition &Type:</string>
|
||||||
@ -26,7 +42,7 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="0" column="1">
|
<item row="1" column="1">
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout">
|
<layout class="QHBoxLayout" name="horizontalLayout">
|
||||||
<item>
|
<item>
|
||||||
<widget class="QRadioButton" name="primaryRadioButton">
|
<widget class="QRadioButton" name="primaryRadioButton">
|
||||||
@ -67,7 +83,7 @@
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item row="2" column="0">
|
<item row="3" column="0">
|
||||||
<widget class="QLabel" name="label_2">
|
<widget class="QLabel" name="label_2">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>F&ile System:</string>
|
<string>F&ile System:</string>
|
||||||
@ -77,27 +93,10 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="2" column="1">
|
<item row="3" column="1">
|
||||||
<widget class="QComboBox" name="fsComboBox"/>
|
<widget class="QComboBox" name="fsComboBox"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="3" column="0">
|
<item row="5" column="1">
|
||||||
<widget class="QLabel" name="label">
|
|
||||||
<property name="text">
|
|
||||||
<string>Si&ze:</string>
|
|
||||||
</property>
|
|
||||||
<property name="buddy">
|
|
||||||
<cstring>sizeSpinBox</cstring>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="3" column="1">
|
|
||||||
<widget class="PartitionSizeWidget" name="sizeSpinBox">
|
|
||||||
<property name="suffix">
|
|
||||||
<string> MB</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="4" column="1">
|
|
||||||
<spacer name="verticalSpacer_2">
|
<spacer name="verticalSpacer_2">
|
||||||
<property name="orientation">
|
<property name="orientation">
|
||||||
<enum>Qt::Vertical</enum>
|
<enum>Qt::Vertical</enum>
|
||||||
@ -113,7 +112,7 @@
|
|||||||
</property>
|
</property>
|
||||||
</spacer>
|
</spacer>
|
||||||
</item>
|
</item>
|
||||||
<item row="5" column="0">
|
<item row="6" column="0">
|
||||||
<widget class="QLabel" name="mountPointLabel">
|
<widget class="QLabel" name="mountPointLabel">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>&Mount Point:</string>
|
<string>&Mount Point:</string>
|
||||||
@ -123,7 +122,7 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="5" column="1">
|
<item row="6" column="1">
|
||||||
<widget class="QComboBox" name="mountPointComboBox">
|
<widget class="QComboBox" name="mountPointComboBox">
|
||||||
<property name="editable">
|
<property name="editable">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
@ -163,7 +162,7 @@
|
|||||||
</item>
|
</item>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="6" column="0">
|
<item row="7" column="0">
|
||||||
<spacer name="verticalSpacer">
|
<spacer name="verticalSpacer">
|
||||||
<property name="orientation">
|
<property name="orientation">
|
||||||
<enum>Qt::Vertical</enum>
|
<enum>Qt::Vertical</enum>
|
||||||
@ -176,7 +175,7 @@
|
|||||||
</property>
|
</property>
|
||||||
</spacer>
|
</spacer>
|
||||||
</item>
|
</item>
|
||||||
<item row="1" column="1">
|
<item row="2" column="1">
|
||||||
<spacer name="verticalSpacer_3">
|
<spacer name="verticalSpacer_3">
|
||||||
<property name="orientation">
|
<property name="orientation">
|
||||||
<enum>Qt::Vertical</enum>
|
<enum>Qt::Vertical</enum>
|
||||||
@ -189,6 +188,23 @@
|
|||||||
</property>
|
</property>
|
||||||
</spacer>
|
</spacer>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="0" column="0">
|
||||||
|
<widget class="QLabel" name="label">
|
||||||
|
<property name="text">
|
||||||
|
<string>Si&ze:</string>
|
||||||
|
</property>
|
||||||
|
<property name="buddy">
|
||||||
|
<cstring>sizeSpinBox</cstring>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="0" column="1">
|
||||||
|
<widget class="QSpinBox" name="sizeSpinBox">
|
||||||
|
<property name="suffix">
|
||||||
|
<string> MB</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
@ -205,15 +221,15 @@
|
|||||||
</widget>
|
</widget>
|
||||||
<customwidgets>
|
<customwidgets>
|
||||||
<customwidget>
|
<customwidget>
|
||||||
<class>PartitionSizeWidget</class>
|
<class>PartResizerWidget</class>
|
||||||
<extends>QSpinBox</extends>
|
<extends>QWidget</extends>
|
||||||
<header location="global">PartitionSizeWidget.h</header>
|
<header location="global">gui/partresizerwidget.h</header>
|
||||||
|
<container>1</container>
|
||||||
</customwidget>
|
</customwidget>
|
||||||
</customwidgets>
|
</customwidgets>
|
||||||
<tabstops>
|
<tabstops>
|
||||||
<tabstop>primaryRadioButton</tabstop>
|
<tabstop>primaryRadioButton</tabstop>
|
||||||
<tabstop>fsComboBox</tabstop>
|
<tabstop>fsComboBox</tabstop>
|
||||||
<tabstop>sizeSpinBox</tabstop>
|
|
||||||
</tabstops>
|
</tabstops>
|
||||||
<resources/>
|
<resources/>
|
||||||
<connections>
|
<connections>
|
||||||
|
@ -18,8 +18,10 @@
|
|||||||
|
|
||||||
#include <EditExistingPartitionDialog.h>
|
#include <EditExistingPartitionDialog.h>
|
||||||
|
|
||||||
|
#include <ColorUtils.h>
|
||||||
#include <PartitionCoreModule.h>
|
#include <PartitionCoreModule.h>
|
||||||
#include <PartitionInfo.h>
|
#include <PartitionInfo.h>
|
||||||
|
#include <PartitionSizeController.h>
|
||||||
#include <PMUtils.h>
|
#include <PMUtils.h>
|
||||||
#include <ui_EditExistingPartitionDialog.h>
|
#include <ui_EditExistingPartitionDialog.h>
|
||||||
#include <utils/Logger.h>
|
#include <utils/Logger.h>
|
||||||
@ -27,6 +29,7 @@
|
|||||||
// CalaPM
|
// CalaPM
|
||||||
#include <core/device.h>
|
#include <core/device.h>
|
||||||
#include <core/partition.h>
|
#include <core/partition.h>
|
||||||
|
#include <fs/filesystemfactory.h>
|
||||||
|
|
||||||
// Qt
|
// Qt
|
||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
@ -36,10 +39,26 @@ EditExistingPartitionDialog::EditExistingPartitionDialog( Device* device, Partit
|
|||||||
, m_ui( new Ui_EditExistingPartitionDialog )
|
, m_ui( new Ui_EditExistingPartitionDialog )
|
||||||
, m_device( device )
|
, m_device( device )
|
||||||
, m_partition( partition )
|
, m_partition( partition )
|
||||||
|
, m_partitionSizeController( new PartitionSizeController( this ) )
|
||||||
{
|
{
|
||||||
m_ui->setupUi( this );
|
m_ui->setupUi( this );
|
||||||
m_ui->sizeSpinBox->init( device, partition );
|
|
||||||
|
// Create a partition for partResizerWidget because it alters the first and
|
||||||
|
// last sectors when used
|
||||||
|
m_partResizerWidgetPartition.reset( PMUtils::clonePartition( m_device, m_partition ) );
|
||||||
|
|
||||||
|
QColor color = ColorUtils::colorForPartition( m_partition );
|
||||||
|
m_partitionSizeController->init( m_device, m_partResizerWidgetPartition.data(), color );
|
||||||
|
m_partitionSizeController->setSpinBox( m_ui->sizeSpinBox );
|
||||||
|
|
||||||
m_ui->mountPointComboBox->setCurrentText( PartitionInfo::mountPoint( partition ) );
|
m_ui->mountPointComboBox->setCurrentText( PartitionInfo::mountPoint( partition ) );
|
||||||
|
|
||||||
|
replacePartResizerWidget();
|
||||||
|
|
||||||
|
connect( m_ui->formatRadioButton, &QAbstractButton::toggled, [ this ]( bool )
|
||||||
|
{
|
||||||
|
replacePartResizerWidget();
|
||||||
|
} );
|
||||||
}
|
}
|
||||||
|
|
||||||
EditExistingPartitionDialog::~EditExistingPartitionDialog()
|
EditExistingPartitionDialog::~EditExistingPartitionDialog()
|
||||||
@ -50,9 +69,12 @@ EditExistingPartitionDialog::applyChanges( PartitionCoreModule* core )
|
|||||||
{
|
{
|
||||||
PartitionInfo::setMountPoint( m_partition, m_ui->mountPointComboBox->currentText() );
|
PartitionInfo::setMountPoint( m_partition, m_ui->mountPointComboBox->currentText() );
|
||||||
|
|
||||||
if ( m_ui->sizeSpinBox->isDirty() )
|
qint64 newFirstSector = m_partResizerWidgetPartition->firstSector();
|
||||||
|
qint64 newLastSector = m_partResizerWidgetPartition->lastSector();
|
||||||
|
bool partitionChanged = newFirstSector != m_partition->firstSector() || newLastSector != m_partition->lastSector();
|
||||||
|
|
||||||
|
if ( partitionChanged )
|
||||||
{
|
{
|
||||||
PartitionSizeWidget::SectorRange range = m_ui->sizeSpinBox->sectorRange();
|
|
||||||
if ( m_ui->formatRadioButton->isChecked() )
|
if ( m_ui->formatRadioButton->isChecked() )
|
||||||
{
|
{
|
||||||
Partition* newPartition = PMUtils::createNewPartition(
|
Partition* newPartition = PMUtils::createNewPartition(
|
||||||
@ -60,8 +82,8 @@ EditExistingPartitionDialog::applyChanges( PartitionCoreModule* core )
|
|||||||
*m_device,
|
*m_device,
|
||||||
m_partition->roles(),
|
m_partition->roles(),
|
||||||
m_partition->fileSystem().type(),
|
m_partition->fileSystem().type(),
|
||||||
range.first,
|
newFirstSector,
|
||||||
range.second );
|
newLastSector );
|
||||||
PartitionInfo::setMountPoint( newPartition, PartitionInfo::mountPoint( m_partition ) );
|
PartitionInfo::setMountPoint( newPartition, PartitionInfo::mountPoint( m_partition ) );
|
||||||
PartitionInfo::setFormat( newPartition, true );
|
PartitionInfo::setFormat( newPartition, true );
|
||||||
|
|
||||||
@ -69,9 +91,7 @@ EditExistingPartitionDialog::applyChanges( PartitionCoreModule* core )
|
|||||||
core->createPartition( m_device, newPartition );
|
core->createPartition( m_device, newPartition );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
core->resizePartition( m_device, m_partition, newFirstSector, newLastSector );
|
||||||
//core->resizePartition( m_device, m_partition );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -82,3 +102,39 @@ EditExistingPartitionDialog::applyChanges( PartitionCoreModule* core )
|
|||||||
core->refreshPartition( m_device, m_partition );
|
core->refreshPartition( m_device, m_partition );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
EditExistingPartitionDialog::replacePartResizerWidget()
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* There is no way to reliably update the partition used by
|
||||||
|
* PartResizerWidget, which is necessary when we switch between "format" and
|
||||||
|
* "keep". This is a hack which replaces the existing PartResizerWidget
|
||||||
|
* with a new one.
|
||||||
|
*/
|
||||||
|
|
||||||
|
bool format = m_ui->formatRadioButton->isChecked();
|
||||||
|
|
||||||
|
qint64 used = format ? 0 : m_partition->fileSystem().sectorsUsed();
|
||||||
|
m_partResizerWidgetPartition->fileSystem().setSectorsUsed( used );
|
||||||
|
|
||||||
|
qint64 minFirstSector = m_partition->firstSector() - m_device->partitionTable()->freeSectorsBefore( *m_partition );
|
||||||
|
qint64 maxLastSector = m_partition->lastSector() + m_device->partitionTable()->freeSectorsAfter( *m_partition );
|
||||||
|
|
||||||
|
PartResizerWidget* widget = new PartResizerWidget( this );
|
||||||
|
widget->init( *m_device, *m_partResizerWidgetPartition, minFirstSector, maxLastSector );
|
||||||
|
|
||||||
|
if ( !format )
|
||||||
|
{
|
||||||
|
// If we are not formatting, make sure the space between the first and
|
||||||
|
// last sectors is big enough to fit the existing content.
|
||||||
|
widget->updateLastSector( m_partResizerWidgetPartition->lastSector() );
|
||||||
|
widget->updateFirstSector( m_partResizerWidgetPartition->firstSector() );
|
||||||
|
}
|
||||||
|
|
||||||
|
layout()->replaceWidget( m_ui->partResizerWidget, widget );
|
||||||
|
delete m_ui->partResizerWidget;
|
||||||
|
m_ui->partResizerWidget = widget;
|
||||||
|
|
||||||
|
m_partitionSizeController->setPartResizerWidget( widget );
|
||||||
|
}
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
class PartitionCoreModule;
|
class PartitionCoreModule;
|
||||||
class Device;
|
class Device;
|
||||||
class Partition;
|
class Partition;
|
||||||
|
class PartitionSizeController;
|
||||||
class Ui_EditExistingPartitionDialog;
|
class Ui_EditExistingPartitionDialog;
|
||||||
|
|
||||||
class EditExistingPartitionDialog : public QDialog
|
class EditExistingPartitionDialog : public QDialog
|
||||||
@ -40,6 +41,10 @@ private:
|
|||||||
QScopedPointer< Ui_EditExistingPartitionDialog > m_ui;
|
QScopedPointer< Ui_EditExistingPartitionDialog > m_ui;
|
||||||
Device* m_device;
|
Device* m_device;
|
||||||
Partition* m_partition;
|
Partition* m_partition;
|
||||||
|
QScopedPointer< Partition > m_partResizerWidgetPartition;
|
||||||
|
PartitionSizeController* m_partitionSizeController;
|
||||||
|
|
||||||
|
void replacePartResizerWidget();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* EDITEXISTINGPARTITIONDIALOG_H */
|
#endif /* EDITEXISTINGPARTITIONDIALOG_H */
|
||||||
|
@ -7,11 +7,11 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>350</width>
|
<width>350</width>
|
||||||
<height>203</height>
|
<height>236</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="sizePolicy">
|
<property name="sizePolicy">
|
||||||
<sizepolicy hsizetype="Minimum" vsizetype="Minimum">
|
<sizepolicy hsizetype="Preferred" vsizetype="Minimum">
|
||||||
<horstretch>0</horstretch>
|
<horstretch>0</horstretch>
|
||||||
<verstretch>0</verstretch>
|
<verstretch>0</verstretch>
|
||||||
</sizepolicy>
|
</sizepolicy>
|
||||||
@ -20,29 +20,31 @@
|
|||||||
<string>Edit Existing Partition</string>
|
<string>Edit Existing Partition</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
|
<property name="sizeConstraint">
|
||||||
|
<enum>QLayout::SetMinimumSize</enum>
|
||||||
|
</property>
|
||||||
|
<item>
|
||||||
|
<widget class="PartResizerWidget" name="partResizerWidget" native="true">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Fixed">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="minimumSize">
|
||||||
|
<size>
|
||||||
|
<width>0</width>
|
||||||
|
<height>59</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<layout class="QFormLayout" name="formLayout">
|
<layout class="QFormLayout" name="formLayout">
|
||||||
<property name="fieldGrowthPolicy">
|
<property name="fieldGrowthPolicy">
|
||||||
<enum>QFormLayout::ExpandingFieldsGrow</enum>
|
<enum>QFormLayout::ExpandingFieldsGrow</enum>
|
||||||
</property>
|
</property>
|
||||||
<item row="0" column="0">
|
<item row="2" column="0">
|
||||||
<widget class="QLabel" name="label">
|
|
||||||
<property name="text">
|
|
||||||
<string>Si&ze:</string>
|
|
||||||
</property>
|
|
||||||
<property name="buddy">
|
|
||||||
<cstring>sizeSpinBox</cstring>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="0" column="1">
|
|
||||||
<widget class="PartitionSizeWidget" name="sizeSpinBox">
|
|
||||||
<property name="suffix">
|
|
||||||
<string> MB</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="1" column="0">
|
|
||||||
<widget class="QLabel" name="label_3">
|
<widget class="QLabel" name="label_3">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Content:</string>
|
<string>Content:</string>
|
||||||
@ -52,7 +54,7 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="1" column="1">
|
<item row="2" column="1">
|
||||||
<widget class="QRadioButton" name="keepRadioButton">
|
<widget class="QRadioButton" name="keepRadioButton">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Keep</string>
|
<string>Keep</string>
|
||||||
@ -62,15 +64,21 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="2" column="1">
|
<item row="3" column="1">
|
||||||
<widget class="QRadioButton" name="formatRadioButton">
|
<widget class="QRadioButton" name="formatRadioButton">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Format</string>
|
<string>Format</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="3" column="1">
|
<item row="4" column="1">
|
||||||
<widget class="QLabel" name="label_2">
|
<widget class="QLabel" name="label_2">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Fixed">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Warning: Formatting the partition will erase all existing data.</string>
|
<string>Warning: Formatting the partition will erase all existing data.</string>
|
||||||
</property>
|
</property>
|
||||||
@ -129,6 +137,19 @@
|
|||||||
</item>
|
</item>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="1" column="0">
|
||||||
|
<widget class="QLabel" name="label">
|
||||||
|
<property name="text">
|
||||||
|
<string>Size:</string>
|
||||||
|
</property>
|
||||||
|
<property name="buddy">
|
||||||
|
<cstring>sizeSpinBox</cstring>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="1">
|
||||||
|
<widget class="QSpinBox" name="sizeSpinBox"/>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
@ -145,9 +166,10 @@
|
|||||||
</widget>
|
</widget>
|
||||||
<customwidgets>
|
<customwidgets>
|
||||||
<customwidget>
|
<customwidget>
|
||||||
<class>PartitionSizeWidget</class>
|
<class>PartResizerWidget</class>
|
||||||
<extends>QSpinBox</extends>
|
<extends>QWidget</extends>
|
||||||
<header location="global">PartitionSizeWidget.h</header>
|
<header location="global">gui/partresizerwidget.h</header>
|
||||||
|
<container>1</container>
|
||||||
</customwidget>
|
</customwidget>
|
||||||
</customwidgets>
|
</customwidgets>
|
||||||
<tabstops>
|
<tabstops>
|
||||||
|
238
src/modules/partition/MoveFileSystemJob.cpp
Normal file
238
src/modules/partition/MoveFileSystemJob.cpp
Normal file
@ -0,0 +1,238 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// This class is heavily based on the MoveFileSystemJob class from KDE Partition
|
||||||
|
// Manager.
|
||||||
|
// The copyBlock functions come from Partition Manager Job class.
|
||||||
|
// Original copyright follow:
|
||||||
|
|
||||||
|
/***************************************************************************
|
||||||
|
* Copyright (C) 2008 by Volker Lanz <vl@fidra.de> *
|
||||||
|
* *
|
||||||
|
* This program 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 2 of the License, or *
|
||||||
|
* (at your option) any later version. *
|
||||||
|
* *
|
||||||
|
* This program 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 this program; if not, write to the *
|
||||||
|
* Free Software Foundation, Inc., *
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
|
||||||
|
***************************************************************************/
|
||||||
|
|
||||||
|
#include <MoveFileSystemJob.h>
|
||||||
|
|
||||||
|
#include <utils/Logger.h>
|
||||||
|
|
||||||
|
// CalaPM
|
||||||
|
#include <core/copysourcedevice.h>
|
||||||
|
#include <core/copytargetdevice.h>
|
||||||
|
#include <core/device.h>
|
||||||
|
#include <core/partition.h>
|
||||||
|
#include <fs/filesystem.h>
|
||||||
|
#include <util/report.h>
|
||||||
|
|
||||||
|
MoveFileSystemJob::MoveFileSystemJob( Device* device, Partition* partition, qint64 oldFirstSector, qint64 newFirstSector, qint64 length )
|
||||||
|
: PartitionJob( partition )
|
||||||
|
, m_device( device )
|
||||||
|
, m_oldFirstSector( oldFirstSector )
|
||||||
|
, m_newFirstSector( newFirstSector )
|
||||||
|
, m_length( length )
|
||||||
|
{}
|
||||||
|
|
||||||
|
QString
|
||||||
|
MoveFileSystemJob::prettyName() const
|
||||||
|
{
|
||||||
|
return tr( "Move file system of partition %1." ).arg( partition()->partitionPath() );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult
|
||||||
|
MoveFileSystemJob::exec()
|
||||||
|
{
|
||||||
|
Report report( nullptr );
|
||||||
|
QString partitionPath = partition()->partitionPath();
|
||||||
|
CopySourceDevice moveSource( *m_device, m_oldFirstSector, m_oldFirstSector + m_length - 1 );
|
||||||
|
CopyTargetDevice moveTarget( *m_device, m_newFirstSector, m_newFirstSector + m_length - 1 );
|
||||||
|
|
||||||
|
if ( !moveSource.open() )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Could not open file system on partition %1 for moving." ).arg( partitionPath )
|
||||||
|
);
|
||||||
|
|
||||||
|
if ( !moveTarget.open() )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Could not create target for moving file system on partition %1." ).arg( partitionPath )
|
||||||
|
);
|
||||||
|
|
||||||
|
bool ok = copyBlocks( report, moveTarget, moveSource );
|
||||||
|
if ( !ok )
|
||||||
|
{
|
||||||
|
if ( rollbackCopyBlocks( report, moveTarget, moveSource ) )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Moving of partition %1 failed, changes have been rolled back." ).arg( partitionPath )
|
||||||
|
+ '\n' + report.toText()
|
||||||
|
);
|
||||||
|
else
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Moving of partition %1 failed. Roll back of the changes have failed." ).arg( partitionPath )
|
||||||
|
+ '\n' + report.toText()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
FileSystem& fs = partition()->fileSystem();
|
||||||
|
fs.setFirstSector( m_newFirstSector );
|
||||||
|
fs.setLastSector( m_newFirstSector + m_length - 1 );
|
||||||
|
|
||||||
|
if ( !fs.updateBootSector( report, partitionPath ) )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Updating boot sector after the moving of partition %1 failed." ).arg( partitionPath )
|
||||||
|
+ '\n' + report.toText()
|
||||||
|
);
|
||||||
|
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
MoveFileSystemJob::copyBlocks( Report& report, CopyTargetDevice& target, CopySourceDevice& source )
|
||||||
|
{
|
||||||
|
/** @todo copyBlocks() assumes that source.sectorSize() == target.sectorSize(). */
|
||||||
|
|
||||||
|
if ( source.sectorSize() != target.sectorSize() )
|
||||||
|
{
|
||||||
|
report.line() << tr( "The logical sector sizes in the source and target for copying are not the same. This is currently unsupported." );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rval = true;
|
||||||
|
const qint64 blockSize = 16065 * 8; // number of sectors per block to copy
|
||||||
|
const qint64 blocksToCopy = source.length() / blockSize;
|
||||||
|
|
||||||
|
qint64 readOffset = source.firstSector();
|
||||||
|
qint64 writeOffset = target.firstSector();
|
||||||
|
qint32 copyDir = 1;
|
||||||
|
|
||||||
|
if ( target.firstSector() > source.firstSector() )
|
||||||
|
{
|
||||||
|
readOffset = source.firstSector() + source.length() - blockSize;
|
||||||
|
writeOffset = target.firstSector() + source.length() - blockSize;
|
||||||
|
copyDir = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
qint64 blocksCopied = 0;
|
||||||
|
|
||||||
|
void* buffer = malloc( blockSize * source.sectorSize() );
|
||||||
|
int percent = 0;
|
||||||
|
|
||||||
|
while ( blocksCopied < blocksToCopy )
|
||||||
|
{
|
||||||
|
rval = source.readSectors( buffer, readOffset + blockSize * blocksCopied * copyDir, blockSize );
|
||||||
|
if ( !rval )
|
||||||
|
break;
|
||||||
|
|
||||||
|
rval = target.writeSectors( buffer, writeOffset + blockSize * blocksCopied * copyDir, blockSize );
|
||||||
|
if ( !rval )
|
||||||
|
break;
|
||||||
|
|
||||||
|
if ( ++blocksCopied * 100 / blocksToCopy != percent )
|
||||||
|
{
|
||||||
|
percent = blocksCopied * 100 / blocksToCopy;
|
||||||
|
progress( qreal( percent ) / 100. );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const qint64 lastBlock = source.length() % blockSize;
|
||||||
|
|
||||||
|
// copy the remainder
|
||||||
|
if ( rval && lastBlock > 0 )
|
||||||
|
{
|
||||||
|
Q_ASSERT( lastBlock < blockSize );
|
||||||
|
|
||||||
|
if ( lastBlock >= blockSize )
|
||||||
|
cLog() << "warning: lastBlock: " << lastBlock << ", blockSize: " << blockSize;
|
||||||
|
|
||||||
|
const qint64 lastBlockReadOffset = copyDir > 0 ? readOffset + blockSize * blocksCopied : source.firstSector();
|
||||||
|
const qint64 lastBlockWriteOffset = copyDir > 0 ? writeOffset + blockSize * blocksCopied : target.firstSector();
|
||||||
|
|
||||||
|
rval = source.readSectors( buffer, lastBlockReadOffset, lastBlock );
|
||||||
|
|
||||||
|
if ( rval )
|
||||||
|
rval = target.writeSectors( buffer, lastBlockWriteOffset, lastBlock );
|
||||||
|
|
||||||
|
if ( rval )
|
||||||
|
emit progress( 1.0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
free( buffer );
|
||||||
|
|
||||||
|
return rval;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
MoveFileSystemJob::rollbackCopyBlocks( Report& report, CopyTargetDevice& origTarget, CopySourceDevice& origSource )
|
||||||
|
{
|
||||||
|
if ( !origSource.overlaps( origTarget ) )
|
||||||
|
{
|
||||||
|
report.line() << tr( "Source and target for copying do not overlap: Rollback is not required." );
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// default: use values as if we were copying from front to back.
|
||||||
|
qint64 undoSourceFirstSector = origTarget.firstSector();
|
||||||
|
qint64 undoSourceLastSector = origTarget.firstSector() + origTarget.sectorsWritten() - 1;
|
||||||
|
|
||||||
|
qint64 undoTargetFirstSector = origSource.firstSector();
|
||||||
|
qint64 undoTargetLastSector = origSource.firstSector() + origTarget.sectorsWritten() - 1;
|
||||||
|
|
||||||
|
if ( origTarget.firstSector() > origSource.firstSector() )
|
||||||
|
{
|
||||||
|
// we were copying from back to front
|
||||||
|
undoSourceFirstSector = origTarget.firstSector() + origSource.length() - origTarget.sectorsWritten();
|
||||||
|
undoSourceLastSector = origTarget.firstSector() + origSource.length() - 1;
|
||||||
|
|
||||||
|
undoTargetFirstSector = origSource.lastSector() - origTarget.sectorsWritten() + 1;
|
||||||
|
undoTargetLastSector = origSource.lastSector();
|
||||||
|
}
|
||||||
|
|
||||||
|
CopySourceDevice undoSource( origTarget.device(), undoSourceFirstSector, undoSourceLastSector );
|
||||||
|
if ( !undoSource.open() )
|
||||||
|
{
|
||||||
|
report.line() << tr( "Could not open device %1 to rollback copying." )
|
||||||
|
.arg( origTarget.device().deviceNode() );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
CopyTargetDevice undoTarget( origSource.device(), undoTargetFirstSector, undoTargetLastSector );
|
||||||
|
if ( !undoTarget.open() )
|
||||||
|
{
|
||||||
|
report.line() << tr( "Could not open device %1 to rollback copying." )
|
||||||
|
.arg( origSource.device().deviceNode() );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return copyBlocks( report, undoTarget, undoSource );
|
||||||
|
}
|
69
src/modules/partition/MoveFileSystemJob.h
Normal file
69
src/modules/partition/MoveFileSystemJob.h
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// This class is heavily based on the MoveFileSystemJob class from KDE Partition
|
||||||
|
// Manager. Original copyright follow:
|
||||||
|
|
||||||
|
/***************************************************************************
|
||||||
|
* Copyright (C) 2008 by Volker Lanz <vl@fidra.de> *
|
||||||
|
* *
|
||||||
|
* This program 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 2 of the License, or *
|
||||||
|
* (at your option) any later version. *
|
||||||
|
* *
|
||||||
|
* This program 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 this program; if not, write to the *
|
||||||
|
* Free Software Foundation, Inc., *
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
|
||||||
|
***************************************************************************/
|
||||||
|
#ifndef MOVEFILESYSTEMJOB_H
|
||||||
|
#define MOVEFILESYSTEMJOB_H
|
||||||
|
|
||||||
|
#include <PartitionJob.h>
|
||||||
|
|
||||||
|
class CopySourceDevice;
|
||||||
|
class CopyTargetDevice;
|
||||||
|
class Device;
|
||||||
|
class Partition;
|
||||||
|
class Report;
|
||||||
|
|
||||||
|
class MoveFileSystemJob : public PartitionJob
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
MoveFileSystemJob( Device* device, Partition* partition, qint64 oldFirstSector, qint64 newFirstSector, qint64 length );
|
||||||
|
|
||||||
|
QString prettyName() const override;
|
||||||
|
|
||||||
|
Calamares::JobResult exec() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Device* m_device;
|
||||||
|
qint64 m_oldFirstSector;
|
||||||
|
qint64 m_newFirstSector;
|
||||||
|
qint64 m_length;
|
||||||
|
bool copyBlocks( Report& report, CopyTargetDevice& target, CopySourceDevice& source );
|
||||||
|
bool rollbackCopyBlocks( Report& report, CopyTargetDevice& origTarget, CopySourceDevice& origSource );
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* MOVEFILESYSTEMJOB_H */
|
@ -66,4 +66,21 @@ createNewPartition( PartitionNode* parent, const Device& device, const Partition
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Partition*
|
||||||
|
clonePartition( Device* device, Partition* partition )
|
||||||
|
{
|
||||||
|
FileSystem* fs = FileSystemFactory::create(
|
||||||
|
partition->fileSystem().type(),
|
||||||
|
partition->firstSector(),
|
||||||
|
partition->lastSector()
|
||||||
|
);
|
||||||
|
return new Partition(
|
||||||
|
partition->parent(),
|
||||||
|
*device,
|
||||||
|
partition->roles(),
|
||||||
|
fs, fs->firstSector(), fs->lastSector(),
|
||||||
|
partition->partitionPath()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
@ -40,6 +40,7 @@ Partition* findPartitionByMountPoint( const QList< Device* >& devices, const QSt
|
|||||||
|
|
||||||
Partition* createNewPartition( PartitionNode* parent, const Device& device, const PartitionRole& role, FileSystem::Type fsType, qint64 firstSector, qint64 lastSector );
|
Partition* createNewPartition( PartitionNode* parent, const Device& device, const PartitionRole& role, FileSystem::Type fsType, qint64 firstSector, qint64 lastSector );
|
||||||
|
|
||||||
|
Partition* clonePartition( Device* device, Partition* partition );
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* PMUTILS_H */
|
#endif /* PMUTILS_H */
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
#include <PartitionIterator.h>
|
#include <PartitionIterator.h>
|
||||||
#include <PartitionModel.h>
|
#include <PartitionModel.h>
|
||||||
#include <PMUtils.h>
|
#include <PMUtils.h>
|
||||||
|
#include <ResizePartitionJob.h>
|
||||||
#include <Typedefs.h>
|
#include <Typedefs.h>
|
||||||
#include <utils/Logger.h>
|
#include <utils/Logger.h>
|
||||||
|
|
||||||
@ -38,6 +39,7 @@
|
|||||||
#include <core/partition.h>
|
#include <core/partition.h>
|
||||||
#include <backend/corebackend.h>
|
#include <backend/corebackend.h>
|
||||||
#include <backend/corebackendmanager.h>
|
#include <backend/corebackendmanager.h>
|
||||||
|
#include <fs/filesystemfactory.h>
|
||||||
|
|
||||||
// Qt
|
// Qt
|
||||||
#include <QStandardItemModel>
|
#include <QStandardItemModel>
|
||||||
@ -89,10 +91,9 @@ PartitionCoreModule::PartitionCoreModule( QObject* parent )
|
|||||||
, m_deviceModel( new DeviceModel( this ) )
|
, m_deviceModel( new DeviceModel( this ) )
|
||||||
, m_bootLoaderModel( new BootLoaderModel( this ) )
|
, m_bootLoaderModel( new BootLoaderModel( this ) )
|
||||||
{
|
{
|
||||||
// FIXME: Should be done at startup
|
|
||||||
if ( !CalaPM::init() )
|
if ( !CalaPM::init() )
|
||||||
qFatal( "Failed to init CalaPM" );
|
qFatal( "Failed to init CalaPM" );
|
||||||
|
FileSystemFactory::init();
|
||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -257,6 +258,20 @@ PartitionCoreModule::formatPartition( Device* device, Partition* partition )
|
|||||||
refresh();
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
QList< Calamares::job_ptr >
|
QList< Calamares::job_ptr >
|
||||||
PartitionCoreModule::jobs() const
|
PartitionCoreModule::jobs() const
|
||||||
{
|
{
|
||||||
|
@ -69,6 +69,8 @@ public:
|
|||||||
|
|
||||||
void formatPartition( Device* device, Partition* partition );
|
void formatPartition( Device* device, Partition* partition );
|
||||||
|
|
||||||
|
void resizePartition( Device* device, Partition* partition, qint64 first, qint64 last );
|
||||||
|
|
||||||
void setBootLoaderInstallPath( const QString& path );
|
void setBootLoaderInstallPath( const QString& path );
|
||||||
|
|
||||||
QList< Calamares::job_ptr > jobs() const;
|
QList< Calamares::job_ptr > jobs() const;
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
*/
|
*/
|
||||||
#include <PartitionModel.h>
|
#include <PartitionModel.h>
|
||||||
|
|
||||||
|
#include <ColorUtils.h>
|
||||||
#include <PartitionInfo.h>
|
#include <PartitionInfo.h>
|
||||||
#include <PMUtils.h>
|
#include <PMUtils.h>
|
||||||
#include <utils/Logger.h>
|
#include <utils/Logger.h>
|
||||||
@ -33,26 +34,6 @@
|
|||||||
// Qt
|
// Qt
|
||||||
#include <QColor>
|
#include <QColor>
|
||||||
|
|
||||||
static QColor COLORS[ 4 ] =
|
|
||||||
{
|
|
||||||
"#448eca",
|
|
||||||
"#a5cc42",
|
|
||||||
"#d87e30",
|
|
||||||
"#ffbdbd",
|
|
||||||
};
|
|
||||||
static QColor FREE_SPACE_COLOR = "#777777";
|
|
||||||
static QColor EXTENDED_COLOR = "#aaaaaa";
|
|
||||||
|
|
||||||
static QColor colorForPartition( Partition* partition, int row )
|
|
||||||
{
|
|
||||||
if ( PMUtils::isPartitionFreeSpace( partition ) )
|
|
||||||
return FREE_SPACE_COLOR;
|
|
||||||
if ( partition->roles().has( PartitionRole::Extended ) )
|
|
||||||
return EXTENDED_COLOR;
|
|
||||||
// No partition-specific color needed, pick one from our list
|
|
||||||
return COLORS[ row % 4 ];
|
|
||||||
}
|
|
||||||
|
|
||||||
//- ResetHelper --------------------------------------------
|
//- ResetHelper --------------------------------------------
|
||||||
PartitionModel::ResetHelper::ResetHelper( PartitionModel* model )
|
PartitionModel::ResetHelper::ResetHelper( PartitionModel* model )
|
||||||
: m_model( model )
|
: m_model( model )
|
||||||
@ -170,7 +151,7 @@ PartitionModel::data( const QModelIndex& index, int role ) const
|
|||||||
}
|
}
|
||||||
case Qt::DecorationRole:
|
case Qt::DecorationRole:
|
||||||
if ( index.column() == NameColumn )
|
if ( index.column() == NameColumn )
|
||||||
return colorForPartition( partition, index.row() );
|
return ColorUtils::colorForPartition( partition );
|
||||||
else
|
else
|
||||||
return QVariant();
|
return QVariant();
|
||||||
case SizeRole:
|
case SizeRole:
|
||||||
|
123
src/modules/partition/PartitionSizeController.cpp
Normal file
123
src/modules/partition/PartitionSizeController.cpp
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <PartitionSizeController.h>
|
||||||
|
|
||||||
|
#include <ColorUtils.h>
|
||||||
|
|
||||||
|
// Qt
|
||||||
|
#include <QSpinBox>
|
||||||
|
|
||||||
|
// CalaPM
|
||||||
|
#include <core/device.h>
|
||||||
|
#include <core/partition.h>
|
||||||
|
#include <gui/partresizerwidget.h>
|
||||||
|
|
||||||
|
// stdc++
|
||||||
|
#include <limits>
|
||||||
|
|
||||||
|
PartitionSizeController::PartitionSizeController( QObject* parent )
|
||||||
|
: QObject( parent )
|
||||||
|
{}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::setPartResizerWidget( PartResizerWidget* widget )
|
||||||
|
{
|
||||||
|
if ( m_partResizerWidget )
|
||||||
|
disconnect( m_partResizerWidget, 0, this, 0 );
|
||||||
|
m_partResizerWidget = widget;
|
||||||
|
// FIXME: Should be set by PartResizerWidget itself
|
||||||
|
m_partResizerWidget->setFixedHeight( PartResizerWidget::handleHeight() );
|
||||||
|
|
||||||
|
QPalette pal = widget->palette();
|
||||||
|
pal.setColor( QPalette::Base, ColorUtils::freeSpaceColor() );
|
||||||
|
pal.setColor( QPalette::Button, m_partitionColor );
|
||||||
|
m_partResizerWidget->setPalette( pal );
|
||||||
|
updateConnections();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::setSpinBox( QSpinBox* spinBox )
|
||||||
|
{
|
||||||
|
if ( m_spinBox )
|
||||||
|
disconnect( m_spinBox, 0, this, 0 );
|
||||||
|
m_spinBox = spinBox;
|
||||||
|
m_spinBox->setMaximum( std::numeric_limits< int >::max() );
|
||||||
|
updateConnections();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::init( Device* device, Partition* partition, const QColor& color )
|
||||||
|
{
|
||||||
|
m_device = device;
|
||||||
|
m_partition = partition;
|
||||||
|
m_partitionColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::updateConnections()
|
||||||
|
{
|
||||||
|
if ( !m_spinBox || !m_partResizerWidget )
|
||||||
|
return;
|
||||||
|
|
||||||
|
connect( m_spinBox, SIGNAL( editingFinished() ), SLOT( updatePartResizerWidget() ) );
|
||||||
|
connect( m_partResizerWidget, SIGNAL( firstSectorChanged( qint64 ) ), SLOT( updateSpinBox() ) );
|
||||||
|
connect( m_partResizerWidget, SIGNAL( lastSectorChanged( qint64 ) ), SLOT( updateSpinBox() ) );
|
||||||
|
updateSpinBox();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::updatePartResizerWidget()
|
||||||
|
{
|
||||||
|
if ( m_updating )
|
||||||
|
return;
|
||||||
|
m_updating = true;
|
||||||
|
qint64 sectorSize = qint64( m_spinBox->value() ) * 1024 * 1024 / m_device->logicalSectorSize();
|
||||||
|
|
||||||
|
qint64 firstSector = m_partition->firstSector();
|
||||||
|
qint64 lastSector = firstSector + sectorSize - 1;
|
||||||
|
if ( lastSector > m_partResizerWidget->maximumLastSector() )
|
||||||
|
{
|
||||||
|
qint64 delta = lastSector - m_partResizerWidget->maximumLastSector();
|
||||||
|
firstSector -= delta;
|
||||||
|
lastSector -= delta;
|
||||||
|
}
|
||||||
|
m_partResizerWidget->updateLastSector( lastSector );
|
||||||
|
m_partResizerWidget->updateFirstSector( firstSector );
|
||||||
|
|
||||||
|
// Update spinbox value in case it was an impossible value
|
||||||
|
doUpdateSpinBox();
|
||||||
|
m_updating = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::updateSpinBox()
|
||||||
|
{
|
||||||
|
if ( m_updating )
|
||||||
|
return;
|
||||||
|
m_updating = true;
|
||||||
|
doUpdateSpinBox();
|
||||||
|
m_updating = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PartitionSizeController::doUpdateSpinBox()
|
||||||
|
{
|
||||||
|
qint64 mbSize = ( m_partition->lastSector() - m_partition->firstSector() + 1 ) * m_device->logicalSectorSize() / 1024 / 1024;
|
||||||
|
m_spinBox->setValue( mbSize );
|
||||||
|
}
|
60
src/modules/partition/PartitionSizeController.h
Normal file
60
src/modules/partition/PartitionSizeController.h
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PARTITIONSIZECONTROLLER_H
|
||||||
|
#define PARTITIONSIZECONTROLLER_H
|
||||||
|
|
||||||
|
#include <QColor>
|
||||||
|
#include <QObject>
|
||||||
|
#include <QPointer>
|
||||||
|
|
||||||
|
class QSpinBox;
|
||||||
|
|
||||||
|
class Device;
|
||||||
|
class Partition;
|
||||||
|
class PartResizerWidget;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Synchronize a PartResizerWidget and a QSpinBox
|
||||||
|
*/
|
||||||
|
class PartitionSizeController : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit PartitionSizeController( QObject* parent = nullptr );
|
||||||
|
void setPartResizerWidget( PartResizerWidget* widget );
|
||||||
|
void setSpinBox( QSpinBox* spinBox );
|
||||||
|
void init( Device* device, Partition* partition, const QColor& color );
|
||||||
|
|
||||||
|
private:
|
||||||
|
QPointer< PartResizerWidget > m_partResizerWidget;
|
||||||
|
QPointer< QSpinBox > m_spinBox;
|
||||||
|
Device* m_device = nullptr;
|
||||||
|
Partition* m_partition = nullptr;
|
||||||
|
QColor m_partitionColor;
|
||||||
|
bool m_updating = false;
|
||||||
|
|
||||||
|
void updateConnections();
|
||||||
|
void doUpdateSpinBox();
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void updatePartResizerWidget();
|
||||||
|
void updateSpinBox();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* PARTITIONSIZECONTROLLER_H */
|
@ -1,100 +0,0 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
|
||||||
*
|
|
||||||
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <PartitionSizeWidget.h>
|
|
||||||
|
|
||||||
#include <utils/Logger.h>
|
|
||||||
|
|
||||||
// CalaPM
|
|
||||||
#include <core/device.h>
|
|
||||||
#include <core/partition.h>
|
|
||||||
#include <core/partitiontable.h>
|
|
||||||
|
|
||||||
PartitionSizeWidget::PartitionSizeWidget( QWidget* parent )
|
|
||||||
: QSpinBox( parent )
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
PartitionSizeWidget::init( Device* device, Partition* partition )
|
|
||||||
{
|
|
||||||
m_device = device;
|
|
||||||
m_partition = partition;
|
|
||||||
Q_ASSERT( m_device->partitionTable() );
|
|
||||||
|
|
||||||
qint64 minSector = computeMinSector();
|
|
||||||
qint64 maxSector = computeMaxSector();
|
|
||||||
cLog() << minSector << maxSector;
|
|
||||||
setMaximum( mbSizeForSectorRange( minSector, maxSector ) );
|
|
||||||
|
|
||||||
m_initialValue = mbSizeForSectorRange( partition->firstSector(), partition->lastSector() );
|
|
||||||
setValue( m_initialValue );
|
|
||||||
}
|
|
||||||
|
|
||||||
PartitionSizeWidget::SectorRange
|
|
||||||
PartitionSizeWidget::sectorRange() const
|
|
||||||
{
|
|
||||||
qint64 minSector = computeMinSector();
|
|
||||||
qint64 maxSector = computeMaxSector();
|
|
||||||
|
|
||||||
int mbSize = value();
|
|
||||||
if ( mbSize == maximum() )
|
|
||||||
{
|
|
||||||
// If we are at the maximum value, select the last sector to avoid
|
|
||||||
// potential rounding errors which could leave a few sectors at the end
|
|
||||||
// unused
|
|
||||||
return SectorRange( minSector, maxSector );
|
|
||||||
}
|
|
||||||
|
|
||||||
qint64 lastSector = minSector + qint64( mbSize ) * 1024 * 1024 / m_device->logicalSectorSize();
|
|
||||||
Q_ASSERT( lastSector <= maxSector );
|
|
||||||
if ( lastSector > maxSector )
|
|
||||||
{
|
|
||||||
cLog() << "lastSector (" << lastSector << ") > maxSector (" << maxSector << "). This should not happen!";
|
|
||||||
lastSector = maxSector;
|
|
||||||
}
|
|
||||||
return SectorRange( minSector, lastSector );
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
PartitionSizeWidget::isDirty() const
|
|
||||||
{
|
|
||||||
return m_initialValue != value();
|
|
||||||
}
|
|
||||||
|
|
||||||
qint64
|
|
||||||
PartitionSizeWidget::mbSizeForSectorRange( qint64 first, qint64 last ) const
|
|
||||||
{
|
|
||||||
return ( last - first + 1 ) * m_device->logicalSectorSize() / 1024 / 1024;
|
|
||||||
}
|
|
||||||
|
|
||||||
qint64
|
|
||||||
PartitionSizeWidget::computeMaxSector() const
|
|
||||||
{
|
|
||||||
Q_ASSERT( m_device );
|
|
||||||
Q_ASSERT( m_partition );
|
|
||||||
return m_partition->lastSector() + m_device->partitionTable()->freeSectorsAfter( *m_partition );
|
|
||||||
}
|
|
||||||
|
|
||||||
qint64
|
|
||||||
PartitionSizeWidget::computeMinSector() const
|
|
||||||
{
|
|
||||||
Q_ASSERT( m_device );
|
|
||||||
Q_ASSERT( m_partition );
|
|
||||||
return m_partition->firstSector() - m_device->partitionTable()->freeSectorsBefore( *m_partition );
|
|
||||||
}
|
|
302
src/modules/partition/ResizePartitionJob.cpp
Normal file
302
src/modules/partition/ResizePartitionJob.cpp
Normal file
@ -0,0 +1,302 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// This class is heavily based on the ResizeOperation class from KDE Partition
|
||||||
|
// Manager. Original copyright follow:
|
||||||
|
|
||||||
|
/***************************************************************************
|
||||||
|
* Copyright (C) 2008,2012 by Volker Lanz <vl@fidra.de> *
|
||||||
|
* *
|
||||||
|
* This program 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 2 of the License, or *
|
||||||
|
* (at your option) any later version. *
|
||||||
|
* *
|
||||||
|
* This program 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 this program; if not, write to the *
|
||||||
|
* Free Software Foundation, Inc., *
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
|
||||||
|
***************************************************************************/
|
||||||
|
|
||||||
|
#include <ResizePartitionJob.h>
|
||||||
|
|
||||||
|
#include <CheckFileSystemJob.h>
|
||||||
|
#include <MoveFileSystemJob.h>
|
||||||
|
#include <utils/Logger.h>
|
||||||
|
|
||||||
|
// CalaPM
|
||||||
|
#include <backend/corebackend.h>
|
||||||
|
#include <backend/corebackendmanager.h>
|
||||||
|
#include <backend/corebackenddevice.h>
|
||||||
|
#include <backend/corebackendpartition.h>
|
||||||
|
#include <backend/corebackendpartitiontable.h>
|
||||||
|
#include <core/device.h>
|
||||||
|
#include <core/partition.h>
|
||||||
|
#include <util/report.h>
|
||||||
|
|
||||||
|
// Qt
|
||||||
|
#include <QScopedPointer>
|
||||||
|
|
||||||
|
//- Context --------------------------------------------------------------------
|
||||||
|
struct Context
|
||||||
|
{
|
||||||
|
Context( ResizePartitionJob* job_ )
|
||||||
|
: job( job_ )
|
||||||
|
{}
|
||||||
|
|
||||||
|
ResizePartitionJob* job;
|
||||||
|
qint64 oldFirstSector;
|
||||||
|
qint64 oldLastSector;
|
||||||
|
|
||||||
|
QScopedPointer< CoreBackendPartitionTable > backendPartitionTable;
|
||||||
|
};
|
||||||
|
|
||||||
|
//- ResizeFileSystemJob --------------------------------------------------------
|
||||||
|
class ResizeFileSystemJob : public Calamares::Job
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ResizeFileSystemJob( Context* context, qint64 length )
|
||||||
|
: m_context( context )
|
||||||
|
, m_length( length )
|
||||||
|
{}
|
||||||
|
|
||||||
|
QString prettyName() const override
|
||||||
|
{
|
||||||
|
QString path = m_context->job->partition()->partitionPath();
|
||||||
|
return tr( "Resize file system on partition %1." ).arg( path );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult exec() override
|
||||||
|
{
|
||||||
|
Report report( nullptr );
|
||||||
|
Device* device = m_context->job->device();
|
||||||
|
Partition* partition = m_context->job->partition();
|
||||||
|
FileSystem& fs = partition->fileSystem();
|
||||||
|
FileSystem::CommandSupportType support = m_length < fs.length() ? fs.supportShrink() : fs.supportGrow();
|
||||||
|
|
||||||
|
switch ( support )
|
||||||
|
{
|
||||||
|
case FileSystem::cmdSupportBackend:
|
||||||
|
if ( !backendResize( &report ) )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Parted failed to resize filesystem." ) + '\n' + report.toText()
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case FileSystem::cmdSupportFileSystem:
|
||||||
|
{
|
||||||
|
qint64 byteLength = device->logicalSectorSize() * m_length;
|
||||||
|
bool ok = fs.resize( report, partition->partitionPath(), byteLength );
|
||||||
|
if ( !ok )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Failed to resize filesystem." ) + '\n' + report.toText()
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.setLastSector( fs.firstSector() + m_length - 1 );
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Context* m_context;
|
||||||
|
qint64 m_length;
|
||||||
|
|
||||||
|
bool backendResize( Report* report )
|
||||||
|
{
|
||||||
|
Partition* partition = m_context->job->partition();
|
||||||
|
bool ok = m_context->backendPartitionTable->resizeFileSystem( *report, *partition, m_length );
|
||||||
|
if ( !ok )
|
||||||
|
return false;
|
||||||
|
m_context->backendPartitionTable->commit();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
//- SetPartGeometryJob ---------------------------------------------------------
|
||||||
|
class SetPartGeometryJob : public Calamares::Job
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
SetPartGeometryJob( Context* context, qint64 firstSector, qint64 length )
|
||||||
|
: m_context( context )
|
||||||
|
, m_firstSector( firstSector )
|
||||||
|
, m_length( length )
|
||||||
|
{}
|
||||||
|
|
||||||
|
QString prettyName() const override
|
||||||
|
{
|
||||||
|
QString path = m_context->job->partition()->partitionPath();
|
||||||
|
return tr( "Update geometry of partition %1." ).arg( path );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult exec() override
|
||||||
|
{
|
||||||
|
Report report( nullptr );
|
||||||
|
Partition* partition = m_context->job->partition();
|
||||||
|
qint64 lastSector = m_firstSector + m_length - 1;
|
||||||
|
bool ok = m_context->backendPartitionTable->updateGeometry( report, *partition, m_firstSector, lastSector );
|
||||||
|
if ( !ok )
|
||||||
|
{
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
QString(),
|
||||||
|
tr( "Failed to change the geometry of the partition." ) + '\n' + report.toText() );
|
||||||
|
}
|
||||||
|
partition->setFirstSector( m_firstSector );
|
||||||
|
partition->setLastSector( lastSector );
|
||||||
|
m_context->backendPartitionTable->commit();
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Context* m_context;
|
||||||
|
qint64 m_firstSector;
|
||||||
|
qint64 m_length;
|
||||||
|
};
|
||||||
|
|
||||||
|
//- ResizePartitionJob ---------------------------------------------------------
|
||||||
|
ResizePartitionJob::ResizePartitionJob( Device* device, Partition* partition, qint64 firstSector, qint64 lastSector )
|
||||||
|
: PartitionJob( partition )
|
||||||
|
, m_device( device )
|
||||||
|
, m_oldFirstSector( partition->firstSector() ) // Keep a copy of old sectors because they will be overwritten in updatePreview()
|
||||||
|
, m_oldLastSector( partition->lastSector() )
|
||||||
|
, m_newFirstSector( firstSector )
|
||||||
|
, m_newLastSector( lastSector )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
QString
|
||||||
|
ResizePartitionJob::prettyName() const
|
||||||
|
{
|
||||||
|
// FIXME: Copy PM ResizeOperation code which generates a description of the
|
||||||
|
// operation
|
||||||
|
return tr( "Resize partition %1." ).arg( partition()->partitionPath() );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult
|
||||||
|
ResizePartitionJob::exec()
|
||||||
|
{
|
||||||
|
qint64 oldLength = m_oldLastSector - m_oldFirstSector + 1;
|
||||||
|
qint64 newLength = m_newLastSector - m_newFirstSector + 1;
|
||||||
|
|
||||||
|
// Assuming updatePreview() has been called, `partition` uses its new
|
||||||
|
// position and size. Reset it to the old values: part of the libparted
|
||||||
|
// backend relies on this (for example:
|
||||||
|
// LibPartedPartitionTable::updateGeometry())
|
||||||
|
// The jobs are responsible for updating the partition back when they are
|
||||||
|
// done.
|
||||||
|
m_partition->setFirstSector( m_oldFirstSector );
|
||||||
|
m_partition->setLastSector( m_oldLastSector );
|
||||||
|
|
||||||
|
// Setup context
|
||||||
|
QString partitionPath = m_partition->partitionPath();
|
||||||
|
Context context( this );
|
||||||
|
context.oldFirstSector = m_oldFirstSector;
|
||||||
|
context.oldLastSector = m_oldLastSector;
|
||||||
|
|
||||||
|
CoreBackend* backend = CoreBackendManager::self()->backend();
|
||||||
|
QScopedPointer<CoreBackendDevice> backendDevice( backend->openDevice( m_device->deviceNode() ) );
|
||||||
|
if ( !backendDevice.data() )
|
||||||
|
{
|
||||||
|
QString errorMessage = tr( "The installer failed to resize partition %1 on disk '%2'." )
|
||||||
|
.arg( m_partition->partitionPath() )
|
||||||
|
.arg( m_device->name() );
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
errorMessage,
|
||||||
|
tr( "Could not open device '%1'." ).arg( m_device->deviceNode() )
|
||||||
|
);
|
||||||
|
}
|
||||||
|
context.backendPartitionTable.reset( backendDevice->openPartitionTable() );
|
||||||
|
|
||||||
|
// Create jobs
|
||||||
|
QList< Calamares::job_ptr > jobs;
|
||||||
|
jobs << Calamares::job_ptr( new CheckFileSystemJob( partition() ) );
|
||||||
|
if ( m_partition->roles().has( PartitionRole::Extended ) )
|
||||||
|
jobs << Calamares::job_ptr( new SetPartGeometryJob( &context, m_newFirstSector, newLength ) );
|
||||||
|
else
|
||||||
|
{
|
||||||
|
bool shrink = newLength < oldLength;
|
||||||
|
bool grow = newLength > oldLength;
|
||||||
|
bool moveRight = m_newFirstSector > m_oldFirstSector;
|
||||||
|
bool moveLeft = m_newFirstSector < m_oldFirstSector;
|
||||||
|
if ( shrink )
|
||||||
|
{
|
||||||
|
jobs << Calamares::job_ptr( new ResizeFileSystemJob( &context, newLength ) );
|
||||||
|
jobs << Calamares::job_ptr( new SetPartGeometryJob( &context, m_oldFirstSector, newLength ) );
|
||||||
|
}
|
||||||
|
if ( moveRight || moveLeft )
|
||||||
|
{
|
||||||
|
// At this point, we need to set the partition's length to either the resized length, if it has already been
|
||||||
|
// shrunk, or to the original length (it may or may not then later be grown, we don't care here)
|
||||||
|
const qint64 length = shrink ? newLength : oldLength;
|
||||||
|
jobs << Calamares::job_ptr( new SetPartGeometryJob( &context, m_newFirstSector, length ) );
|
||||||
|
jobs << Calamares::job_ptr( new MoveFileSystemJob( m_device, m_partition, m_oldFirstSector, m_newFirstSector, length ) );
|
||||||
|
}
|
||||||
|
if ( grow )
|
||||||
|
{
|
||||||
|
jobs << Calamares::job_ptr( new SetPartGeometryJob( &context, m_newFirstSector, newLength ) );
|
||||||
|
jobs << Calamares::job_ptr( new ResizeFileSystemJob( &context, newLength ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
jobs << Calamares::job_ptr( new CheckFileSystemJob( partition() ) );
|
||||||
|
return execJobList( jobs );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ResizePartitionJob::updatePreview()
|
||||||
|
{
|
||||||
|
m_device->partitionTable()->removeUnallocated();
|
||||||
|
m_partition->parent()->remove( m_partition );
|
||||||
|
m_partition->setFirstSector( m_newFirstSector );
|
||||||
|
m_partition->setLastSector( m_newLastSector );
|
||||||
|
m_partition->parent()->insert( m_partition );
|
||||||
|
m_device->partitionTable()->updateUnallocated( *m_device );
|
||||||
|
}
|
||||||
|
|
||||||
|
Calamares::JobResult
|
||||||
|
ResizePartitionJob::execJobList( const QList< Calamares::job_ptr >& jobs )
|
||||||
|
{
|
||||||
|
QString errorMessage = tr( "The installer failed to resize partition %1 on disk '%2'." )
|
||||||
|
.arg( m_partition->partitionPath() )
|
||||||
|
.arg( m_device->name() );
|
||||||
|
|
||||||
|
int nbJobs = jobs.size();
|
||||||
|
int count = 0;
|
||||||
|
for ( Calamares::job_ptr job : jobs )
|
||||||
|
{
|
||||||
|
cLog() << "- " + job->prettyName();
|
||||||
|
Calamares::JobResult result = job->exec();
|
||||||
|
if ( !result )
|
||||||
|
{
|
||||||
|
if ( result.message().isEmpty() )
|
||||||
|
result.setMessage( errorMessage );
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
++count;
|
||||||
|
progress( qreal( count ) / nbJobs );
|
||||||
|
}
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
57
src/modules/partition/ResizePartitionJob.h
Normal file
57
src/modules/partition/ResizePartitionJob.h
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014, Aurélien Gâteau <agateau@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef RESIZEPARTITIONJOB_H
|
||||||
|
#define RESIZEPARTITIONJOB_H
|
||||||
|
|
||||||
|
#include <PartitionJob.h>
|
||||||
|
|
||||||
|
class Device;
|
||||||
|
class Partition;
|
||||||
|
class FileSystem;
|
||||||
|
|
||||||
|
struct Context;
|
||||||
|
|
||||||
|
class ResizePartitionJob : public PartitionJob
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
ResizePartitionJob( Device* device, Partition* partition, qint64 firstSector, qint64 lastSector );
|
||||||
|
QString prettyName() const override;
|
||||||
|
Calamares::JobResult exec() override;
|
||||||
|
|
||||||
|
void updatePreview();
|
||||||
|
|
||||||
|
Device* device() const
|
||||||
|
{
|
||||||
|
return m_device;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Device* m_device;
|
||||||
|
qint64 m_oldFirstSector;
|
||||||
|
qint64 m_oldLastSector;
|
||||||
|
qint64 m_newFirstSector;
|
||||||
|
qint64 m_newLastSector;
|
||||||
|
|
||||||
|
Calamares::JobResult execJobList( const QList< Calamares::job_ptr >& jobs );
|
||||||
|
|
||||||
|
friend struct Context;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* RESIZEPARTITIONJOB_H */
|
@ -1 +1 @@
|
|||||||
Subproject commit f958c7be734135f22454e464518c1f7110298a95
|
Subproject commit 8726fc2b3f2a703b9a72cf76038be1bab22d8a3f
|
@ -2,13 +2,16 @@ find_package( Qt5 COMPONENTS Test REQUIRED )
|
|||||||
include( ECMAddTests )
|
include( ECMAddTests )
|
||||||
|
|
||||||
set( jobtests_SRCS
|
set( jobtests_SRCS
|
||||||
|
${PartitionModule_SOURCE_DIR}/CheckFileSystemJob.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/CreatePartitionJob.cpp
|
${PartitionModule_SOURCE_DIR}/CreatePartitionJob.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/CreatePartitionTableJob.cpp
|
${PartitionModule_SOURCE_DIR}/CreatePartitionTableJob.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/DeletePartitionJob.cpp
|
${PartitionModule_SOURCE_DIR}/DeletePartitionJob.cpp
|
||||||
|
${PartitionModule_SOURCE_DIR}/MoveFileSystemJob.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/PartitionInfo.cpp
|
${PartitionModule_SOURCE_DIR}/PartitionInfo.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/PartitionIterator.cpp
|
${PartitionModule_SOURCE_DIR}/PartitionIterator.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/PartitionJob.cpp
|
${PartitionModule_SOURCE_DIR}/PartitionJob.cpp
|
||||||
${PartitionModule_SOURCE_DIR}/PMUtils.cpp
|
${PartitionModule_SOURCE_DIR}/PMUtils.cpp
|
||||||
|
${PartitionModule_SOURCE_DIR}/ResizePartitionJob.cpp
|
||||||
JobTests.cpp
|
JobTests.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
#include <CreatePartitionJob.h>
|
#include <CreatePartitionJob.h>
|
||||||
#include <CreatePartitionTableJob.h>
|
#include <CreatePartitionTableJob.h>
|
||||||
|
#include <ResizePartitionJob.h>
|
||||||
#include <PMUtils.h>
|
#include <PMUtils.h>
|
||||||
|
|
||||||
// CalaPM
|
// CalaPM
|
||||||
@ -12,6 +13,7 @@
|
|||||||
|
|
||||||
// Qt
|
// Qt
|
||||||
#include <QEventLoop>
|
#include <QEventLoop>
|
||||||
|
#include <QProcess>
|
||||||
#include <QtTest/QtTest>
|
#include <QtTest/QtTest>
|
||||||
|
|
||||||
QTEST_GUILESS_MAIN( JobTests )
|
QTEST_GUILESS_MAIN( JobTests )
|
||||||
@ -20,8 +22,77 @@ static const qint64 MB = 1024 * 1024;
|
|||||||
|
|
||||||
using namespace Calamares;
|
using namespace Calamares;
|
||||||
|
|
||||||
static
|
class PartitionMounter
|
||||||
Partition* firstFreePartition( PartitionNode* parent )
|
{
|
||||||
|
public:
|
||||||
|
PartitionMounter( const QString& devicePath )
|
||||||
|
: m_mountPointDir( "calamares-partitiontests-mountpoint" )
|
||||||
|
{
|
||||||
|
QStringList args = QStringList() << devicePath << m_mountPointDir.path();
|
||||||
|
int ret = QProcess::execute( "mount", args );
|
||||||
|
m_mounted = ret == 0;
|
||||||
|
QCOMPARE( ret, 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
~PartitionMounter()
|
||||||
|
{
|
||||||
|
if ( !m_mounted )
|
||||||
|
return;
|
||||||
|
int ret = QProcess::execute( "umount", QStringList() << m_mountPointDir.path() );
|
||||||
|
QCOMPARE( ret, 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
QString mountPoint() const
|
||||||
|
{
|
||||||
|
return m_mounted ? m_mountPointDir.path() : QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString m_devicePath;
|
||||||
|
QTemporaryDir m_mountPointDir;
|
||||||
|
bool m_mounted;
|
||||||
|
};
|
||||||
|
|
||||||
|
static QByteArray
|
||||||
|
generateTestData( qint64 size )
|
||||||
|
{
|
||||||
|
QByteArray ba;
|
||||||
|
ba.resize( size );
|
||||||
|
// Fill the array explicitly to keep Valgrind happy
|
||||||
|
for ( auto it = ba.data() ; it < ba.data() + size ; ++it )
|
||||||
|
{
|
||||||
|
*it = rand() % 256;
|
||||||
|
}
|
||||||
|
return ba;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
writeFile( const QString& path, const QByteArray data )
|
||||||
|
{
|
||||||
|
QFile file( path );
|
||||||
|
QVERIFY( file.open( QIODevice::WriteOnly ) );
|
||||||
|
|
||||||
|
const char* ptr = data.constData();
|
||||||
|
const char* end = data.constData() + data.size();
|
||||||
|
const qint64 chunkSize = 16384;
|
||||||
|
|
||||||
|
while ( ptr < end )
|
||||||
|
{
|
||||||
|
qint64 count = file.write( ptr, chunkSize );
|
||||||
|
if ( count < 0 )
|
||||||
|
{
|
||||||
|
QString msg = QString( "Writing file failed. Only %1 bytes written out of %2. Error: '%3'." )
|
||||||
|
.arg( ptr - data.constData() )
|
||||||
|
.arg( data.size() )
|
||||||
|
.arg( file.errorString() );
|
||||||
|
QFAIL( qPrintable( msg ) );
|
||||||
|
}
|
||||||
|
ptr += count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static Partition*
|
||||||
|
firstFreePartition( PartitionNode* parent )
|
||||||
{
|
{
|
||||||
for( auto child : parent->children() )
|
for( auto child : parent->children() )
|
||||||
if ( PMUtils::isPartitionFreeSpace( child ) )
|
if ( PMUtils::isPartitionFreeSpace( child ) )
|
||||||
@ -29,6 +100,7 @@ Partition* firstFreePartition( PartitionNode* parent )
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//- QueueRunner ---------------------------------------------------------------
|
||||||
QueueRunner::QueueRunner( JobQueue* queue )
|
QueueRunner::QueueRunner( JobQueue* queue )
|
||||||
: m_queue( queue )
|
: m_queue( queue )
|
||||||
{
|
{
|
||||||
@ -62,7 +134,7 @@ QueueRunner::onFailed( const QString& message, const QString& details )
|
|||||||
QFAIL( qPrintable( msg ) );
|
QFAIL( qPrintable( msg ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------
|
//- JobTests ------------------------------------------------------------------
|
||||||
JobTests::JobTests()
|
JobTests::JobTests()
|
||||||
: m_runner( &m_queue )
|
: m_runner( &m_queue )
|
||||||
{}
|
{}
|
||||||
@ -77,12 +149,18 @@ JobTests::initTestCase()
|
|||||||
}
|
}
|
||||||
|
|
||||||
QVERIFY( CalaPM::init() );
|
QVERIFY( CalaPM::init() );
|
||||||
|
FileSystemFactory::init();
|
||||||
|
|
||||||
|
refreshDevice();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
JobTests::refreshDevice()
|
||||||
|
{
|
||||||
|
QString devicePath = qgetenv( "CALAMARES_TEST_DISK" );
|
||||||
CoreBackend* backend = CoreBackendManager::self()->backend();
|
CoreBackend* backend = CoreBackendManager::self()->backend();
|
||||||
m_device.reset( backend->scanDevice( devicePath ) );
|
m_device.reset( backend->scanDevice( devicePath ) );
|
||||||
QVERIFY( !m_device.isNull() );
|
QVERIFY( !m_device.isNull() );
|
||||||
|
|
||||||
FileSystemFactory::init();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -216,3 +294,101 @@ JobTests::testCreatePartitionExtended()
|
|||||||
QCOMPARE( extendedPartition->partitionPath(), devicePath + "2" );
|
QCOMPARE( extendedPartition->partitionPath(), devicePath + "2" );
|
||||||
QCOMPARE( partition2->partitionPath(), devicePath + "5" );
|
QCOMPARE( partition2->partitionPath(), devicePath + "5" );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
JobTests::testResizePartition_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn< int >( "oldStartMB" );
|
||||||
|
QTest::addColumn< int >( "oldSizeMB" );
|
||||||
|
QTest::addColumn< int >( "newStartMB" );
|
||||||
|
QTest::addColumn< int >( "newSizeMB" );
|
||||||
|
|
||||||
|
QTest::newRow("grow") << 10 << 50 << 10 << 70;
|
||||||
|
QTest::newRow("shrink") << 10 << 70 << 10 << 50;
|
||||||
|
QTest::newRow("moveLeft") << 10 << 50 << 8 << 50;
|
||||||
|
QTest::newRow("moveRight") << 10 << 50 << 12 << 50;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
JobTests::testResizePartition()
|
||||||
|
{
|
||||||
|
QFETCH( int, oldStartMB );
|
||||||
|
QFETCH( int, oldSizeMB );
|
||||||
|
QFETCH( int, newStartMB );
|
||||||
|
QFETCH( int, newSizeMB );
|
||||||
|
|
||||||
|
const qint64 sectorForMB = MB / m_device->logicalSectorSize();
|
||||||
|
|
||||||
|
qint64 oldFirst = sectorForMB * oldStartMB;
|
||||||
|
qint64 oldLast = oldFirst + sectorForMB * oldSizeMB - 1;
|
||||||
|
qint64 newFirst = sectorForMB * newStartMB;
|
||||||
|
qint64 newLast = newFirst + sectorForMB * newSizeMB - 1;
|
||||||
|
|
||||||
|
// Make the test data file smaller than the full size of the partition to
|
||||||
|
// accomodate for the file system overhead
|
||||||
|
const QByteArray testData = generateTestData( ( qMin( oldSizeMB, newSizeMB ) ) * MB * 3 / 4 );
|
||||||
|
const QString testName = "test.data";
|
||||||
|
|
||||||
|
// Setup: create the test partition
|
||||||
|
{
|
||||||
|
queuePartitionTableCreation( PartitionTable::msdos );
|
||||||
|
|
||||||
|
Partition* freePartition = firstFreePartition( m_device->partitionTable() );
|
||||||
|
QVERIFY( freePartition );
|
||||||
|
Partition* partition = PMUtils::createNewPartition( freePartition->parent(), *m_device, PartitionRole( PartitionRole::Primary ), FileSystem::Ext4, oldFirst, oldLast );
|
||||||
|
CreatePartitionJob* job = new CreatePartitionJob( m_device.data(), partition );
|
||||||
|
job->updatePreview();
|
||||||
|
m_queue.enqueue( job_ptr( job ) );
|
||||||
|
|
||||||
|
QVERIFY( m_runner.run() );
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// Write a test file in the partition
|
||||||
|
refreshDevice();
|
||||||
|
QVERIFY( m_device->partitionTable() );
|
||||||
|
Partition* partition = m_device->partitionTable()->findPartitionBySector( oldFirst, PartitionRole( PartitionRole::Primary ) );
|
||||||
|
QVERIFY( partition );
|
||||||
|
QCOMPARE( partition->firstSector(), oldFirst );
|
||||||
|
QCOMPARE( partition->lastSector(), oldLast );
|
||||||
|
{
|
||||||
|
PartitionMounter mounter( partition->partitionPath() );
|
||||||
|
QString mountPoint = mounter.mountPoint();
|
||||||
|
QVERIFY( !mountPoint.isEmpty() );
|
||||||
|
writeFile( mountPoint + '/' + testName, testData );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resize
|
||||||
|
ResizePartitionJob* job = new ResizePartitionJob( m_device.data(), partition, newFirst, newLast );
|
||||||
|
job->updatePreview();
|
||||||
|
m_queue.enqueue( job_ptr( job ) );
|
||||||
|
QVERIFY( m_runner.run() );
|
||||||
|
|
||||||
|
QCOMPARE( partition->firstSector(), newFirst );
|
||||||
|
QCOMPARE( partition->lastSector(), newLast );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test
|
||||||
|
{
|
||||||
|
refreshDevice();
|
||||||
|
QVERIFY( m_device->partitionTable() );
|
||||||
|
Partition* partition = m_device->partitionTable()->findPartitionBySector( newFirst, PartitionRole( PartitionRole::Primary ) );
|
||||||
|
QVERIFY( partition );
|
||||||
|
QCOMPARE( partition->firstSector(), newFirst );
|
||||||
|
QCOMPARE( partition->lastSector(), newLast );
|
||||||
|
QCOMPARE( partition->fileSystem().firstSector(), newFirst );
|
||||||
|
QCOMPARE( partition->fileSystem().lastSector(), newLast );
|
||||||
|
|
||||||
|
|
||||||
|
PartitionMounter mounter( partition->partitionPath() );
|
||||||
|
QString mountPoint = mounter.mountPoint();
|
||||||
|
QVERIFY( !mountPoint.isEmpty() );
|
||||||
|
{
|
||||||
|
QFile file( mountPoint + '/' + testName );
|
||||||
|
QVERIFY( file.open( QIODevice::ReadOnly ) );
|
||||||
|
QByteArray outData = file.readAll();
|
||||||
|
QCOMPARE( outData.size(), testData.size() );
|
||||||
|
QCOMPARE( outData, testData );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -43,6 +43,8 @@ private Q_SLOTS:
|
|||||||
void testPartitionTable();
|
void testPartitionTable();
|
||||||
void testCreatePartition();
|
void testCreatePartition();
|
||||||
void testCreatePartitionExtended();
|
void testCreatePartitionExtended();
|
||||||
|
void testResizePartition_data();
|
||||||
|
void testResizePartition();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QScopedPointer< Device > m_device;
|
QScopedPointer< Device > m_device;
|
||||||
@ -51,6 +53,7 @@ private:
|
|||||||
|
|
||||||
void queuePartitionTableCreation( PartitionTable::TableType type );
|
void queuePartitionTableCreation( PartitionTable::TableType type );
|
||||||
CreatePartitionJob* newCreatePartitionJob( Partition* freeSpacePartition, PartitionRole, FileSystem::Type type, qint64 size );
|
CreatePartitionJob* newCreatePartitionJob( Partition* freeSpacePartition, PartitionRole, FileSystem::Type type, qint64 size );
|
||||||
|
void refreshDevice();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* JOBTESTS_H */
|
#endif /* JOBTESTS_H */
|
||||||
|
Loading…
Reference in New Issue
Block a user