kpmcore/src/core/lvmdevice.cpp

244 lines
8.0 KiB
C++
Raw Normal View History

2016-06-08 16:50:40 +01:00
/*************************************************************************
* Copyright (C) 2016 by Chantara Tith <tith.chantara@gmail.com> *
2016-06-08 16:50:40 +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 "core/lvmdevice.h"
#include "fs/filesystem.h"
#include "fs/filesystemfactory.h"
#include "core/partition.h"
2016-06-08 16:50:40 +01:00
#include "core/partitiontable.h"
#include "util/externalcommand.h"
2016-06-13 00:39:54 +01:00
#include "util/helpers.h"
2016-06-08 16:50:40 +01:00
#include <QRegularExpression>
#include <QStringList>
2016-06-13 00:39:54 +01:00
#include <KMountPoint>
#include <KDiskFreeSpaceInfo>
2016-06-08 16:50:40 +01:00
/** Constructs a representation of LVM device with functionning LV as Partition
*
* @param name Volume Group name
*/
2016-06-08 16:50:40 +01:00
LvmDevice::LvmDevice(const QString& name, const QString& iconname)
: VolumeManagerDevice(name,
(QStringLiteral("/dev/") + name),
getPeSize(name),
getTotalPE(name),
iconname,
Device::LVM_Device)
, m_peSize(getPeSize(name))
, m_totalPE(getTotalPE(name))
, m_allocPE(getAllocatedPE(name))
, m_freePE(getFreePE(name))
, m_UUID(getUUID(name))
2016-06-08 16:50:40 +01:00
{
initPartitions();
}
void LvmDevice::initPartitions()
{
qint64 firstUsable = 0;
qint64 lastusable = totalPE() - 1;
PartitionTable* pTable = new PartitionTable(PartitionTable::vmd, firstUsable, lastusable);
foreach (Partition* p, scanPartitions(*this, pTable)) {
pTable->append(p);
}
2016-06-12 14:09:46 +01:00
setPartitionTable(pTable);
}
/**
* @returns sorted Partition(LV) Array
*/
QList<Partition*> LvmDevice::scanPartitions(const Device& dev, PartitionTable* pTable) const
2016-06-08 16:50:40 +01:00
{
QList<Partition*> pList;
foreach (QString lvPath, lvPathList()) {
pList.append(scanPartition(lvPath, dev, pTable));
}
return pList;
}
/**
* @returns sorted Partition(LV) Array
*/
Partition* LvmDevice::scanPartition(const QString& lvpath, const Device& dev, PartitionTable* pTable) const
{
/*
* NOTE:
* LVM partition have 2 different start and end sector value
* 1. representing the actual LV start from 0 -> size of LV - 1
* 2. representing abstract LV's sector inside a VG partitionTable
* start from size of last Partitions -> size of LV - 1
* Reason for this is for the LV Partition to worrks nicely with other parts of the codebase
* without too many special cases.
*/
qint64 startSector;
qint64 endSector;
qint64 lvSize;
bool mounted = isMounted(lvpath);
QString mountPoint = QString();
KMountPoint::List mountPointList = KMountPoint::currentMountPoints(KMountPoint::NeedRealDeviceName);
mountPointList.append(KMountPoint::possibleMountPoints(KMountPoint::NeedRealDeviceName));
mountPoint = mountPointList.findByDevice(lvpath) ?
mountPointList.findByDevice(lvpath)->mountPoint() :
QString();
lvSize = getTotalLE(lvpath);
startSector = mappedSector(lvpath,0);
endSector = startSector + (lvSize - 1);
const KDiskFreeSpaceInfo freeSpaceInfo = KDiskFreeSpaceInfo::freeSpaceInfo(mountPoint);
FileSystem* fs = FileSystemFactory::create(FileSystem::detectFileSystem(lvpath), 0, lvSize - 1);
if (mounted && freeSpaceInfo.isValid() && mountPoint != QString()) {
//TODO: fix used space report. currently incorrect
fs->setSectorsUsed(freeSpaceInfo.used() / logicalSize());
}
2016-06-13 00:39:54 +01:00
Partition* part = new Partition(pTable,
dev,
PartitionRole(PartitionRole::Lvm_Lv),
fs,
startSector,
endSector,
lvpath,
PartitionTable::Flag::FlagLvm,
mountPoint,
mounted);
return part;
2016-06-08 16:50:40 +01:00
}
qint64 LvmDevice::mappedSector(const QString& lvpath, qint64 sector) const
{
qint64 mSector = 0;
QList<QString> lvpathList = lvPathList();
qint32 devIndex = lvpathList.indexOf(lvpath);
if (devIndex) {
for (int i = 0; i < devIndex; i++) {
//TODO: currently going over the same LV again and again is wasteful. Could use some more optimization
mSector += getTotalLE(lvpathList[i]);
}
mSector += sector;
}
return mSector;
}
2016-06-12 14:09:46 +01:00
QList<QString> LvmDevice::deviceNodeList() const
2016-06-08 16:50:40 +01:00
{
2016-06-12 14:09:46 +01:00
QList<QString> devPathList;
QString cmdOutput = getField(QStringLiteral("pv_name"), name());
if (cmdOutput.size()) {
QList<QString> tempPathList = cmdOutput.split(QStringLiteral("\n"), QString::SkipEmptyParts);
foreach(QString devPath, tempPathList) {
devPathList.append(devPath.trimmed());
2016-06-08 16:50:40 +01:00
}
}
2016-06-12 14:09:46 +01:00
return devPathList;
}
QList<QString> LvmDevice::lvPathList() const
{
QList<QString> lvPathList;
QString cmdOutput = getField(QStringLiteral("lv_path"), name());
if (cmdOutput.size()) {
QList<QString> tempPathList = cmdOutput.split(QStringLiteral("\n"), QString::SkipEmptyParts);
foreach(QString lvPath, tempPathList) {
lvPathList.append(lvPath.trimmed());
}
}
return lvPathList;
}
2016-06-12 14:09:46 +01:00
qint32 LvmDevice::getPeSize(const QString& vgname)
{
QString val = getField(QStringLiteral("vg_extent_size"), vgname);
return val.isEmpty() ? -1 : val.toInt();
2016-06-08 16:50:40 +01:00
}
qint32 LvmDevice::getTotalPE(const QString& vgname)
{
2016-06-12 14:09:46 +01:00
QString val = getField(QStringLiteral("vg_extent_count"), vgname);
return val.isEmpty() ? -1 : val.toInt();
2016-06-08 16:50:40 +01:00
}
qint32 LvmDevice::getAllocatedPE(const QString& vgname)
{
2016-06-12 14:09:46 +01:00
return getTotalPE(vgname) - getFreePE(vgname);
2016-06-08 16:50:40 +01:00
}
qint32 LvmDevice::getFreePE(const QString& vgname)
{
2016-06-12 14:09:46 +01:00
QString val = getField(QStringLiteral("vg_free_count"), vgname);
return val.isEmpty() ? -1 : val.toInt();
2016-06-08 16:50:40 +01:00
}
QString LvmDevice::getUUID(const QString& vgname)
2016-06-12 14:09:46 +01:00
{
QString val = getField(QStringLiteral("vg_uuid"), vgname);
return val.isEmpty() ? QStringLiteral("---") : val;
}
2016-06-13 00:39:54 +01:00
/** Query LVM details with field name
*
* @param fieldName lvm field name
* @param vgname
* @returns raw output of command output, usully with manay spaces within the returned string
* */
2016-06-12 14:09:46 +01:00
QString LvmDevice::getField(const QString& fieldName, const QString& vgname)
{
ExternalCommand cmd(QStringLiteral("lvm"),
2016-06-12 14:09:46 +01:00
{ QStringLiteral("vgs"),
QStringLiteral("--foreign"),
QStringLiteral("--readonly"),
QStringLiteral("--noheadings"),
QStringLiteral("--units"),
QStringLiteral("B"),
2016-06-12 14:09:46 +01:00
QStringLiteral("--nosuffix"),
QStringLiteral("--options"),
fieldName,
vgname });
if (cmd.run(-1) && cmd.exitCode() == 0) {
2016-06-12 14:09:46 +01:00
return cmd.output().trimmed();
}
2016-06-12 14:09:46 +01:00
return QString();
}
qint32 LvmDevice::getTotalLE(const QString& lvpath)
{
ExternalCommand cmd(QStringLiteral("lvm"),
{ QStringLiteral("lvdisplay"),
lvpath});
if (cmd.run(-1) && cmd.exitCode() == 0) {
QRegularExpression re(QStringLiteral("Current LE\\h+(\\d+)"));
QRegularExpressionMatch match = re.match(cmd.output());
if (match.hasMatch()) {
return match.captured(1).toInt();
}
}
return -1;
}