kpmcore/src/fs/filesystem.cpp

569 lines
18 KiB
C++

/*************************************************************************
* Copyright (C) 2012 by Volker Lanz <vl@fidra.de> *
* Copyright (C) 2015 by Teo Mrnjavac <teo@kde.org> *
* Copyright (C) 2016 by Andrius Štikonas <andrius@stikonas.eu> *
* *
* 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/fstab.h"
#include "fs/filesystem.h"
#include "fs/lvm2_pv.h"
#include "backend/corebackend.h"
#include "backend/corebackendmanager.h"
#include "util/externalcommand.h"
#include "util/capacity.h"
#include "util/helpers.h"
#include <blkid/blkid.h>
#include <KLocalizedString>
#include <QFileInfo>
#include <QStorageInfo>
const std::array< QColor, FileSystem::__lastType > FileSystem::defaultColorCode =
{
{
QColor( 220,205,175 ),
QColor( 187,249,207 ),
QColor( 102,121,150 ),
QColor( 122,145,180 ),
QColor( 143,170,210 ),
QColor( 155,155,130 ),
QColor( 204,179,215 ),
QColor( 229,201,240 ),
QColor( 244,214,255 ),
QColor( 216,220,135 ),
QColor( 251,255,157 ),
QColor( 200,255,254 ),
QColor( 137,200,198 ),
QColor( 210,136,142 ),
QColor( 240,165,171 ),
QColor( 151,220,134 ),
QColor( 220,205,175 ),
QColor( 173,205,255 ),
QColor( 176,155,185 ),
QColor( 170,30,77 ),
QColor( 96,140,85 ),
QColor( 33,137,108 ),
QColor( 250,230,255 ),
QColor( 242,155,104 ),
QColor( 160,210,180 ),
QColor( 255,170,0 ),
QColor( 170, 120, 255 ),
QColor( 177, 82, 69 )
}
};
/** Creates a new FileSystem object
@param firstsector the first sector used by this FileSystem on the Device
@param lastsector the last sector used by this FileSystem on the Device
@param sectorsused the number of sectors in use on the FileSystem
@param l the FileSystem label
@param t the FileSystem type
*/
FileSystem::FileSystem(qint64 firstsector, qint64 lastsector, qint64 sectorsused, const QString& l, FileSystem::Type t) :
m_Type(t),
m_FirstSector(firstsector),
m_LastSector(lastsector),
m_SectorsUsed(sectorsused),
m_Label(l),
m_UUID()
{
}
/** Reads the capacity in use on this FileSystem
@param deviceNode the device node for the Partition the FileSystem is on
@return the used capacity in bytes or -1 in case of an error
*/
qint64 FileSystem::readUsedCapacity(const QString& deviceNode) const
{
Q_UNUSED(deviceNode)
return -1;
}
static QString readBlkIdValue(const QString& deviceNode, const QString& tag)
{
blkid_cache cache;
QString rval;
if (blkid_get_cache(&cache, nullptr) == 0) {
blkid_dev dev;
char* label = nullptr;
if ((dev = blkid_get_dev(cache, deviceNode.toLocal8Bit().constData(), BLKID_DEV_NORMAL)) != nullptr &&
(label = blkid_get_tag_value(cache, tag.toLocal8Bit().constData(), deviceNode.toLocal8Bit().constData()))) {
rval = QString::fromLocal8Bit(label);
free(label);
}
blkid_put_cache(cache);
}
return rval;
}
FileSystem::Type FileSystem::detectFileSystem(const QString& partitionPath)
{
return CoreBackendManager::self()->backend()->detectFileSystem(partitionPath);
}
QString FileSystem::detectMountPoint(FileSystem* fs, const QString& partitionPath)
{
if (fs->type() == FileSystem::Lvm2_PV)
return FS::lvm2_pv::getVGName(partitionPath);
if (partitionPath.isEmpty()) // Happens when during initial scan LUKS is closed
return QString();
QStringList mountPoints;
QFileInfo partitionPathFileInfo(partitionPath);
QString partitionCanonicalPath = partitionPathFileInfo.canonicalFilePath();
const QList<QStorageInfo> mountedVolumes = QStorageInfo::mountedVolumes();
for (const QStorageInfo &storage : mountedVolumes) {
if (partitionCanonicalPath == QFileInfo(QString::fromLocal8Bit(storage.device())).canonicalFilePath() ) {
mountPoints.append(storage.rootPath());
}
}
mountPoints.append(possibleMountPoints(partitionPath));
return mountPoints.isEmpty() ? QString() : mountPoints.first();
}
bool FileSystem::detectMountStatus(FileSystem* fs, const QString& partitionPath)
{
bool mounted = false;
if (fs->type() == FileSystem::Lvm2_PV) {
mounted = FS::lvm2_pv::getVGName(partitionPath) != QString();
} else {
mounted = isMounted(partitionPath);
}
return mounted;
}
/** Reads the label for this FileSystem
@param deviceNode the device node for the Partition the FileSystem is on
@return the FileSystem label or an empty string in case of error
*/
QString FileSystem::readLabel(const QString& deviceNode) const
{
return readBlkIdValue(deviceNode, QStringLiteral("LABEL"));
}
/** Creates a new FileSystem
@param report Report to write status information to
@param deviceNode the device node for the Partition to create the FileSystem on
@return true if successful
*/
bool FileSystem::create(Report& report, const QString& deviceNode)
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
return true;
}
/** Creates a new FileSystem with a specified Label
@param report Report to write status information to
@param deviceNode the device node for the Partition to create the FileSystem on
@param label the new label for the FileSystem
@return true if successful
*/
bool FileSystem::createWithLabel(Report& report, const QString& deviceNode, const QString& label)
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(label)
return true;
}
/** Scans a new FileSystem and load file system specific class variables.
* @param deviceNode the device node for the Partition to create the FileSystem on
*/
void FileSystem::scan(const QString& deviceNode)
{
Q_UNUSED(deviceNode)
}
/** Resize a FileSystem to a given new length
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@param newLength the new length for the FileSystem in bytes
@return true on success
*/
bool FileSystem::resize(Report& report, const QString& deviceNode, qint64 newLength) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(newLength)
return true;
}
/** Resize a mounted FileSystem to a given new length
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@param mountPoint the mount point where FileSystem is mounted on
@param newLength the new length for the FileSystem in bytes
@return true on success
*/
bool FileSystem::resizeOnline(Report& report, const QString& deviceNode, const QString& mountPoint, qint64 newLength) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(mountPoint)
Q_UNUSED(newLength)
return true;
}
/** Move a FileSystem to a new start sector
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@param newStartSector the new start sector for the FileSystem
@return true on success
*/
bool FileSystem::move(Report& report, const QString& deviceNode, qint64 newStartSector) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(newStartSector)
return true;
}
/** Writes a label for the FileSystem to disk
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@param newLabel the new label for the FileSystem
@return true on success
*/
bool FileSystem::writeLabel(Report& report, const QString& deviceNode, const QString& newLabel)
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(newLabel)
return true;
}
/** Writes a label for the FileSystem to disk
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@param mountPoint the mount point where FileSystem is mounted on
@param newLabel the new label for the FileSystem
@return true on success
*/
bool FileSystem::writeLabelOnline(Report& report, const QString& deviceNode, const QString& mountPoint, const QString& newLabel)
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
Q_UNUSED(mountPoint)
Q_UNUSED(newLabel)
return true;
}
/** Copies a FileSystem from one Partition to another
@param report Report to write status information to
@param targetDeviceNode device node of the target Partition
@param sourceDeviceNode device node of the source Partition
@return true on success
*/
bool FileSystem::copy(Report& report, const QString& targetDeviceNode, const QString& sourceDeviceNode) const
{
Q_UNUSED(report)
Q_UNUSED(targetDeviceNode)
Q_UNUSED(sourceDeviceNode)
return true;
}
/** Backs up a FileSystem to a file
@param report Report to write status information to
@param sourceDevice Device the source FileSystem is on
@param deviceNode device node of the source Partition
@param filename name of the file to backup to
@return true on success
*/
bool FileSystem::backup(Report& report, const Device& sourceDevice, const QString& deviceNode, const QString& filename) const
{
Q_UNUSED(report)
Q_UNUSED(sourceDevice)
Q_UNUSED(deviceNode)
Q_UNUSED(filename)
return false;
}
/** Removes a FileSystem
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@return true if FileSystem is removed
*/
bool FileSystem::remove(Report& report, const QString& deviceNode) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
return true;
}
/** Checks a FileSystem for errors
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@return true if FileSystem is error-free
*/
bool FileSystem::check(Report& report, const QString& deviceNode) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
return true;
}
/** Updates a FileSystem UUID on disk
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@return true on success
*/
bool FileSystem::updateUUID(Report& report, const QString& deviceNode) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
return true;
}
/** Returns the FileSystem UUID by calling a FileSystem-specific helper program
@param deviceNode the device node for the Partition the FileSystem is on
@return the UUID or an empty string if the FileSystem does not support UUIDs
*/
QString FileSystem::readUUID(const QString& deviceNode) const
{
return readBlkIdValue(deviceNode, QStringLiteral("UUID"));
}
/** Give implementations of FileSystem a chance to update the boot sector after the
file system has been moved or copied.
@param report Report to write status information to
@param deviceNode the device node for the Partition the FileSystem is on
@return true on success
*/
bool FileSystem::updateBootSector(Report& report, const QString& deviceNode) const
{
Q_UNUSED(report)
Q_UNUSED(deviceNode)
return true;
}
/** @return the minimum capacity valid for this FileSystem in bytes */
qint64 FileSystem::minCapacity() const
{
return 8 * Capacity::unitFactor(Capacity::Byte, Capacity::MiB);
}
/** @return the maximum capacity valid for this FileSystem in bytes */
qint64 FileSystem::maxCapacity() const
{
return Capacity::unitFactor(Capacity::Byte, Capacity::EiB);
}
/** @return the maximum label length valid for this FileSystem */
int FileSystem::maxLabelLength() const
{
return 16;
}
/** Validates the label for this FileSystem
* @param parent the parent widget passed to the QObject constructor
* @return QValidator to validate the file system label line edit input */
QValidator* FileSystem::labelValidator(QObject *parent) const
{
Q_UNUSED(parent)
return nullptr;
}
/** @return this FileSystem's type as printable name */
QString FileSystem::name(const QStringList& languages) const
{
return nameForType(type(), languages);
}
/** @return a pointer to a QString C array with all FileSystem names */
static const KLocalizedString* typeNames()
{
static const KLocalizedString s[] = {
kxi18nc("@item filesystem name", "unknown"),
kxi18nc("@item filesystem name", "extended"),
kxi18nc("@item filesystem name", "ext2"),
kxi18nc("@item filesystem name", "ext3"),
kxi18nc("@item filesystem name", "ext4"),
kxi18nc("@item filesystem name", "linuxswap"),
kxi18nc("@item filesystem name", "fat16"),
kxi18nc("@item filesystem name", "fat32"),
kxi18nc("@item filesystem name", "ntfs"),
kxi18nc("@item filesystem name", "reiser"),
kxi18nc("@item filesystem name", "reiser4"),
kxi18nc("@item filesystem name", "xfs"),
kxi18nc("@item filesystem name", "jfs"),
kxi18nc("@item filesystem name", "hfs"),
kxi18nc("@item filesystem name", "hfsplus"),
kxi18nc("@item filesystem name", "ufs"),
kxi18nc("@item filesystem name", "unformatted"),
kxi18nc("@item filesystem name", "btrfs"),
kxi18nc("@item filesystem name", "hpfs"),
kxi18nc("@item filesystem name", "luks"),
kxi18nc("@item filesystem name", "ocfs2"),
kxi18nc("@item filesystem name", "zfs"),
kxi18nc("@item filesystem name", "exfat"),
kxi18nc("@item filesystem name", "nilfs2"),
kxi18nc("@item filesystem name", "lvm2 pv"),
kxi18nc("@item filesystem name", "f2fs"),
kxi18nc("@item filesystem name", "udf"),
kxi18nc("@item filesystem name", "iso9660"),
};
return s;
}
/** @param t the type to get the name for
@return the printable name for the given type
*/
QString FileSystem::nameForType(FileSystem::Type t, const QStringList& languages)
{
Q_ASSERT(t >= 0);
Q_ASSERT(t < __lastType);
return typeNames()[t].toString(languages);
}
/** @param s the name to get the type for
@return the type for the name or FileSystem::Unknown if not found
*/
FileSystem::Type FileSystem::typeForName(const QString& s, const QStringList& languages )
{
for (quint32 i = 0; i < __lastType; i++)
if (typeNames()[i].toString(languages) == s)
return static_cast<FileSystem::Type>(i);
return Unknown;
}
/** @return a QList of all known types */
QList<FileSystem::Type> FileSystem::types()
{
QList<FileSystem::Type> result;
int i = Ext2; // first "real" filesystem
while (i != __lastType)
result.append(static_cast<FileSystem::Type>(i++));
return result;
}
/** @return printable menu title for mounting this FileSystem */
QString FileSystem::mountTitle() const
{
return xi18nc("@title:menu", "Mount");
}
/** @return printable menu title for unmounting this FileSystem */
QString FileSystem::unmountTitle() const
{
return xi18nc("@title:menu", "Unmount");
}
/** Moves a FileSystem to a new start sector.
@param newStartSector where the FileSystem should be moved to
*/
void FileSystem::move(qint64 newStartSector)
{
const qint64 savedLength = length();
setFirstSector(newStartSector);
setLastSector(newStartSector + savedLength - 1);
}
bool FileSystem::canMount(const QString& deviceNode, const QString& mountPoint) const
{
Q_UNUSED(deviceNode)
// cannot mount if we have no mount points
return !mountPoint.isEmpty();
}
/** Attempt to mount this FileSystem on a given mount point
@param report the report to write information to
@param deviceNode the path to the device that is to be unmounted
@param mountPoint the mount point to mount the FileSystem on
@return true on success
*/
bool FileSystem::mount(Report& report, const QString &deviceNode, const QString &mountPoint)
{
ExternalCommand mountCmd( report,
QStringLiteral("mount"),
{ QStringLiteral("--verbose"),
deviceNode,
mountPoint });
if (mountCmd.run() && mountCmd.exitCode() == 0) {
return true;
}
return false;
}
/** Attempt to unmount this FileSystem
@param report the report to write information to
@param deviceNode the path to the device that is to be unmounted
@return true on success
*/
bool FileSystem::unmount(Report& report, const QString& deviceNode)
{
ExternalCommand umountCmd( report,
QStringLiteral("umount"),
{ QStringLiteral("--verbose"),
QStringLiteral("--all-targets"),
deviceNode });
if ( umountCmd.run() && umountCmd.exitCode() == 0 )
return true;
return false;
}
bool FileSystem::findExternal(const QString& cmdName, const QStringList& args, int expectedCode)
{
ExternalCommand cmd(cmdName, args);
if (!cmd.run())
return false;
return cmd.exitCode() == 0 || cmd.exitCode() == expectedCode;
}
bool FileSystem::supportToolFound() const
{
return false;
}
FileSystem::SupportTool FileSystem::supportToolName() const
{
return SupportTool();
}