kpmcore/src/fs/lvm2_pv.cpp

305 lines
10 KiB
C++
Raw Normal View History

2020-09-28 00:45:21 +01:00
/*
SPDX-FileCopyrightText: 2012-2018 Andrius Štikonas <andrius@stikonas.eu>
SPDX-FileCopyrightText: 2016 Chantara Tith <tith.chantara@gmail.com>
SPDX-FileCopyrightText: 2016 Teo Mrnjavac <teo@kde.org>
SPDX-FileCopyrightText: 2018 Caio Jordão Carvalho <caiojcarvalho@gmail.com>
2020-09-28 00:45:21 +01:00
SPDX-FileCopyrightText: 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/lvm2_pv.h"
#include "core/device.h"
#include "util/externalcommand.h"
#include "util/capacity.h"
#include <QString>
#include <KLocalizedString>
namespace FS
{
2015-07-13 15:16:36 +01:00
FileSystem::CommandSupportType lvm2_pv::m_GetUsed = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_GetLabel = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Create = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Grow = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Shrink = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Move = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Check = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Copy = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_Backup = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_SetLabel = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_UpdateUUID = FileSystem::cmdSupportNone;
FileSystem::CommandSupportType lvm2_pv::m_GetUUID = FileSystem::cmdSupportNone;
lvm2_pv::lvm2_pv(qint64 firstsector, qint64 lastsector,
qint64 sectorsused, const QString& label, const QVariantMap& features)
: FileSystem(firstsector, lastsector, sectorsused, label, features, FileSystem::Type::Lvm2_PV)
, m_PESize(0)
, m_TotalPE(0)
, m_AllocatedPE(0)
2015-07-13 15:16:36 +01:00
{
}
void lvm2_pv::init()
{
2021-02-27 12:00:00 +00:00
CommandSupportType lvmFound = findExternal(QStringLiteral("lvm"), {}, 3) ? cmdSupportFileSystem : cmdSupportNone;
2015-07-13 15:16:36 +01:00
m_Create = lvmFound;
m_Check = lvmFound;
m_Grow = lvmFound;
m_Shrink = lvmFound;
m_UpdateUUID = lvmFound;
m_GetUsed = lvmFound;
2015-07-13 15:16:36 +01:00
m_Move = (m_Check != cmdSupportNone) ? cmdSupportCore : cmdSupportNone;
m_GetLabel = cmdSupportCore;
m_Backup = cmdSupportCore;
m_GetUUID = cmdSupportCore;
2015-07-13 15:16:36 +01:00
m_GetLabel = cmdSupportNone;
m_Copy = cmdSupportNone; // Copying PV can confuse LVM
2015-07-13 15:16:36 +01:00
}
void lvm2_pv::scan(const QString& deviceNode)
{
getPESize(deviceNode);
2016-09-08 01:42:04 +01:00
m_AllocatedPE = getAllocatedPE(deviceNode);
m_TotalPE = getTotalPE(deviceNode);
}
2015-07-13 15:16:36 +01:00
bool lvm2_pv::supportToolFound() const
{
return
m_GetUsed != cmdSupportNone &&
2015-07-13 15:16:36 +01:00
m_Create != cmdSupportNone &&
m_Check != cmdSupportNone &&
m_UpdateUUID != cmdSupportNone &&
2016-05-24 21:13:23 +01:00
m_Grow != cmdSupportNone &&
m_Shrink != cmdSupportNone &&
2015-07-13 15:16:36 +01:00
m_Move != cmdSupportNone &&
m_Backup != cmdSupportNone &&
m_GetUUID != cmdSupportNone;
}
FileSystem::SupportTool lvm2_pv::supportToolName() const
{
2019-11-22 13:34:10 +00:00
return SupportTool(QStringLiteral("lvm2"), QUrl(QStringLiteral("https://sourceware.org/lvm2/")));
2015-07-13 15:16:36 +01:00
}
qint64 lvm2_pv::maxCapacity() const
{
2018-04-09 15:14:34 +01:00
return Capacity::unitFactor(Capacity::Unit::Byte, Capacity::Unit::EiB);
2015-07-13 15:16:36 +01:00
}
qint64 lvm2_pv::readUsedCapacity(const QString& deviceNode) const
{
QString pvUsed = getpvField(QStringLiteral("pv_used"), deviceNode);
2016-08-06 22:15:50 +01:00
QString metadataOffset = getpvField(QStringLiteral("pe_start"), deviceNode);
return pvUsed.isEmpty() ? -1 : pvUsed.toLongLong() + metadataOffset.toLongLong();
}
2015-07-13 15:16:36 +01:00
bool lvm2_pv::check(Report& report, const QString& deviceNode) const
{
ExternalCommand cmd(report, QStringLiteral("lvm"), { QStringLiteral("pvck"), QStringLiteral("--verbose"), deviceNode });
2015-07-13 15:16:36 +01:00
return cmd.run(-1) && cmd.exitCode() == 0;
}
2016-09-05 12:10:56 +01:00
bool lvm2_pv::create(Report& report, const QString& deviceNode)
2015-07-13 15:16:36 +01:00
{
ExternalCommand cmd(report, QStringLiteral("lvm"), { QStringLiteral("pvcreate"), QStringLiteral("--force"), deviceNode });
2015-07-13 15:16:36 +01:00
return cmd.run(-1) && cmd.exitCode() == 0;
}
bool lvm2_pv::remove(Report& report, const QString& deviceNode) const
{
ExternalCommand cmd(report, QStringLiteral("lvm"), { QStringLiteral("pvremove"), QStringLiteral("--force"), QStringLiteral("--force"), QStringLiteral("--yes"), deviceNode });
2015-07-13 15:16:36 +01:00
return cmd.run(-1) && cmd.exitCode() == 0;
}
2016-05-24 21:13:23 +01:00
bool lvm2_pv::resize(Report& report, const QString& deviceNode, qint64 length) const
{
2016-08-06 13:36:57 +01:00
bool rval = true;
2016-08-06 22:15:50 +01:00
qint64 metadataOffset = getpvField(QStringLiteral("pe_start"), deviceNode).toLongLong();
qint64 lastPE = getTotalPE(deviceNode) - 1; // starts from 0
2016-08-06 13:36:57 +01:00
if (lastPE > 0) { // make sure that the PV is already in a VG
qint64 targetPE = (length - metadataOffset) / peSize() - 1; // starts from 0
2016-08-06 13:36:57 +01:00
if (targetPE < lastPE) { //shrinking FS
2016-08-06 22:15:50 +01:00
qint64 firstMovedPE = qMax(targetPE + 1, getAllocatedPE(deviceNode)); // starts from 1
2016-08-06 13:36:57 +01:00
ExternalCommand moveCmd(report,
QStringLiteral("lvm"), {
QStringLiteral("pvmove"),
QStringLiteral("--alloc"),
QStringLiteral("anywhere"),
2016-08-06 22:15:50 +01:00
deviceNode + QStringLiteral(":") + QString::number(firstMovedPE) + QStringLiteral("-") + QString::number(lastPE),
deviceNode + QStringLiteral(":") + QStringLiteral("0-") + QString::number(firstMovedPE - 1)
2016-08-06 13:36:57 +01:00
});
rval = moveCmd.run(-1) && (moveCmd.exitCode() == 0 || moveCmd.exitCode() == 5); // FIXME: exit code 5: NO data to move
}
}
ExternalCommand cmd(report, QStringLiteral("lvm"), {
QStringLiteral("pvresize"),
QStringLiteral("--yes"),
QStringLiteral("--setphysicalvolumesize"),
QString::number(length) + QStringLiteral("B"),
deviceNode });
return rval && cmd.run(-1) && cmd.exitCode() == 0;
2016-05-24 21:13:23 +01:00
}
bool lvm2_pv::resizeOnline(Report& report, const QString& deviceNode, const QString& mountPoint, qint64 length) const
{
Q_UNUSED(mountPoint)
return resize(report, deviceNode, length);
}
2015-07-13 15:16:36 +01:00
bool lvm2_pv::updateUUID(Report& report, const QString& deviceNode) const
{
ExternalCommand cmd(report, QStringLiteral("lvm"), { QStringLiteral("pvchange"), QStringLiteral("--uuid"), deviceNode });
2015-07-13 15:16:36 +01:00
return cmd.run(-1) && cmd.exitCode() == 0;
}
QString lvm2_pv::readUUID(const QString& deviceNode) const
{
return getpvField(QStringLiteral("pv_uuid"), deviceNode);
}
bool lvm2_pv::mount(Report& report, const QString& deviceNode, const QString& mountPoint)
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(mountPoint)
return false;
}
2016-05-23 11:29:33 +01:00
bool lvm2_pv::unmount(Report& report, const QString& deviceNode)
2016-05-23 11:29:33 +01:00
{
Q_UNUSED(deviceNode)
Q_UNUSED(report)
return false;
}
bool lvm2_pv::canMount(const QString& deviceNode, const QString& mountPoint) const
{
Q_UNUSED(deviceNode)
Q_UNUSED(mountPoint)
return false;
2016-05-23 11:29:33 +01:00
}
bool lvm2_pv::canUnmount(const QString& deviceNode) const
{
Q_UNUSED(deviceNode)
return false;
}
2016-05-25 18:03:03 +01:00
2016-07-08 10:19:33 +01:00
qint64 lvm2_pv::getTotalPE(const QString& deviceNode)
{
QString pvPeCount = getpvField(QStringLiteral("pv_pe_count"), deviceNode);
return pvPeCount.isEmpty() ? -1 : pvPeCount.toLongLong();
2016-05-25 18:03:03 +01:00
}
2016-07-08 10:19:33 +01:00
qint64 lvm2_pv::getAllocatedPE(const QString& deviceNode)
{
QString pvPeAllocCount = getpvField(QStringLiteral("pv_pe_alloc_count"), deviceNode);
return pvPeAllocCount.isEmpty() ? -1 : pvPeAllocCount.toLongLong();
}
void lvm2_pv::getPESize(const QString& deviceNode)
{
QString vgExtentSize = getpvField(QStringLiteral("vg_extent_size"), deviceNode);
m_PESize = vgExtentSize.isEmpty() ? -1 : vgExtentSize.toLongLong();
}
2016-08-15 07:18:40 +01:00
/** Get pvs command output with field name
*
* @param fieldName LVM field name
* @param deviceNode path to PV
* @return raw output of pvs command, usually with many spaces
*/
QString lvm2_pv::getpvField(const QString& fieldName, const QString& deviceNode)
{
2016-07-03 12:09:56 +01:00
QStringList args = { QStringLiteral("pvs"),
QStringLiteral("--foreign"),
QStringLiteral("--readonly"),
QStringLiteral("--noheadings"),
QStringLiteral("--units"),
QStringLiteral("B"),
QStringLiteral("--nosuffix"),
QStringLiteral("--options"),
2016-08-15 07:18:40 +01:00
fieldName };
2016-07-03 12:09:56 +01:00
if (!deviceNode.isEmpty()) {
args << deviceNode;
}
ExternalCommand cmd(QStringLiteral("lvm"), args, QProcess::ProcessChannelMode::SeparateChannels);
2016-05-25 18:03:03 +01:00
if (cmd.run(-1) && cmd.exitCode() == 0) {
return cmd.output().trimmed();
2016-05-25 18:03:03 +01:00
}
return QString();
}
QString lvm2_pv::getVGName(const QString& deviceNode)
{
return getpvField(QStringLiteral("vg_name"), deviceNode);
2016-05-25 18:03:03 +01:00
}
QList<LvmPV> lvm2_pv::getPVinNode(const PartitionNode* parent)
2016-07-03 12:09:56 +01:00
{
QList<LvmPV> partitions;
if (parent == nullptr)
return partitions;
2016-07-03 12:09:56 +01:00
for (const auto &node : parent->children()) {
const Partition* p = dynamic_cast<const Partition*>(node);
if (p == nullptr)
continue;
if (node->children().size() > 0)
partitions.append(getPVinNode(node));
// FIXME: reenable newly created PVs (before applying) once everything works
2018-04-07 19:54:30 +01:00
if(p->fileSystem().type() == FileSystem::Type::Lvm2_PV && p->deviceNode() == p->partitionPath())
partitions.append(LvmPV(p->mountPoint(), p));
2018-04-07 19:54:30 +01:00
if(p->fileSystem().type() == FileSystem::Type::Luks && p->deviceNode() == p->partitionPath())
partitions.append(LvmPV(p->mountPoint(), p, true));
2016-07-03 12:09:56 +01:00
}
return partitions;
2016-07-03 12:09:56 +01:00
}
2016-09-08 01:42:04 +01:00
/** construct a list of Partition objects for LVM PVs that are either unused or belong to some VG.
*
* @param devices list of Devices which we scan for LVM PVs
* @return list of LVM PVs
*/
QList<LvmPV> lvm2_pv::getPVs(const QList<Device*>& devices)
2016-07-03 12:09:56 +01:00
{
QList<LvmPV> partitions;
for (auto const &d : devices)
partitions.append(getPVinNode(d->partitionTable()));
return partitions;
2016-07-03 12:09:56 +01:00
}
}
namespace LVM {
QList<LvmPV> pvList::m_list;
}
LvmPV::LvmPV(const QString vgName, const Partition* p, bool isLuks)
: m_vgName(vgName)
, m_p(p)
, m_isLuks(isLuks)
{
}