mirror of
https://github.com/bkaradzic/bx.git
synced 2026-02-17 20:52:37 +01:00
607 lines
12 KiB
C++
607 lines
12 KiB
C++
/*
|
|
* Copyright 2010-2018 Branimir Karadzic. All rights reserved.
|
|
* License: https://github.com/bkaradzic/bx#license-bsd-2-clause
|
|
*/
|
|
|
|
#include "bx_p.h"
|
|
#include <bx/file.h>
|
|
|
|
#if BX_CRT_NONE
|
|
# include "crt0.h"
|
|
#else
|
|
# include <stdio.h>
|
|
# include <sys/stat.h>
|
|
#endif // !BX_CRT_NONE
|
|
|
|
#ifndef BX_CONFIG_CRT_FILE_READER_WRITER
|
|
# define BX_CONFIG_CRT_FILE_READER_WRITER !(0 \
|
|
|| BX_CRT_NONE \
|
|
)
|
|
#endif // BX_CONFIG_CRT_FILE_READER_WRITER
|
|
|
|
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, _size, _err);
|
|
return 0;
|
|
}
|
|
};
|
|
|
|
#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_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_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
if (NULL != m_file)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_ALREADY_OPEN, "FileReader: File is already open.");
|
|
return false;
|
|
}
|
|
|
|
m_file = fopen(_filePath.get(), "rb");
|
|
if (NULL == m_file)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "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_CHECK(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_CHECK(NULL != m_file, "Reader/Writer file is not open.");
|
|
BX_CHECK(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, BX_ERROR_READERWRITER_EOF, "FileReader: EOF.");
|
|
}
|
|
else if (0 != ferror(m_file) )
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_READ, "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_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
if (NULL != m_file)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_ALREADY_OPEN, "FileReader: File is already open.");
|
|
return false;
|
|
}
|
|
|
|
m_file = fopen(_filePath.get(), _append ? "ab" : "wb");
|
|
|
|
if (NULL == m_file)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "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_CHECK(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_CHECK(NULL != m_file, "Reader/Writer file is not open.");
|
|
BX_CHECK(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, BX_ERROR_READERWRITER_WRITE, "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_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
if (0 != m_fd)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_ALREADY_OPEN, "FileReader: File is already open.");
|
|
return false;
|
|
}
|
|
|
|
m_fd = crt0::open(_filePath.get(), crt0::Open::Read, 0);
|
|
|
|
if (0 >= m_fd)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "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_CHECK(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_CHECK(0 != m_fd, "Reader/Writer file is not open.");
|
|
BX_CHECK(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, BX_ERROR_READERWRITER_EOF, "FileReader: EOF.");
|
|
// }
|
|
// else if (0 != ferror(m_file) )
|
|
// {
|
|
// BX_ERROR_SET(_err, BX_ERROR_READERWRITER_READ, "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_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
if (0 != m_fd)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_ALREADY_OPEN, "FileReader: File is already open.");
|
|
return false;
|
|
}
|
|
|
|
m_fd = crt0::open(_filePath.get(), _append ? crt0::Open::Append : crt0::Open::Write, 0600);
|
|
|
|
if (0 >= m_fd)
|
|
{
|
|
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "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_CHECK(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_CHECK(0 != m_fd, "Reader/Writer file is not open.");
|
|
BX_CHECK(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, BX_ERROR_READERWRITER_WRITE, "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);
|
|
}
|
|
|
|
bool stat(const FilePath& _filePath, FileInfo& _outFileInfo)
|
|
{
|
|
#if BX_CRT_NONE
|
|
BX_UNUSED(_filePath, _outFileInfo);
|
|
return false;
|
|
#else
|
|
_outFileInfo.m_size = 0;
|
|
_outFileInfo.m_type = FileInfo::Count;
|
|
|
|
# if BX_COMPILER_MSVC
|
|
struct ::_stat64 st;
|
|
int32_t result = ::_stat64(_filePath.get(), &st);
|
|
|
|
if (0 != result)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (0 != (st.st_mode & _S_IFREG) )
|
|
{
|
|
_outFileInfo.m_type = FileInfo::Regular;
|
|
}
|
|
else if (0 != (st.st_mode & _S_IFDIR) )
|
|
{
|
|
_outFileInfo.m_type = FileInfo::Directory;
|
|
}
|
|
# else
|
|
struct ::stat st;
|
|
int32_t result = ::stat(_filePath.get(), &st);
|
|
if (0 != result)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (0 != (st.st_mode & S_IFREG) )
|
|
{
|
|
_outFileInfo.m_type = FileInfo::Regular;
|
|
}
|
|
else if (0 != (st.st_mode & S_IFDIR) )
|
|
{
|
|
_outFileInfo.m_type = FileInfo::Directory;
|
|
}
|
|
# endif // BX_COMPILER_MSVC
|
|
|
|
_outFileInfo.m_size = st.st_size;
|
|
|
|
return true;
|
|
#endif // BX_CRT_NONE
|
|
}
|
|
|
|
} // namespace bx
|