mame/3rdparty/bx/src/file.cpp
2023-09-08 05:14:35 +10:00

959 lines
19 KiB
C++

/*
* Copyright 2010-2022 Branimir Karadzic. All rights reserved.
* License: https://github.com/bkaradzic/bx/blob/master/LICENSE
*/
#include <bx/file.h>
#ifndef BX_CONFIG_CRT_FILE_READER_WRITER
# define BX_CONFIG_CRT_FILE_READER_WRITER !BX_CRT_NONE
#endif // BX_CONFIG_CRT_FILE_READER_WRITER
#ifndef BX_CONFIG_CRT_DIRECTORY_READER
# define BX_CONFIG_CRT_DIRECTORY_READER (BX_PLATFORM_OS_DESKTOP && !BX_CRT_NONE)
#endif // BX_CONFIG_CRT_DIRECTORY_READER
#if BX_CRT_NONE
# include "crt0.h"
#else
# if BX_CONFIG_CRT_DIRECTORY_READER
# include <dirent.h>
# endif // BX_CONFIG_CRT_DIRECTORY_READER
# include <stdio.h> // remove
# include <sys/stat.h> // stat, mkdir
# if BX_CRT_MSVC
# include <direct.h> // _getcwd
# else
# include <unistd.h> // getcwd
# endif // BX_CRT_MSVC
#endif // !BX_CRT_NONE
namespace bx
{
class NoopWriterImpl : public FileWriterI
{
public:
NoopWriterImpl(void*)
{
}
virtual ~NoopWriterImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, bool _append, Error* _err) override
{
BX_UNUSED(_filePath, _append, _err);
return false;
}
virtual void close() override
{
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_UNUSED(_offset, _whence);
return 0;
}
virtual int32_t write(const void* _data, int32_t _size, Error* _err) override
{
BX_UNUSED(_data, _err);
return _size;
}
};
#if BX_CONFIG_CRT_FILE_READER_WRITER
# if BX_CRT_MSVC
# define fseeko64 _fseeki64
# define ftello64 _ftelli64
# elif 0 \
|| BX_PLATFORM_ANDROID \
|| BX_PLATFORM_BSD \
|| BX_PLATFORM_HAIKU \
|| BX_PLATFORM_IOS \
|| BX_PLATFORM_OSX
# define fseeko64 fseeko
# define ftello64 ftello
# elif BX_PLATFORM_PS4
# define fseeko64 fseek
# define ftello64 ftell
# endif // BX_
class FileReaderImpl : public FileReaderI
{
public:
FileReaderImpl(FILE* _file)
: m_file(_file)
, m_open(false)
{
}
virtual ~FileReaderImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
if (NULL != m_file)
{
BX_ERROR_SET(_err, kErrorReaderWriterAlreadyOpen, "FileReader: File is already open.");
return false;
}
m_file = fopen(_filePath.getCPtr(), "rb");
if (NULL == m_file)
{
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "FileReader: Failed to open file.");
return false;
}
m_open = true;
return true;
}
virtual void close() override
{
if (m_open
&& NULL != m_file)
{
fclose(m_file);
m_file = NULL;
}
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_ASSERT(NULL != m_file, "Reader/Writer file is not open.");
fseeko64(m_file, _offset, _whence);
return ftello64(m_file);
}
virtual int32_t read(void* _data, int32_t _size, Error* _err) override
{
BX_ASSERT(NULL != m_file, "Reader/Writer file is not open.");
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
int32_t size = (int32_t)fread(_data, 1, _size, m_file);
if (size != _size)
{
if (0 != feof(m_file) )
{
BX_ERROR_SET(_err, kErrorReaderWriterEof, "FileReader: EOF.");
}
else if (0 != ferror(m_file) )
{
BX_ERROR_SET(_err, kErrorReaderWriterRead, "FileReader: read error.");
}
return size >= 0 ? size : 0;
}
return size;
}
private:
FILE* m_file;
bool m_open;
};
class FileWriterImpl : public FileWriterI
{
public:
FileWriterImpl(FILE* _file)
: m_file(_file)
, m_open(false)
{
}
virtual ~FileWriterImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, bool _append, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
if (NULL != m_file)
{
BX_ERROR_SET(_err, kErrorReaderWriterAlreadyOpen, "FileReader: File is already open.");
return false;
}
m_file = fopen(_filePath.getCPtr(), _append ? "ab" : "wb");
if (NULL == m_file)
{
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "FileWriter: Failed to open file.");
return false;
}
m_open = true;
return true;
}
virtual void close() override
{
if (m_open
&& NULL != m_file)
{
fclose(m_file);
m_file = NULL;
}
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_ASSERT(NULL != m_file, "Reader/Writer file is not open.");
fseeko64(m_file, _offset, _whence);
return ftello64(m_file);
}
virtual int32_t write(const void* _data, int32_t _size, Error* _err) override
{
BX_ASSERT(NULL != m_file, "Reader/Writer file is not open.");
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
int32_t size = (int32_t)fwrite(_data, 1, _size, m_file);
if (size != _size)
{
BX_ERROR_SET(_err, kErrorReaderWriterWrite, "FileWriter: write failed.");
return size >= 0 ? size : 0;
}
return size;
}
private:
FILE* m_file;
bool m_open;
};
ReaderI* getStdIn()
{
static FileReaderImpl s_stdIn(stdout);
return &s_stdIn;
}
WriterI* getStdOut()
{
static FileWriterImpl s_stdOut(stdout);
return &s_stdOut;
}
WriterI* getStdErr()
{
static FileWriterImpl s_stdOut(stderr);
return &s_stdOut;
}
#elif BX_CRT_NONE
class FileReaderImpl : public FileReaderI
{
public:
FileReaderImpl(void* _file)
: m_fd(int32_t(intptr_t(_file) ) )
, m_open(false)
{
}
virtual ~FileReaderImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
if (0 != m_fd)
{
BX_ERROR_SET(_err, kErrorReaderWriterAlreadyOpen, "FileReader: File is already open.");
return false;
}
m_fd = crt0::open(_filePath.getCPtr(), crt0::Open::Read, 0);
if (0 >= m_fd)
{
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "FileReader: Failed to open file.");
return false;
}
m_open = true;
return true;
}
virtual void close() override
{
if (m_open
&& 0 != m_fd)
{
crt0::close(m_fd);
m_fd = 0;
}
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_ASSERT(0 != m_fd, "Reader/Writer file is not open.");
return crt0::seek(m_fd, _offset, crt0::Whence::Enum(_whence) );
}
virtual int32_t read(void* _data, int32_t _size, Error* _err) override
{
BX_ASSERT(0 != m_fd, "Reader/Writer file is not open.");
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
int32_t size = crt0::read(m_fd, _data, _size);
if (size != _size)
{
BX_UNUSED(_err);
// if (0 != feof(m_file) )
// {
// BX_ERROR_SET(_err, kErrorReaderWriterEof, "FileReader: EOF.");
// }
// else if (0 != ferror(m_file) )
// {
// BX_ERROR_SET(_err, kErrorReaderWriterRead, "FileReader: read error.");
// }
return size >= 0 ? size : 0;
}
return size;
}
private:
int32_t m_fd;
bool m_open;
};
class FileWriterImpl : public FileWriterI
{
public:
FileWriterImpl(void* _file)
: m_fd(int32_t(intptr_t(_file) ) )
, m_open(false)
{
}
virtual ~FileWriterImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, bool _append, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
if (0 != m_fd)
{
BX_ERROR_SET(_err, kErrorReaderWriterAlreadyOpen, "FileReader: File is already open.");
return false;
}
m_fd = crt0::open(_filePath.getCPtr(), _append ? crt0::Open::Append : crt0::Open::Write, 0600);
if (0 >= m_fd)
{
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "FileWriter: Failed to open file.");
return false;
}
m_open = true;
return true;
}
virtual void close() override
{
if (m_open
&& 0 != m_fd)
{
crt0::close(m_fd);
m_fd = 0;
}
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_ASSERT(0 != m_fd, "Reader/Writer file is not open.");
return crt0::seek(m_fd, _offset, crt0::Whence::Enum(_whence) );
}
virtual int32_t write(const void* _data, int32_t _size, Error* _err) override
{
BX_ASSERT(0 != m_fd, "Reader/Writer file is not open.");
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
int32_t size = crt0::write(m_fd, _data, _size);
if (size != _size)
{
BX_ERROR_SET(_err, kErrorReaderWriterWrite, "FileWriter: write failed.");
return size >= 0 ? size : 0;
}
return size;
}
private:
int32_t m_fd;
bool m_open;
};
ReaderI* getStdIn()
{
static FileReaderImpl s_stdIn( (void*)intptr_t(crt0::Io::In) );
return &s_stdIn;
}
WriterI* getStdOut()
{
static FileWriterImpl s_stdOut( (void*)intptr_t(crt0::Io::Out) );
return &s_stdOut;
}
WriterI* getStdErr()
{
static FileWriterImpl s_stdOut( (void*)intptr_t(crt0::Io::Err) );
return &s_stdOut;
}
#else
class FileReaderImpl : public FileReaderI
{
public:
FileReaderImpl(void*)
{
}
virtual ~FileReaderImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, Error* _err) override
{
BX_UNUSED(_filePath, _err);
return false;
}
virtual void close() override
{
}
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) override
{
BX_UNUSED(_offset, _whence);
return 0;
}
virtual int32_t read(void* _data, int32_t _size, Error* _err) override
{
BX_UNUSED(_data, _size, _err);
return 0;
}
};
typedef NoopWriterImpl FileWriterImpl;
ReaderI* getStdIn()
{
static FileReaderImpl s_stdIn(NULL);
return &s_stdIn;
}
WriterI* getStdOut()
{
static FileWriterImpl s_stdOut(NULL);
return &s_stdOut;
}
WriterI* getStdErr()
{
static FileWriterImpl s_stdOut(NULL);
return &s_stdOut;
}
#endif // BX_CONFIG_CRT_FILE_READER_WRITER
WriterI* getNullOut()
{
static NoopWriterImpl s_nullOut(NULL);
return &s_nullOut;
}
FileReader::FileReader()
{
BX_STATIC_ASSERT(sizeof(FileReaderImpl) <= sizeof(m_internal) );
BX_PLACEMENT_NEW(m_internal, FileReaderImpl)(NULL);
}
FileReader::~FileReader()
{
FileReaderImpl* impl = reinterpret_cast<FileReaderImpl*>(m_internal);
impl->~FileReaderImpl();
}
bool FileReader::open(const FilePath& _filePath, Error* _err)
{
FileReaderImpl* impl = reinterpret_cast<FileReaderImpl*>(m_internal);
return impl->open(_filePath, _err);
}
void FileReader::close()
{
FileReaderImpl* impl = reinterpret_cast<FileReaderImpl*>(m_internal);
impl->close();
}
int64_t FileReader::seek(int64_t _offset, Whence::Enum _whence)
{
FileReaderImpl* impl = reinterpret_cast<FileReaderImpl*>(m_internal);
return impl->seek(_offset, _whence);
}
int32_t FileReader::read(void* _data, int32_t _size, Error* _err)
{
FileReaderImpl* impl = reinterpret_cast<FileReaderImpl*>(m_internal);
return impl->read(_data, _size, _err);
}
FileWriter::FileWriter()
{
BX_STATIC_ASSERT(sizeof(FileWriterImpl) <= sizeof(m_internal) );
BX_PLACEMENT_NEW(m_internal, FileWriterImpl)(NULL);
}
FileWriter::~FileWriter()
{
FileWriterImpl* impl = reinterpret_cast<FileWriterImpl*>(m_internal);
impl->~FileWriterImpl();
}
bool FileWriter::open(const FilePath& _filePath, bool _append, Error* _err)
{
FileWriterImpl* impl = reinterpret_cast<FileWriterImpl*>(m_internal);
return impl->open(_filePath, _append, _err);
}
void FileWriter::close()
{
FileWriterImpl* impl = reinterpret_cast<FileWriterImpl*>(m_internal);
impl->close();
}
int64_t FileWriter::seek(int64_t _offset, Whence::Enum _whence)
{
FileWriterImpl* impl = reinterpret_cast<FileWriterImpl*>(m_internal);
return impl->seek(_offset, _whence);
}
int32_t FileWriter::write(const void* _data, int32_t _size, Error* _err)
{
FileWriterImpl* impl = reinterpret_cast<FileWriterImpl*>(m_internal);
return impl->write(_data, _size, _err);
}
#if BX_CONFIG_CRT_DIRECTORY_READER
class DirectoryReaderImpl : public ReaderOpenI, public CloserI, public ReaderI
{
public:
DirectoryReaderImpl()
: m_dir(NULL)
, m_pos(0)
{
}
virtual ~DirectoryReaderImpl()
{
close();
}
virtual bool open(const FilePath& _filePath, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
m_dir = opendir(_filePath.getCPtr() );
if (NULL == m_dir)
{
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "DirectoryReader: Failed to open directory.");
return false;
}
m_pos = 0;
return true;
}
virtual void close() override
{
if (NULL != m_dir)
{
closedir(m_dir);
m_dir = NULL;
}
}
virtual int32_t read(void* _data, int32_t _size, Error* _err) override
{
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
int32_t total = 0;
uint8_t* out = (uint8_t*)_data;
while (0 < _size)
{
if (0 == m_pos)
{
if (!fetch(m_cache, m_dir) )
{
BX_ERROR_SET(_err, kErrorReaderWriterEof, "DirectoryReader: EOF.");
return total;
}
}
const uint8_t* src = (const uint8_t*)&m_cache;
int32_t size = min<int32_t>(_size, sizeof(m_cache)-m_pos);
memCopy(&out[total], &src[m_pos], size);
total += size;
_size -= size;
m_pos += size;
m_pos %= sizeof(m_cache);
}
return total;
}
static bool fetch(FileInfo& _out, DIR* _dir)
{
for (;;)
{
const dirent* item = readdir(_dir);
if (NULL == item)
{
break;
}
if (0 != (item->d_type & DT_DIR) )
{
_out.type = FileType::Dir;
_out.size = UINT64_MAX;
_out.filePath.set(item->d_name);
return true;
}
if (0 != (item->d_type & DT_REG) )
{
_out.type = FileType::File;
_out.size = UINT64_MAX;
_out.filePath.set(item->d_name);
return true;
}
}
return false;
}
FileInfo m_cache;
DIR* m_dir;
int32_t m_pos;
};
#else
class DirectoryReaderImpl : public ReaderOpenI, public CloserI, public ReaderI
{
public:
DirectoryReaderImpl()
{
}
virtual ~DirectoryReaderImpl()
{
}
virtual bool open(const FilePath& _filePath, Error* _err) override
{
BX_UNUSED(_filePath);
BX_ERROR_SET(_err, kErrorReaderWriterOpen, "DirectoryReader: Failed to open directory.");
return false;
}
virtual void close() override
{
}
virtual int32_t read(void* _data, int32_t _size, Error* _err) override
{
BX_UNUSED(_data, _size);
BX_ASSERT(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
BX_ERROR_SET(_err, kErrorReaderWriterEof, "DirectoryReader: EOF.");
return 0;
}
};
#endif // BX_CONFIG_CRT_DIRECTORY_READER
DirectoryReader::DirectoryReader()
{
BX_STATIC_ASSERT(sizeof(DirectoryReaderImpl) <= sizeof(m_internal) );
BX_PLACEMENT_NEW(m_internal, DirectoryReaderImpl);
}
DirectoryReader::~DirectoryReader()
{
DirectoryReaderImpl* impl = reinterpret_cast<DirectoryReaderImpl*>(m_internal);
impl->~DirectoryReaderImpl();
}
bool DirectoryReader::open(const FilePath& _filePath, Error* _err)
{
DirectoryReaderImpl* impl = reinterpret_cast<DirectoryReaderImpl*>(m_internal);
return impl->open(_filePath, _err);
}
void DirectoryReader::close()
{
DirectoryReaderImpl* impl = reinterpret_cast<DirectoryReaderImpl*>(m_internal);
impl->close();
}
int32_t DirectoryReader::read(void* _data, int32_t _size, Error* _err)
{
DirectoryReaderImpl* impl = reinterpret_cast<DirectoryReaderImpl*>(m_internal);
return impl->read(_data, _size, _err);
}
bool stat(FileInfo& _outFileInfo, const FilePath& _filePath)
{
#if BX_CRT_NONE
BX_UNUSED(_filePath, _outFileInfo);
return false;
#else
_outFileInfo.size = 0;
_outFileInfo.type = FileType::Count;
# if BX_COMPILER_MSVC
struct ::_stat64 st;
int32_t result = ::_stat64(_filePath.getCPtr(), &st);
if (0 != result)
{
return false;
}
if (0 != (st.st_mode & _S_IFREG) )
{
_outFileInfo.type = FileType::File;
}
else if (0 != (st.st_mode & _S_IFDIR) )
{
_outFileInfo.type = FileType::Dir;
}
# else
struct ::stat st;
int32_t result = ::stat(_filePath.getCPtr(), &st);
if (0 != result)
{
return false;
}
if (0 != (st.st_mode & S_IFREG) )
{
_outFileInfo.type = FileType::File;
}
else if (0 != (st.st_mode & S_IFDIR) )
{
_outFileInfo.type = FileType::Dir;
}
# endif // BX_COMPILER_MSVC
_outFileInfo.size = st.st_size;
return true;
#endif // BX_CRT_NONE
}
bool make(const FilePath& _filePath, Error* _err)
{
BX_ERROR_SCOPE(_err);
if (!_err->isOk() )
{
return false;
}
#if BX_CRT_MSVC
int32_t result = ::_mkdir(_filePath.getCPtr() );
#elif BX_CRT_MINGW
int32_t result = ::mkdir(_filePath.getCPtr());
#elif BX_CRT_NONE
BX_UNUSED(_filePath);
int32_t result = -1;
#else
int32_t result = ::mkdir(_filePath.getCPtr(), 0700);
#endif // BX_CRT_MSVC
if (0 != result)
{
BX_ERROR_SET(_err, kErrorAccess, "The parent directory does not allow write permission to the process.");
return false;
}
return true;
}
bool makeAll(const FilePath& _filePath, Error* _err)
{
BX_ERROR_SCOPE(_err);
if (!_err->isOk() )
{
return false;
}
FileInfo fi;
if (stat(fi, _filePath) )
{
if (FileType::Dir == fi.type)
{
return true;
}
BX_ERROR_SET(_err, kErrorNotDirectory, "File already exist, and is not directory.");
return false;
}
const StringView dir = strRTrim(_filePath, "/");
const StringView slash = strRFind(dir, '/');
if (!slash.isEmpty()
&& slash.getPtr() - dir.getPtr() > 1)
{
if (!makeAll(StringView(dir.getPtr(), slash.getPtr() ), _err) )
{
return false;
}
}
FilePath path(dir);
return make(path, _err);
}
bool remove(const FilePath& _filePath, Error* _err)
{
BX_ERROR_SCOPE(_err);
if (!_err->isOk() )
{
return false;
}
#if BX_CRT_MSVC
int32_t result = -1;
FileInfo fi;
if (stat(fi, _filePath) )
{
if (FileType::Dir == fi.type)
{
result = ::_rmdir(_filePath.getCPtr() );
}
else
{
result = ::remove(_filePath.getCPtr() );
}
}
#elif BX_CRT_NONE
BX_UNUSED(_filePath);
int32_t result = -1;
#else
int32_t result = ::remove(_filePath.getCPtr() );
#endif // BX_CRT_MSVC
if (0 != result)
{
BX_ERROR_SET(_err, kErrorAccess, "The parent directory does not allow write permission to the process.");
return false;
}
return true;
}
bool removeAll(const FilePath& _filePath, Error* _err)
{
BX_ERROR_SCOPE(_err);
if (remove(_filePath, _err) )
{
return true;
}
_err->reset();
FileInfo fi;
if (!stat(fi, _filePath) )
{
BX_ERROR_SET(_err, kErrorAccess, "The parent directory does not allow write permission to the process.");
return false;
}
if (FileType::Dir != fi.type)
{
BX_ERROR_SET(_err, kErrorNotDirectory, "File already exist, and is not directory.");
return false;
}
Error err;
DirectoryReader dr;
if (!open(&dr, _filePath, &err) )
{
BX_ERROR_SET(_err, kErrorNotDirectory, "File already exist, and is not directory.");
return false;
}
while (err.isOk() )
{
read(&dr, fi, &err);
if (err.isOk() )
{
if (0 == strCmp(fi.filePath, ".")
|| 0 == strCmp(fi.filePath, "..") )
{
continue;
}
FilePath path(_filePath);
path.join(fi.filePath);
if (!removeAll(path, _err) )
{
_err->reset();
break;
}
}
}
close(&dr);
return remove(_filePath, _err);
}
} // namespace bx