2015-06-04 01:29:22 +01:00
|
|
|
/*************************************************************************
|
|
|
|
* Copyright (C) 2012 by Volker Lanz <vl@fidra.de> *
|
2016-02-25 23:29:33 +00:00
|
|
|
* Copyright (C) 2016 by Andrius Štikonas <andrius@stikonas.eu> *
|
2015-06-04 01:29:22 +01:00
|
|
|
* *
|
|
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.*
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
#include "fs/lvm2_pv.h"
|
2016-09-10 22:08:21 +01:00
|
|
|
#include "core/device.h"
|
2015-06-04 01:29:22 +01:00
|
|
|
|
|
|
|
#include "util/externalcommand.h"
|
|
|
|
#include "util/capacity.h"
|
|
|
|
|
2016-06-24 00:13:31 +01:00
|
|
|
#include <KLocalizedString>
|
|
|
|
|
2015-06-04 01:29:22 +01:00
|
|
|
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;
|
|
|
|
|
2016-12-13 15:40:47 +00:00
|
|
|
lvm2_pv::lvm2_pv(qint64 firstsector, qint64 lastsector,
|
|
|
|
qint64 sectorsused, const QString& label)
|
2018-04-07 19:54:30 +01:00
|
|
|
: FileSystem(firstsector, lastsector, sectorsused, label, FileSystem::Type::Lvm2_PV)
|
2016-12-13 15:40:47 +00:00
|
|
|
, m_PESize(0)
|
|
|
|
, m_TotalPE(0)
|
|
|
|
, m_AllocatedPE(0)
|
2015-07-13 15:16:36 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void lvm2_pv::init()
|
|
|
|
{
|
2016-05-26 15:27:03 +01:00
|
|
|
CommandSupportType lvmFound = findExternal(QStringLiteral("lvm")) ? cmdSupportFileSystem : cmdSupportNone;
|
2015-07-13 15:16:36 +01:00
|
|
|
|
2016-05-26 15:27:03 +01:00
|
|
|
m_Create = lvmFound;
|
|
|
|
m_Check = lvmFound;
|
|
|
|
m_Grow = lvmFound;
|
|
|
|
m_Shrink = lvmFound;
|
|
|
|
m_UpdateUUID = lvmFound;
|
2016-06-24 00:13:31 +01:00
|
|
|
m_GetUsed = lvmFound;
|
2015-07-13 15:16:36 +01:00
|
|
|
|
|
|
|
m_Move = (m_Check != cmdSupportNone) ? cmdSupportCore : cmdSupportNone;
|
|
|
|
|
2016-05-26 15:27:03 +01:00
|
|
|
m_GetLabel = cmdSupportCore;
|
|
|
|
m_Backup = cmdSupportCore;
|
|
|
|
m_GetUUID = cmdSupportCore;
|
|
|
|
|
2015-07-13 15:16:36 +01:00
|
|
|
m_GetLabel = cmdSupportNone;
|
2016-05-26 15:27:03 +01:00
|
|
|
m_Copy = cmdSupportNone; // Copying PV can confuse LVM
|
2015-07-13 15:16:36 +01:00
|
|
|
}
|
|
|
|
|
2016-09-05 14:48: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);
|
2016-09-05 14:48:36 +01:00
|
|
|
}
|
|
|
|
|
2015-07-13 15:16:36 +01:00
|
|
|
bool lvm2_pv::supportToolFound() const
|
|
|
|
{
|
|
|
|
return
|
2016-06-24 00:13:31 +01:00
|
|
|
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
|
|
|
|
{
|
|
|
|
return SupportTool(QStringLiteral("lvm2"), QUrl(QStringLiteral("http://sourceware.org/lvm2/")));
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2016-06-24 00:13:31 +01:00
|
|
|
qint64 lvm2_pv::readUsedCapacity(const QString& deviceNode) const
|
|
|
|
{
|
2017-02-05 15:50:40 +00:00
|
|
|
QString pvUsed = getpvField(QStringLiteral("pv_used"), deviceNode);
|
2016-08-06 22:15:50 +01:00
|
|
|
QString metadataOffset = getpvField(QStringLiteral("pe_start"), deviceNode);
|
2017-02-05 15:50:40 +00:00
|
|
|
return pvUsed.isEmpty() ? -1 : pvUsed.toLongLong() + metadataOffset.toLongLong();
|
2016-06-24 00:13:31 +01:00
|
|
|
}
|
|
|
|
|
2015-07-13 15:16:36 +01:00
|
|
|
bool lvm2_pv::check(Report& report, const QString& deviceNode) const
|
|
|
|
{
|
2016-04-21 22:48:05 +01:00
|
|
|
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
|
|
|
{
|
2016-08-08 16:12:52 +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
|
|
|
|
{
|
2016-04-21 22:48:05 +01:00
|
|
|
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
|
2016-09-05 14:48:36 +01:00
|
|
|
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 });
|
2016-07-30 18:48:04 +01:00
|
|
|
return rval && cmd.run(-1) && cmd.exitCode() == 0;
|
2016-05-24 21:13:23 +01:00
|
|
|
}
|
|
|
|
|
2016-10-30 01:46:38 +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
|
|
|
|
{
|
2016-04-21 22:48:05 +01:00
|
|
|
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;
|
|
|
|
}
|
2016-05-24 17:26:12 +01:00
|
|
|
|
2016-06-24 00:13:31 +01:00
|
|
|
QString lvm2_pv::readUUID(const QString& deviceNode) const
|
2016-05-24 17:26:12 +01:00
|
|
|
{
|
2016-06-24 00:13:31 +01:00
|
|
|
return getpvField(QStringLiteral("pv_uuid"), deviceNode);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool lvm2_pv::mount(Report& report, const QString& deviceNode, const QString& mountPoint)
|
|
|
|
{
|
2016-10-30 02:31:46 +00:00
|
|
|
Q_UNUSED(report)
|
|
|
|
Q_UNUSED(deviceNode)
|
|
|
|
Q_UNUSED(mountPoint)
|
2016-05-24 17:26:12 +01:00
|
|
|
return false;
|
|
|
|
}
|
2016-05-23 11:29:33 +01:00
|
|
|
|
2016-06-24 00:13:31 +01:00
|
|
|
bool lvm2_pv::unmount(Report& report, const QString& deviceNode)
|
2016-05-23 11:29:33 +01:00
|
|
|
{
|
2016-10-30 02:31:46 +00:00
|
|
|
Q_UNUSED(deviceNode)
|
|
|
|
Q_UNUSED(report)
|
2016-06-24 00:13:31 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:40:30 +01:00
|
|
|
bool lvm2_pv::canMount(const QString& deviceNode, const QString& mountPoint) const
|
2016-06-24 00:13:31 +01:00
|
|
|
{
|
2016-10-30 02:31:46 +00:00
|
|
|
Q_UNUSED(deviceNode)
|
|
|
|
Q_UNUSED(mountPoint)
|
2016-06-24 00:13:31 +01:00
|
|
|
return false;
|
2016-05-23 11:29:33 +01:00
|
|
|
}
|
|
|
|
|
2016-06-24 00:13:31 +01:00
|
|
|
bool lvm2_pv::canUnmount(const QString& deviceNode) const
|
|
|
|
{
|
2016-10-30 02:31:46 +00:00
|
|
|
Q_UNUSED(deviceNode)
|
2016-09-06 00:14:44 +01:00
|
|
|
return false;
|
2016-06-24 00:13:31 +01:00
|
|
|
}
|
2016-05-25 18:03:03 +01:00
|
|
|
|
2016-07-08 10:19:33 +01:00
|
|
|
qint64 lvm2_pv::getTotalPE(const QString& deviceNode)
|
2016-06-24 00:13:31 +01:00
|
|
|
{
|
2017-02-05 15:50:40 +00:00
|
|
|
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)
|
2016-06-24 00:13:31 +01:00
|
|
|
{
|
2017-02-05 15:50:40 +00:00
|
|
|
QString pvPeAllocCount = getpvField(QStringLiteral("pv_pe_alloc_count"), deviceNode);
|
|
|
|
return pvPeAllocCount.isEmpty() ? -1 : pvPeAllocCount.toLongLong();
|
2016-06-24 00:13:31 +01:00
|
|
|
}
|
|
|
|
|
2016-09-05 14:48:36 +01:00
|
|
|
void lvm2_pv::getPESize(const QString& deviceNode)
|
2016-06-24 00:13:31 +01:00
|
|
|
{
|
2017-02-05 15:50:40 +00:00
|
|
|
QString vgExtentSize = getpvField(QStringLiteral("vg_extent_size"), deviceNode);
|
|
|
|
m_PESize = vgExtentSize.isEmpty() ? -1 : vgExtentSize.toLongLong();
|
2016-06-24 00:13:31 +01:00
|
|
|
}
|
|
|
|
|
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-06-24 00:13:31 +01:00
|
|
|
{
|
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;
|
|
|
|
}
|
2017-09-04 19:55:08 +01:00
|
|
|
ExternalCommand cmd(QStringLiteral("lvm"), args, QProcess::ProcessChannelMode::SeparateChannels);
|
2016-05-25 18:03:03 +01:00
|
|
|
if (cmd.run(-1) && cmd.exitCode() == 0) {
|
2016-06-24 00:13:31 +01:00
|
|
|
return cmd.output().trimmed();
|
2016-05-25 18:03:03 +01:00
|
|
|
}
|
2016-06-24 00:13:31 +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
|
|
|
}
|
|
|
|
|
2016-11-02 23:06:55 +00:00
|
|
|
QList<LvmPV> lvm2_pv::getPVinNode(const PartitionNode* parent)
|
2016-07-03 12:09:56 +01:00
|
|
|
{
|
2016-11-02 23:06:55 +00:00
|
|
|
QList<LvmPV> partitions;
|
2016-09-07 14:36:25 +01:00
|
|
|
if (parent == nullptr)
|
|
|
|
return partitions;
|
2016-07-03 12:09:56 +01:00
|
|
|
|
2016-09-07 14:36:25 +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)
|
2016-09-10 22:08:21 +01:00
|
|
|
partitions.append(getPVinNode(node));
|
2016-09-07 14:36:25 +01:00
|
|
|
|
|
|
|
// 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())
|
2016-11-02 23:06:55 +00:00
|
|
|
partitions.append(LvmPV(p->mountPoint(), p));
|
2016-11-03 14:21:18 +00:00
|
|
|
|
2018-04-07 19:54:30 +01:00
|
|
|
if(p->fileSystem().type() == FileSystem::Type::Luks && p->deviceNode() == p->partitionPath())
|
2016-11-03 14:21:18 +00:00
|
|
|
partitions.append(LvmPV(p->mountPoint(), p, true));
|
2016-07-03 12:09:56 +01:00
|
|
|
}
|
2016-07-03 12:11:10 +01:00
|
|
|
|
2016-09-07 14:36:25 +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
|
|
|
|
*/
|
2016-11-02 23:06:55 +00:00
|
|
|
QList<LvmPV> lvm2_pv::getPVs(const QList<Device*>& devices)
|
2016-07-03 12:09:56 +01:00
|
|
|
{
|
2016-11-02 23:06:55 +00:00
|
|
|
QList<LvmPV> partitions;
|
2016-09-07 14:36:25 +01:00
|
|
|
for (auto const &d : devices)
|
2016-09-10 22:08:21 +01:00
|
|
|
partitions.append(getPVinNode(d->partitionTable()));
|
2016-09-07 14:36:25 +01:00
|
|
|
|
|
|
|
return partitions;
|
2016-07-03 12:09:56 +01:00
|
|
|
}
|
|
|
|
|
2015-06-04 01:29:22 +01:00
|
|
|
}
|
2016-11-02 23:06:55 +00:00
|
|
|
|
2018-05-09 08:09:17 +01:00
|
|
|
namespace LVM {
|
|
|
|
|
|
|
|
QList<LvmPV> pvList::m_list;
|
|
|
|
|
|
|
|
}
|
2016-11-03 13:26:14 +00:00
|
|
|
|
2016-11-02 23:06:55 +00:00
|
|
|
LvmPV::LvmPV(const QString vgName, const Partition* p, bool isLuks)
|
|
|
|
: m_vgName(vgName)
|
|
|
|
, m_p(p)
|
|
|
|
, m_isLuks(isLuks)
|
|
|
|
{
|
|
|
|
}
|