2020-04-05 21:16:16 +02:00
|
|
|
// ftpd is a server implementation based on the following:
|
|
|
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
|
|
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
|
|
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
|
|
|
//
|
2024-11-23 09:13:11 +01:00
|
|
|
// Copyright (C) 2024 Michael Theall
|
2020-04-05 21:16:16 +02: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 <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "fs.h"
|
2023-11-19 13:55:58 +01:00
|
|
|
#include "IOAbstraction.h"
|
2024-11-23 09:13:11 +01:00
|
|
|
#include "ioBuffer.h"
|
2020-04-05 21:16:16 +02:00
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
#include <gsl/pointers>
|
|
|
|
#include <gsl/util>
|
|
|
|
|
|
|
|
#include <array>
|
2020-05-19 03:45:46 +02:00
|
|
|
#include <cassert>
|
2024-11-23 09:13:11 +01:00
|
|
|
#include <cerrno>
|
2020-04-05 21:16:16 +02:00
|
|
|
#include <cinttypes>
|
2024-11-23 09:13:11 +01:00
|
|
|
#include <cstdint>
|
2020-04-05 21:16:16 +02:00
|
|
|
#include <cstdio>
|
2024-11-23 09:13:11 +01:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <memory>
|
2020-05-08 06:30:23 +02:00
|
|
|
#include <string>
|
2024-11-23 09:13:11 +01:00
|
|
|
#include <string_view>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
2020-04-05 21:16:16 +02:00
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
#if defined(__NDS__) || defined(__3DS__) || defined(__SWITCH__) || defined(__WIIU__)
|
2020-04-24 22:51:43 +02:00
|
|
|
#define getline __getline
|
|
|
|
#endif
|
|
|
|
|
2020-04-05 21:16:16 +02:00
|
|
|
std::string fs::printSize (std::uint64_t const size_)
|
|
|
|
{
|
|
|
|
constexpr std::uint64_t const KiB = 1024;
|
|
|
|
constexpr std::uint64_t const MiB = 1024 * KiB;
|
|
|
|
constexpr std::uint64_t const GiB = 1024 * MiB;
|
|
|
|
constexpr std::uint64_t const TiB = 1024 * GiB;
|
|
|
|
constexpr std::uint64_t const PiB = 1024 * TiB;
|
|
|
|
constexpr std::uint64_t const EiB = 1024 * PiB;
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
std::array<char, 64> buffer{};
|
2020-04-05 21:16:16 +02:00
|
|
|
|
|
|
|
for (auto const &[name, bin] : {
|
|
|
|
// clang-format off
|
|
|
|
std::make_pair ("EiB", EiB),
|
|
|
|
std::make_pair ("PiB", PiB),
|
|
|
|
std::make_pair ("TiB", TiB),
|
|
|
|
std::make_pair ("GiB", GiB),
|
|
|
|
std::make_pair ("MiB", MiB),
|
|
|
|
std::make_pair ("KiB", KiB)
|
|
|
|
// clang-format on
|
|
|
|
})
|
|
|
|
{
|
2020-04-06 07:36:03 +02:00
|
|
|
// get the integral portion of the number
|
2020-04-05 21:16:16 +02:00
|
|
|
auto const whole = size_ / bin;
|
|
|
|
if (size_ >= 100 * bin)
|
|
|
|
{
|
2020-04-06 07:36:03 +02:00
|
|
|
// >= 100, print xxxXiB
|
2024-11-23 09:13:11 +01:00
|
|
|
std::size_t const size = std::sprintf (buffer.data (), "%" PRIu64 "%s", whole, name);
|
|
|
|
return {buffer.data (), size};
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
2020-04-06 07:36:03 +02:00
|
|
|
// get the fractional portion of the number
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const frac = size_ - (whole * bin);
|
2020-04-05 21:16:16 +02:00
|
|
|
if (size_ >= 10 * bin)
|
|
|
|
{
|
2020-04-06 07:36:03 +02:00
|
|
|
// >= 10, print xx.xXiB
|
2024-11-23 09:13:11 +01:00
|
|
|
std::size_t const size = std::sprintf (
|
|
|
|
buffer.data (), "%" PRIu64 ".%" PRIu64 "%s", whole, frac * 10 / bin, name);
|
|
|
|
return {buffer.data (), size};
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (size_ >= 1000 * (bin / KiB))
|
|
|
|
{
|
2020-04-06 07:36:03 +02:00
|
|
|
// >= 1000 of lesser bin, print x.xxXiB
|
2024-11-23 09:13:11 +01:00
|
|
|
std::size_t const size = std::sprintf (
|
|
|
|
buffer.data (), "%" PRIu64 ".%02" PRIu64 "%s", whole, frac * 100 / bin, name);
|
|
|
|
return {buffer.data (), size};
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-06 07:36:03 +02:00
|
|
|
// < 1KiB, just print the number
|
2024-11-23 09:13:11 +01:00
|
|
|
std::size_t const size = std::sprintf (buffer.data (), "%" PRIu64 "B", size_);
|
|
|
|
return {buffer.data (), size};
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2020-04-24 22:51:43 +02:00
|
|
|
fs::File::~File ()
|
|
|
|
{
|
|
|
|
std::free (m_lineBuffer);
|
|
|
|
}
|
2020-04-05 21:16:16 +02:00
|
|
|
|
|
|
|
fs::File::File () = default;
|
|
|
|
|
|
|
|
fs::File::File (File &&that_) = default;
|
|
|
|
|
|
|
|
fs::File &fs::File::operator= (File &&that_) = default;
|
|
|
|
|
|
|
|
fs::File::operator bool () const
|
|
|
|
{
|
|
|
|
return static_cast<bool> (m_fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
fs::File::operator FILE * () const
|
|
|
|
{
|
|
|
|
return m_fp.get ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void fs::File::setBufferSize (std::size_t const size_)
|
|
|
|
{
|
2024-11-23 09:13:11 +01:00
|
|
|
if (m_buffer.size () != size_)
|
|
|
|
m_buffer.resize (size_);
|
2020-04-05 21:16:16 +02:00
|
|
|
|
|
|
|
if (m_fp)
|
2024-11-23 09:13:11 +01:00
|
|
|
(void)std::setvbuf (m_fp.get (), m_buffer.data (), _IOFBF, m_buffer.size ());
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
bool fs::File::open (gsl::not_null<char const *> const path_,
|
|
|
|
gsl::not_null<char const *> const mode_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2024-11-23 09:13:11 +01:00
|
|
|
gsl::owner<FILE *> fp = IOAbstraction::fopen (path_, mode_);
|
2020-04-05 21:16:16 +02:00
|
|
|
if (!fp)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_fp = std::unique_ptr<std::FILE, int (*) (std::FILE *)> (fp, &std::fclose);
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
if (!m_buffer.empty ())
|
|
|
|
(void)std::setvbuf (m_fp.get (), m_buffer.data (), _IOFBF, m_buffer.size ());
|
2020-04-05 21:16:16 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void fs::File::close ()
|
|
|
|
{
|
|
|
|
m_fp.reset ();
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
std::make_signed_t<std::size_t> fs::File::seek (std::make_signed_t<std::size_t> const pos_,
|
|
|
|
int const origin_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2023-11-19 13:55:58 +01:00
|
|
|
return IOAbstraction::fseek (m_fp.get (), pos_, origin_);
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
std::make_signed_t<std::size_t> fs::File::read (gsl::not_null<void *> const buffer_,
|
|
|
|
std::size_t const size_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2020-05-19 03:45:46 +02:00
|
|
|
assert (buffer_);
|
|
|
|
assert (size_ > 0);
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const rc = IOAbstraction::fread (buffer_, 1, size_, m_fp.get ());
|
|
|
|
if (rc == 0)
|
|
|
|
{
|
|
|
|
if (std::feof (m_fp.get ()))
|
|
|
|
return 0;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gsl::narrow_cast<std::make_signed_t<std::size_t>> (rc);
|
2020-05-19 03:45:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::make_signed_t<std::size_t> fs::File::read (IOBuffer &buffer_)
|
|
|
|
{
|
|
|
|
assert (buffer_.freeSize () > 0);
|
|
|
|
|
|
|
|
auto const rc = read (buffer_.freeArea (), buffer_.freeSize ());
|
|
|
|
if (rc > 0)
|
|
|
|
buffer_.markUsed (rc);
|
|
|
|
|
|
|
|
return rc;
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
2020-04-24 22:51:43 +02:00
|
|
|
std::string_view fs::File::readLine ()
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
auto rc = ::getline (&m_lineBuffer, &m_lineBufferSize, m_fp.get ());
|
|
|
|
if (rc < 0)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
while (rc > 0)
|
|
|
|
{
|
|
|
|
if (m_lineBuffer[rc - 1] != '\r' && m_lineBuffer[rc - 1] != '\n')
|
|
|
|
break;
|
|
|
|
|
|
|
|
m_lineBuffer[--rc] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc > 0)
|
2024-11-23 09:13:11 +01:00
|
|
|
return {m_lineBuffer, gsl::narrow_cast<std::size_t> (rc)};
|
2020-04-24 22:51:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
bool fs::File::readAll (gsl::not_null<void *> const buffer_, std::size_t const size_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2020-05-19 03:45:46 +02:00
|
|
|
assert (buffer_);
|
|
|
|
assert (size_ > 0);
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const p = static_cast<char *> (buffer_.get ());
|
2020-04-05 21:16:16 +02:00
|
|
|
|
2020-05-19 03:45:46 +02:00
|
|
|
std::size_t bytes = 0;
|
2020-04-05 21:16:16 +02:00
|
|
|
while (bytes < size_)
|
|
|
|
{
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const rc = read (p + bytes, size_ - bytes);
|
2020-04-05 21:16:16 +02:00
|
|
|
if (rc <= 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bytes += rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
std::make_signed_t<std::size_t> fs::File::write (gsl::not_null<void const *> const buffer_,
|
|
|
|
std::size_t const size_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2020-05-25 04:21:19 +02:00
|
|
|
assert (buffer_);
|
|
|
|
assert (size_ > 0);
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const rc = IOAbstraction::fwrite (buffer_, 1, size_, m_fp.get ());
|
|
|
|
if (rc == 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return gsl::narrow_cast<std::make_signed_t<std::size_t>> (rc);
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|
|
|
|
|
2020-05-25 04:21:19 +02:00
|
|
|
std::make_signed_t<std::size_t> fs::File::write (IOBuffer &buffer_)
|
|
|
|
{
|
|
|
|
assert (buffer_.usedSize () > 0);
|
|
|
|
|
|
|
|
auto const rc = write (buffer_.usedArea (), buffer_.usedSize ());
|
|
|
|
if (rc > 0)
|
|
|
|
buffer_.markFree (rc);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
bool fs::File::writeAll (gsl::not_null<void const *> const buffer_, std::size_t const size_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2020-05-19 03:45:46 +02:00
|
|
|
assert (buffer_);
|
|
|
|
assert (size_ > 0);
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const p = static_cast<char const *> (buffer_.get ());
|
2020-04-05 21:16:16 +02:00
|
|
|
|
2020-05-19 03:45:46 +02:00
|
|
|
std::size_t bytes = 0;
|
2020-04-05 21:16:16 +02:00
|
|
|
while (bytes < size_)
|
|
|
|
{
|
2024-11-23 09:13:11 +01:00
|
|
|
auto const rc = write (p + bytes, size_ - bytes);
|
2020-04-05 21:16:16 +02:00
|
|
|
if (rc <= 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bytes += rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
fs::Dir::~Dir () = default;
|
|
|
|
|
|
|
|
fs::Dir::Dir () = default;
|
|
|
|
|
|
|
|
fs::Dir::Dir (Dir &&that_) = default;
|
|
|
|
|
|
|
|
fs::Dir &fs::Dir::operator= (Dir &&that_) = default;
|
|
|
|
|
|
|
|
fs::Dir::operator bool () const
|
|
|
|
{
|
|
|
|
return static_cast<bool> (m_dp);
|
|
|
|
}
|
|
|
|
|
|
|
|
fs::Dir::operator DIR * () const
|
|
|
|
{
|
|
|
|
return m_dp.get ();
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
bool fs::Dir::open (gsl::not_null<char const *> const path_)
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2023-11-19 13:55:58 +01:00
|
|
|
auto const dp = IOAbstraction::opendir (path_);
|
2020-04-05 21:16:16 +02:00
|
|
|
if (!dp)
|
|
|
|
return false;
|
|
|
|
|
2023-11-19 13:55:58 +01:00
|
|
|
m_dp = std::unique_ptr<DIR, int (*) (DIR *)> (dp, &IOAbstraction::closedir);
|
2020-04-05 21:16:16 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void fs::Dir::close ()
|
|
|
|
{
|
|
|
|
m_dp.reset ();
|
|
|
|
}
|
|
|
|
|
2024-11-23 09:13:11 +01:00
|
|
|
dirent *fs::Dir::read ()
|
2020-04-05 21:16:16 +02:00
|
|
|
{
|
2020-04-06 07:36:03 +02:00
|
|
|
errno = 0;
|
2023-11-19 13:55:58 +01:00
|
|
|
return IOAbstraction::readdir (m_dp.get ());
|
2020-04-05 21:16:16 +02:00
|
|
|
}
|