2020-08-25 16:05:56 +02:00
|
|
|
/* === This file is part of Calamares - <https://calamares.io> ===
|
2020-04-17 00:24:32 +02:00
|
|
|
*
|
2020-07-30 10:26:58 +02:00
|
|
|
* SPDX-FileCopyrightText: 2020 Adriaan de Groot <groot@kde.org>
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
2020-04-17 00:24:32 +02:00
|
|
|
*
|
2020-08-25 16:05:56 +02:00
|
|
|
* Calamares is Free Software: see the License-Identifier above.
|
2020-04-17 00:24:32 +02:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Config.h"
|
2020-05-18 13:07:12 +02:00
|
|
|
|
|
|
|
#include "GlobalStorage.h"
|
|
|
|
#include "JobQueue.h"
|
2020-05-18 14:03:31 +02:00
|
|
|
#include "utils/Logger.h"
|
2020-05-18 13:07:12 +02:00
|
|
|
#include "utils/Variant.h"
|
|
|
|
|
|
|
|
Config::Config( QObject* parent )
|
|
|
|
: QObject( parent )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-10-02 12:08:42 +02:00
|
|
|
const NamedEnumTable< Config::InstallChoice >&
|
|
|
|
Config::installChoiceNames()
|
|
|
|
{
|
|
|
|
static const NamedEnumTable< InstallChoice > names { { QStringLiteral( "none" ), InstallChoice::NoChoice },
|
|
|
|
{ QStringLiteral( "nochoice" ), InstallChoice::NoChoice },
|
|
|
|
{ QStringLiteral( "alongside" ), InstallChoice::Alongside },
|
|
|
|
{ QStringLiteral( "erase" ), InstallChoice::Erase },
|
|
|
|
{ QStringLiteral( "replace" ), InstallChoice::Replace },
|
|
|
|
{ QStringLiteral( "manual" ), InstallChoice::Manual } };
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
2020-10-02 12:22:53 +02:00
|
|
|
const NamedEnumTable< Config::SwapChoice >&
|
|
|
|
Config::swapChoiceNames()
|
|
|
|
{
|
|
|
|
static const NamedEnumTable< SwapChoice > names { { QStringLiteral( "none" ), SwapChoice::NoSwap },
|
|
|
|
{ QStringLiteral( "small" ), SwapChoice::SmallSwap },
|
|
|
|
{ QStringLiteral( "suspend" ), SwapChoice::FullSwap },
|
|
|
|
{ QStringLiteral( "reuse" ), SwapChoice::ReuseSwap },
|
|
|
|
{ QStringLiteral( "file" ), SwapChoice::SwapFile } };
|
|
|
|
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
|
|
|
Config::SwapChoice
|
|
|
|
pickOne( const Config::SwapChoiceSet& s )
|
|
|
|
{
|
|
|
|
if ( s.count() == 0 )
|
|
|
|
{
|
|
|
|
return Config::SwapChoice::NoSwap;
|
|
|
|
}
|
|
|
|
if ( s.count() == 1 )
|
|
|
|
{
|
|
|
|
return *( s.begin() );
|
|
|
|
}
|
|
|
|
if ( s.contains( Config::SwapChoice::NoSwap ) )
|
|
|
|
{
|
|
|
|
return Config::SwapChoice::NoSwap;
|
|
|
|
}
|
|
|
|
// Here, count > 1 but NoSwap is not a member.
|
|
|
|
return *( s.begin() );
|
|
|
|
}
|
|
|
|
|
2020-10-02 12:08:42 +02:00
|
|
|
|
2020-10-02 12:22:53 +02:00
|
|
|
static Config::SwapChoiceSet
|
2020-05-18 14:03:31 +02:00
|
|
|
getSwapChoices( const QVariantMap& configurationMap )
|
|
|
|
{
|
|
|
|
// SWAP SETTINGS
|
|
|
|
//
|
|
|
|
// This is a bit convoluted because there's legacy settings to handle as well
|
|
|
|
// as the new-style list of choices, with mapping back-and-forth.
|
|
|
|
if ( configurationMap.contains( "userSwapChoices" )
|
|
|
|
&& ( configurationMap.contains( "ensureSuspendToDisk" ) || configurationMap.contains( "neverCreateSwap" ) ) )
|
|
|
|
{
|
|
|
|
cError() << "Partition-module configuration mixes old- and new-style swap settings.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( configurationMap.contains( "ensureSuspendToDisk" ) )
|
|
|
|
{
|
|
|
|
cWarning() << "Partition-module setting *ensureSuspendToDisk* is deprecated.";
|
|
|
|
}
|
|
|
|
bool ensureSuspendToDisk = CalamaresUtils::getBool( configurationMap, "ensureSuspendToDisk", true );
|
|
|
|
|
|
|
|
if ( configurationMap.contains( "neverCreateSwap" ) )
|
|
|
|
{
|
|
|
|
cWarning() << "Partition-module setting *neverCreateSwap* is deprecated.";
|
|
|
|
}
|
|
|
|
bool neverCreateSwap = CalamaresUtils::getBool( configurationMap, "neverCreateSwap", false );
|
|
|
|
|
2020-10-02 12:22:53 +02:00
|
|
|
Config::SwapChoiceSet choices; // Available swap choices
|
2020-05-18 14:03:31 +02:00
|
|
|
if ( configurationMap.contains( "userSwapChoices" ) )
|
|
|
|
{
|
|
|
|
// We've already warned about overlapping settings with the
|
|
|
|
// legacy *ensureSuspendToDisk* and *neverCreateSwap*.
|
|
|
|
QStringList l = configurationMap[ "userSwapChoices" ].toStringList();
|
|
|
|
|
|
|
|
for ( const auto& item : l )
|
|
|
|
{
|
|
|
|
bool ok = false;
|
2020-10-02 12:22:53 +02:00
|
|
|
auto v = Config::swapChoiceNames().find( item, ok );
|
2020-05-18 14:03:31 +02:00
|
|
|
if ( ok )
|
|
|
|
{
|
|
|
|
choices.insert( v );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( choices.isEmpty() )
|
|
|
|
{
|
|
|
|
cWarning() << "Partition-module configuration for *userSwapChoices* is empty:" << l;
|
2020-10-02 12:22:53 +02:00
|
|
|
choices.insert( Config::SwapChoice::FullSwap );
|
2020-05-18 14:03:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// suspend if it's one of the possible choices; suppress swap only if it's
|
|
|
|
// the **only** choice available.
|
2020-10-02 12:22:53 +02:00
|
|
|
ensureSuspendToDisk = choices.contains( Config::SwapChoice::FullSwap );
|
|
|
|
neverCreateSwap = ( choices.count() == 1 ) && choices.contains( Config::SwapChoice::NoSwap );
|
2020-05-18 14:03:31 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Convert the legacy settings into a single setting for now.
|
|
|
|
if ( neverCreateSwap )
|
|
|
|
{
|
2020-10-02 12:22:53 +02:00
|
|
|
choices.insert( Config::SwapChoice::NoSwap );
|
2020-05-18 14:03:31 +02:00
|
|
|
}
|
|
|
|
else if ( ensureSuspendToDisk )
|
|
|
|
{
|
2020-10-02 12:22:53 +02:00
|
|
|
choices.insert( Config::SwapChoice::FullSwap );
|
2020-05-18 14:03:31 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-02 12:22:53 +02:00
|
|
|
choices.insert( Config::SwapChoice::SmallSwap );
|
2020-05-18 14:03:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not all are supported right now // FIXME
|
|
|
|
static const char unsupportedSetting[] = "Partition-module does not support *userSwapChoices* setting";
|
|
|
|
|
|
|
|
#define COMPLAIN_UNSUPPORTED( x ) \
|
|
|
|
if ( choices.contains( x ) ) \
|
|
|
|
{ \
|
2020-07-30 10:51:48 +02:00
|
|
|
bool bogus = false; \
|
2020-10-02 12:22:53 +02:00
|
|
|
cWarning() << unsupportedSetting << Config::swapChoiceNames().find( x, bogus ); \
|
2020-05-18 14:03:31 +02:00
|
|
|
choices.remove( x ); \
|
|
|
|
}
|
|
|
|
|
2020-10-02 12:22:53 +02:00
|
|
|
COMPLAIN_UNSUPPORTED( Config::SwapChoice::ReuseSwap )
|
2020-05-18 14:03:31 +02:00
|
|
|
#undef COMPLAIN_UNSUPPORTED
|
|
|
|
|
|
|
|
return choices;
|
|
|
|
}
|
|
|
|
|
2020-10-06 15:54:26 +02:00
|
|
|
void
|
|
|
|
updateGlobalStorage( Config::InstallChoice installChoice, Config::SwapChoice swapChoice )
|
|
|
|
{
|
|
|
|
auto* gs = Calamares::JobQueue::instance() ? Calamares::JobQueue::instance()->globalStorage() : nullptr;
|
|
|
|
if ( gs )
|
|
|
|
{
|
|
|
|
QVariantMap m;
|
|
|
|
m.insert( "install", Config::installChoiceNames().find( installChoice ) );
|
|
|
|
m.insert( "swap", Config::swapChoiceNames().find( swapChoice ) );
|
|
|
|
gs->insert( "partitionChoices", m );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-29 14:00:49 +02:00
|
|
|
void
|
|
|
|
Config::setInstallChoice( int c )
|
|
|
|
{
|
2020-10-02 12:08:42 +02:00
|
|
|
if ( ( c < InstallChoice::NoChoice ) || ( c > InstallChoice::Manual ) )
|
2020-09-29 14:00:49 +02:00
|
|
|
{
|
|
|
|
cWarning() << "Invalid install choice (int)" << c;
|
2020-10-02 12:08:42 +02:00
|
|
|
c = InstallChoice::NoChoice;
|
2020-09-29 14:00:49 +02:00
|
|
|
}
|
2020-10-02 12:08:42 +02:00
|
|
|
setInstallChoice( static_cast< InstallChoice >( c ) );
|
2020-09-29 14:00:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-10-02 12:08:42 +02:00
|
|
|
Config::setInstallChoice( InstallChoice c )
|
2020-09-29 14:00:49 +02:00
|
|
|
{
|
|
|
|
if ( c != m_installChoice )
|
|
|
|
{
|
|
|
|
m_installChoice = c;
|
|
|
|
emit installChoiceChanged( c );
|
2020-10-06 15:54:26 +02:00
|
|
|
::updateGlobalStorage( c, m_swapChoice );
|
2020-09-29 14:00:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-02 12:40:13 +02:00
|
|
|
void
|
|
|
|
Config::setSwapChoice( int c )
|
|
|
|
{
|
|
|
|
if ( ( c < SwapChoice::NoSwap ) || ( c > SwapChoice::SwapFile ) )
|
|
|
|
{
|
2020-11-11 18:44:41 +01:00
|
|
|
cWarning() << "Invalid swap choice (int)" << c;
|
2020-10-02 12:40:13 +02:00
|
|
|
c = SwapChoice::NoSwap;
|
|
|
|
}
|
|
|
|
setSwapChoice( static_cast< SwapChoice >( c ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Config::setSwapChoice( Config::SwapChoice c )
|
|
|
|
{
|
|
|
|
if ( c != m_swapChoice )
|
|
|
|
{
|
|
|
|
m_swapChoice = c;
|
|
|
|
emit swapChoiceChanged( c );
|
2020-10-06 15:54:26 +02:00
|
|
|
::updateGlobalStorage( m_installChoice, c );
|
2020-10-02 12:40:13 +02:00
|
|
|
}
|
|
|
|
}
|
2020-09-29 14:00:49 +02:00
|
|
|
|
2021-03-31 18:15:02 +02:00
|
|
|
void
|
2021-06-18 11:23:27 +02:00
|
|
|
Config::setEraseFsTypeChoice( const QString& choice )
|
2021-03-31 18:15:02 +02:00
|
|
|
{
|
2021-06-18 11:23:27 +02:00
|
|
|
if ( choice != m_eraseFsTypeChoice )
|
|
|
|
{
|
2021-06-18 11:29:36 +02:00
|
|
|
// FIXME: shouldn't this be a canonical fs name?
|
2021-03-31 18:15:02 +02:00
|
|
|
m_eraseFsTypeChoice = choice;
|
2021-06-18 11:29:36 +02:00
|
|
|
Q_EMIT eraseModeFilesystemChanged( choice );
|
2021-03-31 18:15:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-02 13:04:12 +02:00
|
|
|
bool
|
|
|
|
Config::allowManualPartitioning() const
|
|
|
|
{
|
|
|
|
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
|
|
|
|
return gs->value( "allowManualPartitioning" ).toBool();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-18 13:07:12 +02:00
|
|
|
void
|
|
|
|
Config::setConfigurationMap( const QVariantMap& configurationMap )
|
|
|
|
{
|
|
|
|
// Settings that overlap with the Welcome module
|
|
|
|
m_requiredStorageGiB = CalamaresUtils::getDouble( configurationMap, "requiredStorage", -1.0 );
|
2020-05-18 14:03:31 +02:00
|
|
|
m_swapChoices = getSwapChoices( configurationMap );
|
2020-07-30 11:36:59 +02:00
|
|
|
|
|
|
|
bool nameFound = false; // In the name table (ignored, falls back to first entry in table)
|
2020-10-02 12:22:53 +02:00
|
|
|
m_initialInstallChoice = installChoiceNames().find(
|
2020-08-22 01:19:58 +02:00
|
|
|
CalamaresUtils::getString( configurationMap, "initialPartitioningChoice" ), nameFound );
|
2020-09-29 14:00:49 +02:00
|
|
|
setInstallChoice( m_initialInstallChoice );
|
|
|
|
|
2020-10-02 12:22:53 +02:00
|
|
|
m_initialSwapChoice
|
|
|
|
= swapChoiceNames().find( CalamaresUtils::getString( configurationMap, "initialSwapChoice" ), nameFound );
|
2020-09-28 15:32:47 +02:00
|
|
|
if ( !m_swapChoices.contains( m_initialSwapChoice ) )
|
|
|
|
{
|
|
|
|
cWarning() << "Configuration for *initialSwapChoice* is not one of the *userSwapChoices*";
|
2020-10-02 12:22:53 +02:00
|
|
|
m_initialSwapChoice = pickOne( m_swapChoices );
|
2020-09-28 15:32:47 +02:00
|
|
|
}
|
2020-10-02 12:40:13 +02:00
|
|
|
setSwapChoice( m_initialSwapChoice );
|
2020-10-02 13:04:12 +02:00
|
|
|
|
|
|
|
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
|
|
|
|
gs->insert( "allowManualPartitioning",
|
|
|
|
CalamaresUtils::getBool( configurationMap, "allowManualPartitioning", true ) );
|
2020-05-21 23:13:28 +02:00
|
|
|
|
2021-06-18 11:23:27 +02:00
|
|
|
if ( configurationMap.contains( "availableFileSystemTypes" ) )
|
|
|
|
{
|
|
|
|
if ( configurationMap.value( "availableFileSystemTypes" ).type() == QVariant::List )
|
|
|
|
{
|
2021-03-31 18:15:02 +02:00
|
|
|
m_eraseFsTypes.clear();
|
2021-06-18 11:23:27 +02:00
|
|
|
m_eraseFsTypes.append( configurationMap.value( "availableFileSystemTypes" ).toStringList() );
|
|
|
|
m_eraseFsTypeChoice = m_eraseFsTypes[ 0 ];
|
|
|
|
}
|
|
|
|
else if ( configurationMap.value( "availableFileSystemTypes" ).type() == QVariant::String )
|
|
|
|
{
|
|
|
|
m_eraseFsTypes.append( configurationMap.value( "availableFileSystemTypes" ).toString() );
|
|
|
|
m_eraseFsTypeChoice = m_eraseFsTypes[ 0 ];
|
2021-03-31 18:15:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-12 14:27:01 +02:00
|
|
|
if ( configurationMap.contains( "requiredPartitionTableType" )
|
|
|
|
&& configurationMap.value( "requiredPartitionTableType" ).type() == QVariant::List )
|
2020-05-21 23:13:28 +02:00
|
|
|
{
|
|
|
|
m_requiredPartitionTableType.clear();
|
|
|
|
m_requiredPartitionTableType.append( configurationMap.value( "requiredPartitionTableType" ).toStringList() );
|
|
|
|
}
|
2020-10-12 14:27:01 +02:00
|
|
|
else if ( configurationMap.contains( "requiredPartitionTableType" )
|
|
|
|
&& configurationMap.value( "requiredPartitionTableType" ).type() == QVariant::String )
|
2020-05-21 23:13:28 +02:00
|
|
|
{
|
|
|
|
m_requiredPartitionTableType.clear();
|
|
|
|
m_requiredPartitionTableType.append( configurationMap.value( "requiredPartitionTableType" ).toString() );
|
|
|
|
}
|
2020-10-12 14:27:01 +02:00
|
|
|
gs->insert( "requiredPartitionTableType", m_requiredPartitionTableType );
|
2020-05-18 13:07:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Config::updateGlobalStorage() const
|
|
|
|
{
|
|
|
|
// If there's no setting (e.g. from the welcome page) for required storage
|
|
|
|
// then use ours, if it was set.
|
|
|
|
auto* gs = Calamares::JobQueue::instance() ? Calamares::JobQueue::instance()->globalStorage() : nullptr;
|
|
|
|
if ( m_requiredStorageGiB >= 0.0 && gs && !gs->contains( "requiredStorageGiB" ) )
|
|
|
|
{
|
|
|
|
gs->insert( "requiredStorageGiB", m_requiredStorageGiB );
|
|
|
|
}
|
|
|
|
}
|