kpmcore/src/fs/btrfs.cpp

225 lines
8.7 KiB
C++

/*
SPDX-FileCopyrightText: 2010 Volker Lanz <vl@fidra.de>
SPDX-FileCopyrightText: 2012-2018 Andrius Štikonas <andrius@stikonas.eu>
SPDX-FileCopyrightText: 2012,2019 Yuri Chornoivan <yurchor@ukr.net>
SPDX-FileCopyrightText: 2020 Arnaud Ferraris <arnaud.ferraris@collabora.com>
SPDX-FileCopyrightText: 2020 Gaël PORTAY <gael.portay@collabora.com>
SPDX-License-Identifier: GPL-3.0-or-later
*/
#include "fs/btrfs.h"
#include "util/externalcommand.h"
#include "util/capacity.h"
#include "util/report.h"
#include <QRegularExpression>
#include <QString>
#include <QTemporaryDir>
#include <KLocalizedString>
namespace FS
{
FileSystem::CommandSupportType btrfs::m_GetUsed = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_GetLabel = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Create = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Grow = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Shrink = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Move = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Check = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Copy = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_Backup = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_SetLabel = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_UpdateUUID = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType btrfs::m_GetUUID = FileSystem::cmdSupportNone;
btrfs::btrfs(qint64 firstsector, qint64 lastsector, qint64 sectorsused, const QString& label, const QVariantMap& features) :
FileSystem(firstsector, lastsector, sectorsused, label, features, FileSystem::Type::Btrfs)
{
}
void btrfs::init()
{
m_Create = findExternal(QStringLiteral("mkfs.btrfs")) ? cmdSupportFileSystem : cmdSupportNone;
m_Check = findExternal(QStringLiteral("btrfs")) ? cmdSupportFileSystem : cmdSupportNone;
m_Grow = m_Check;
m_GetUsed = m_Check;
m_Shrink = (m_Grow != cmdSupportNone && m_GetUsed != cmdSupportNone) ? cmdSupportFileSystem : cmdSupportNone;
m_SetLabel = m_Check;
m_UpdateUUID = findExternal(QStringLiteral("btrfstune")) ? cmdSupportFileSystem : cmdSupportNone;
m_Copy = (m_Check != cmdSupportNone) ? cmdSupportCore : cmdSupportNone;
m_Move = (m_Check != cmdSupportNone) ? cmdSupportCore : cmdSupportNone;
m_GetLabel = cmdSupportCore;
m_Backup = cmdSupportCore;
m_GetUUID = cmdSupportCore;
if (m_Create == cmdSupportFileSystem) {
ExternalCommand cmd(QStringLiteral("mkfs.btrfs"), QStringList() << QStringLiteral("-O") << QStringLiteral("list-all"));
if (cmd.run(-1) && cmd.exitCode() == 0) {
QStringList lines = cmd.output().split(QStringLiteral("\n"));
// First line is introductory text, we don't need it
lines.removeFirst();
for (const auto& l: lines) {
if (!l.isEmpty())
addAvailableFeature(l.split(QStringLiteral(" ")).first());
}
}
}
}
bool btrfs::supportToolFound() const
{
return
m_GetUsed != cmdSupportNone &&
m_GetLabel != cmdSupportNone &&
m_SetLabel != cmdSupportNone &&
m_Create != cmdSupportNone &&
m_Check != cmdSupportNone &&
// m_UpdateUUID != cmdSupportNone &&
m_Grow != cmdSupportNone &&
m_Shrink != cmdSupportNone &&
m_Copy != cmdSupportNone &&
m_Move != cmdSupportNone &&
m_Backup != cmdSupportNone &&
m_GetUUID != cmdSupportNone;
}
FileSystem::SupportTool btrfs::supportToolName() const
{
return SupportTool(QStringLiteral("btrfs-tools"), QUrl(QStringLiteral("https://btrfs.wiki.kernel.org/")));
}
qint64 btrfs::minCapacity() const
{
return 256 * Capacity::unitFactor(Capacity::Unit::Byte, Capacity::Unit::MiB);
}
qint64 btrfs::maxCapacity() const
{
return Capacity::unitFactor(Capacity::Unit::Byte, Capacity::Unit::EiB);
}
int btrfs::maxLabelLength() const
{
return 255;
}
qint64 btrfs::readUsedCapacity(const QString& deviceNode) const
{
ExternalCommand cmd(QStringLiteral("btrfs"),
{ QStringLiteral("filesystem"), QStringLiteral("show"), QStringLiteral("--raw"), deviceNode });
if (cmd.run(-1) && cmd.exitCode() == 0) {
QRegularExpression re(QStringLiteral(" used (\\d+) path ") + deviceNode);
QRegularExpressionMatch reBytesUsed = re.match(cmd.output());
if (reBytesUsed.hasMatch())
return reBytesUsed.captured(1).toLongLong();
}
return -1;
}
bool btrfs::check(Report& report, const QString& deviceNode) const
{
ExternalCommand cmd(report, QStringLiteral("btrfs"), { QStringLiteral("check"), QStringLiteral("--repair"), deviceNode });
return cmd.run(-1) && cmd.exitCode() == 0;
}
bool btrfs::create(Report& report, const QString& deviceNode)
{
QStringList args = QStringList();
if (!this->features().isEmpty()) {
QStringList feature_list = QStringList();
for (const auto& k : this->features().keys()) {
const auto& v = this->features().value(k);
if (v.type() == QVariant::Type::Bool) {
if (v.toBool())
feature_list << k;
else
feature_list << (QStringLiteral("^") + k);
} else {
qWarning() << "Ignoring feature" << k << "of type" << v.type() << "; requires type QVariant::bool.";
}
}
args << QStringLiteral("--features") << feature_list.join(QStringLiteral(","));
}
args << QStringLiteral("--force") << deviceNode;
ExternalCommand cmd(report, QStringLiteral("mkfs.btrfs"), args);
return cmd.run(-1) && cmd.exitCode() == 0;
}
bool btrfs::resize(Report& report, const QString& deviceNode, qint64 length) const
{
QTemporaryDir tempDir;
if (!tempDir.isValid()) {
report.line() << xi18nc("@info:progress", "Resizing Btrfs file system on partition <filename>%1</filename> failed: Could not create temp dir.", deviceNode);
return false;
}
bool rval = false;
ExternalCommand mountCmd(report, QStringLiteral("mount"),
{ QStringLiteral("--verbose"), QStringLiteral("--types"), QStringLiteral("btrfs"), deviceNode, tempDir.path() });
if (mountCmd.run(-1) && mountCmd.exitCode() == 0) {
ExternalCommand resizeCmd(report, QStringLiteral("btrfs"),
{ QStringLiteral("filesystem"), QStringLiteral("resize"), QString::number(length), tempDir.path() });
if (resizeCmd.run(-1) && resizeCmd.exitCode() == 0)
rval = true;
else
report.line() << xi18nc("@info:progress", "Resizing Btrfs file system on partition <filename>%1</filename> failed: btrfs file system resize failed.", deviceNode);
ExternalCommand unmountCmd(report, QStringLiteral("umount"), { tempDir.path() });
if (!unmountCmd.run(-1) && unmountCmd.exitCode() == 0)
report.line() << xi18nc("@info:progress", "<warning>Resizing Btrfs file system on partition <filename>%1</filename>: Unmount failed.</warning>", deviceNode);
} else
report.line() << xi18nc("@info:progress", "Resizing Btrfs file system on partition <filename>%1</filename> failed: Initial mount failed.", deviceNode);
return rval;
}
bool btrfs::resizeOnline(Report& report, const QString& deviceNode, const QString& mountPoint, qint64 length) const
{
ExternalCommand resizeCmd(report, QStringLiteral("btrfs"),
{ QStringLiteral("filesystem"), QStringLiteral("resize"), QString::number(length), mountPoint });
if (resizeCmd.run(-1) && resizeCmd.exitCode() == 0)
return true;
report.line() << xi18nc("@info:progress", "Resizing Btrfs file system on partition <filename>%1</filename> failed: btrfs file system resize failed.", deviceNode);
return false;
}
bool btrfs::writeLabel(Report& report, const QString& deviceNode, const QString& newLabel)
{
ExternalCommand cmd(report, QStringLiteral("btrfs"), { QStringLiteral("filesystem"), QStringLiteral("label"), deviceNode, newLabel });
return cmd.run(-1) && cmd.exitCode() == 0;
}
bool btrfs::writeLabelOnline(Report& report, const QString& deviceNode, const QString& mountPoint, const QString& newLabel)
{
Q_UNUSED(deviceNode)
ExternalCommand cmd(report, QStringLiteral("btrfs"), { QStringLiteral("filesystem"), QStringLiteral("label"), mountPoint, newLabel });
return cmd.run(-1) && cmd.exitCode() == 0;
}
bool btrfs::updateUUID(Report& report, const QString& deviceNode) const
{
ExternalCommand cmd(report, QStringLiteral("btrfstune"), { QStringLiteral("-f"), QStringLiteral("-u"), deviceNode });
return cmd.run(-1) && cmd.exitCode() == 0;
}
}