269 lines
8.8 KiB
C++
269 lines
8.8 KiB
C++
/*************************************************************************
|
|
* Copyright (C) 2009, 2010 by Volker Lanz <vl@fidra.de> *
|
|
* Copyright (C) 2016 by Teo Mrnjavac <teo@kde.org> *
|
|
* Copyright (C) 2017 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 "util/externalcommand.h"
|
|
#include "util/report.h"
|
|
|
|
#if defined(Q_OS_LINUX)
|
|
#include <blkid/blkid.h>
|
|
#endif
|
|
|
|
#include <QChar>
|
|
#include <QDebug>
|
|
#include <QFileInfo>
|
|
#include <QRegularExpression>
|
|
#include <QTemporaryFile>
|
|
#include <QTextStream>
|
|
|
|
static void parseFsSpec(const QString& m_fsSpec, FstabEntry::Type& m_entryType, QString& m_deviceNode);
|
|
static QString findBlkIdDevice(const char *token, const QString& value);
|
|
|
|
struct FstabEntryPrivate
|
|
{
|
|
QString m_fsSpec;
|
|
QString m_deviceNode;
|
|
QString m_mountPoint;
|
|
QString m_type;
|
|
QStringList m_options;
|
|
int m_dumpFreq;
|
|
int m_passNumber;
|
|
QString m_comment;
|
|
FstabEntry::Type m_entryType;
|
|
};
|
|
|
|
FstabEntry::FstabEntry(const QString& fsSpec, const QString& mountPoint, const QString& type, const QString& options, int dumpFreq, int passNumber, const QString& comment) :
|
|
d(std::make_unique<FstabEntryPrivate>())
|
|
{
|
|
d->m_fsSpec = fsSpec;
|
|
d->m_mountPoint = mountPoint;
|
|
d->m_type = type;
|
|
d->m_dumpFreq = dumpFreq;
|
|
d->m_passNumber = passNumber;
|
|
d->m_comment = comment;
|
|
|
|
d->m_options = options.split(QLatin1Char(','));
|
|
parseFsSpec(d->m_fsSpec, d->m_entryType, d->m_deviceNode);
|
|
}
|
|
|
|
FstabEntryList readFstabEntries( const QString& fstabPath )
|
|
{
|
|
FstabEntryList fstabEntries;
|
|
QFile fstabFile( fstabPath );
|
|
if ( fstabFile.open( QIODevice::ReadOnly | QIODevice::Text ) )
|
|
{
|
|
const QStringList fstabLines = QString::fromLocal8Bit(fstabFile.readAll()).split( QLatin1Char('\n') );
|
|
for ( const QString& rawLine : fstabLines )
|
|
{
|
|
QString line = rawLine.trimmed();
|
|
if ( line.startsWith( QLatin1Char('#') ) || line.isEmpty()) {
|
|
fstabEntries.push_back( { {}, {}, {}, {}, {}, {}, line } );
|
|
continue;
|
|
}
|
|
|
|
QString comment = line.section( QLatin1Char('#'), 1 );
|
|
QStringList splitLine = line.section( QLatin1Char('#'), 0, 0 ).split( QRegularExpression(QStringLiteral("[\\s]+")), QString::SkipEmptyParts );
|
|
|
|
// We now split the standard components of /etc/fstab entry:
|
|
// (0) path, or UUID, or LABEL, etc,
|
|
// (1) mount point,
|
|
// (2) file system type,
|
|
// (3) options,
|
|
// (4) dump frequency (optional, defaults to 0), no comment is allowed if omitted,
|
|
// (5) pass number (optional, defaults to 0), no comment is allowed if omitted,
|
|
// (#) comment (optional).
|
|
switch (splitLine.length()) {
|
|
case 4:
|
|
fstabEntries.push_back( {splitLine.at(0), splitLine.at(1), splitLine.at(2), splitLine.at(3) } );
|
|
break;
|
|
case 5:
|
|
fstabEntries.push_back( {splitLine.at(0), splitLine.at(1), splitLine.at(2), splitLine.at(3), splitLine.at(4).toInt() } );
|
|
break;
|
|
case 6:
|
|
fstabEntries.push_back( {splitLine.at(0), splitLine.at(1), splitLine.at(2), splitLine.at(3), splitLine.at(4).toInt(), splitLine.at(5).toInt(), comment.isEmpty() ? QString() : QLatin1Char('#') + comment } );
|
|
break;
|
|
default:
|
|
fstabEntries.push_back( { {}, {}, {}, {}, {}, {}, QLatin1Char('#') + line } );
|
|
}
|
|
}
|
|
|
|
fstabFile.close();
|
|
if (fstabEntries.back().entryType() == FstabEntry::Type::comment && fstabEntries.back().comment().isEmpty())
|
|
fstabEntries.pop_back();
|
|
}
|
|
|
|
return fstabEntries;
|
|
}
|
|
|
|
void FstabEntry::setFsSpec(const QString& s)
|
|
{
|
|
d->m_fsSpec = s;
|
|
parseFsSpec(d->m_fsSpec, d->m_entryType, d->m_deviceNode);
|
|
}
|
|
|
|
const QString& FstabEntry::fsSpec() const
|
|
{
|
|
return d->m_fsSpec;
|
|
}
|
|
|
|
const QString& FstabEntry::deviceNode() const
|
|
{
|
|
return d->m_deviceNode;
|
|
}
|
|
|
|
const QString& FstabEntry::mountPoint() const
|
|
{
|
|
return d->m_mountPoint;
|
|
}
|
|
|
|
const QString& FstabEntry::type() const
|
|
{
|
|
return d->m_type;
|
|
}
|
|
|
|
const QStringList& FstabEntry::options() const
|
|
{
|
|
return d->m_options;
|
|
}
|
|
|
|
int FstabEntry::dumpFreq() const
|
|
{
|
|
return d->m_dumpFreq;
|
|
}
|
|
|
|
int FstabEntry::passNumber() const
|
|
{
|
|
return d->m_passNumber;
|
|
}
|
|
|
|
const QString& FstabEntry::comment() const
|
|
{
|
|
return d->m_comment;
|
|
}
|
|
|
|
FstabEntry::Type FstabEntry::entryType() const
|
|
{
|
|
return d->m_entryType;
|
|
}
|
|
|
|
void FstabEntry::setMountPoint(const QString& s)
|
|
{
|
|
d->m_mountPoint = s;
|
|
}
|
|
|
|
void FstabEntry::setOptions(const QStringList& s)
|
|
{
|
|
d->m_options = s;
|
|
}
|
|
|
|
void FstabEntry::setDumpFreq(int s)
|
|
{
|
|
d->m_dumpFreq = s;
|
|
}
|
|
|
|
void FstabEntry::setPassNumber(int s)
|
|
{
|
|
d->m_passNumber = s;
|
|
}
|
|
|
|
QStringList possibleMountPoints(const QString& deviceNode, const QString& fstabPath)
|
|
{
|
|
QStringList mountPoints;
|
|
QString canonicalPath = QFileInfo(deviceNode).canonicalFilePath();
|
|
const FstabEntryList fstabEntryList = readFstabEntries( fstabPath );
|
|
for (const FstabEntry &entry : fstabEntryList)
|
|
if (QFileInfo(entry.deviceNode()).canonicalFilePath() == canonicalPath)
|
|
mountPoints.append(entry.mountPoint());
|
|
|
|
return mountPoints;
|
|
}
|
|
|
|
static QString findBlkIdDevice(const char *token, const QString& value)
|
|
{
|
|
QString rval;
|
|
|
|
#if defined(Q_OS_LINUX)
|
|
if (char* c = blkid_evaluate_tag(token, value.toLocal8Bit().constData(), nullptr)) {
|
|
rval = QString::fromLocal8Bit(c);
|
|
free(c);
|
|
}
|
|
#endif
|
|
|
|
return rval;
|
|
}
|
|
|
|
static void parseFsSpec(const QString& m_fsSpec, FstabEntry::Type& m_entryType, QString& m_deviceNode)
|
|
{
|
|
m_entryType = FstabEntry::Type::comment;
|
|
if (m_fsSpec.startsWith(QStringLiteral("UUID="))) {
|
|
m_entryType = FstabEntry::Type::uuid;
|
|
m_deviceNode = findBlkIdDevice("UUID", QString(m_fsSpec).remove(QStringLiteral("UUID=")));
|
|
} else if (m_fsSpec.startsWith(QStringLiteral("LABEL="))) {
|
|
m_entryType = FstabEntry::Type::label;
|
|
m_deviceNode = findBlkIdDevice("LABEL", QString(m_fsSpec).remove(QStringLiteral("LABEL=")));
|
|
} else if (m_fsSpec.startsWith(QStringLiteral("PARTUUID="))) {
|
|
m_entryType = FstabEntry::Type::uuid;
|
|
m_deviceNode = findBlkIdDevice("PARTUUID", QString(m_fsSpec).remove(QStringLiteral("PARTUUID=")));
|
|
} else if (m_fsSpec.startsWith(QStringLiteral("PARTLABEL="))) {
|
|
m_entryType = FstabEntry::Type::label;
|
|
m_deviceNode = findBlkIdDevice("PARTLABEL", QString(m_fsSpec).remove(QStringLiteral("PARTLABEL=")));
|
|
} else if (m_fsSpec.startsWith(QStringLiteral("/"))) {
|
|
m_entryType = FstabEntry::Type::deviceNode;
|
|
m_deviceNode = m_fsSpec;
|
|
}
|
|
}
|
|
|
|
static void writeEntry(QTextStream& s, const FstabEntry& entry)
|
|
{
|
|
if (entry.entryType() == FstabEntry::Type::comment) {
|
|
s << entry.comment() << "\n";
|
|
return;
|
|
}
|
|
|
|
QString options;
|
|
if (entry.options().size() > 0) {
|
|
options = entry.options().join(QLatin1Char(','));
|
|
if (options.isEmpty())
|
|
options = QStringLiteral("defaults");
|
|
}
|
|
else
|
|
options = QStringLiteral("defaults");
|
|
|
|
s << entry.fsSpec() << "\t"
|
|
<< (entry.mountPoint().isEmpty() ? QStringLiteral("none") : entry.mountPoint()) << "\t"
|
|
<< entry.type() << "\t"
|
|
<< options << "\t"
|
|
<< entry.dumpFreq() << "\t"
|
|
<< entry.passNumber() << "\t"
|
|
<< entry.comment() << "\n";
|
|
}
|
|
|
|
bool writeMountpoints(const FstabEntryList& fstabEntries, const QString& filename)
|
|
{
|
|
Report report(nullptr);
|
|
QByteArray fstabContents;
|
|
QTextStream out(&fstabContents);
|
|
|
|
for (const auto &e : fstabEntries)
|
|
writeEntry(out, e);
|
|
|
|
ExternalCommand cmd;
|
|
return cmd.writeData(report, fstabContents, filename, 0);
|
|
}
|