|
@@ -15,11 +15,13 @@
|
|
|
#endif // !BX_CRT_NONE
|
|
#endif // !BX_CRT_NONE
|
|
|
|
|
|
|
|
#ifndef BX_CONFIG_CRT_FILE_READER_WRITER
|
|
#ifndef BX_CONFIG_CRT_FILE_READER_WRITER
|
|
|
-# define BX_CONFIG_CRT_FILE_READER_WRITER !(0 \
|
|
|
|
|
- || BX_CRT_NONE \
|
|
|
|
|
- )
|
|
|
|
|
|
|
+# define BX_CONFIG_CRT_FILE_READER_WRITER !BX_CRT_NONE
|
|
|
#endif // BX_CONFIG_CRT_FILE_READER_WRITER
|
|
#endif // BX_CONFIG_CRT_FILE_READER_WRITER
|
|
|
|
|
|
|
|
|
|
+#ifndef BX_CONFIG_CRT_DIRECTORY_READER
|
|
|
|
|
+# define BX_CONFIG_CRT_DIRECTORY_READER !BX_CRT_NONE
|
|
|
|
|
+#endif // BX_CONFIG_CRT_DIRECTORY_READER
|
|
|
|
|
+
|
|
|
namespace bx
|
|
namespace bx
|
|
|
{
|
|
{
|
|
|
class NoopWriterImpl : public FileWriterI
|
|
class NoopWriterImpl : public FileWriterI
|
|
@@ -554,129 +556,150 @@ namespace bx
|
|
|
return impl->write(_data, _size, _err);
|
|
return impl->write(_data, _size, _err);
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
+#if BX_CONFIG_CRT_DIRECTORY_READER
|
|
|
|
|
+
|
|
|
class DirectoryReaderImpl : public ReaderOpenI, public CloserI, public ReaderI
|
|
class DirectoryReaderImpl : public ReaderOpenI, public CloserI, public ReaderI
|
|
|
{
|
|
{
|
|
|
public:
|
|
public:
|
|
|
- ///
|
|
|
|
|
- DirectoryReaderImpl();
|
|
|
|
|
-
|
|
|
|
|
- ///
|
|
|
|
|
- virtual ~DirectoryReaderImpl();
|
|
|
|
|
|
|
+ DirectoryReaderImpl()
|
|
|
|
|
+ : m_dir(NULL)
|
|
|
|
|
+ , m_pos(0)
|
|
|
|
|
+ {
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- ///
|
|
|
|
|
- virtual bool open(const FilePath& _filePath, Error* _err) override;
|
|
|
|
|
|
|
+ virtual ~DirectoryReaderImpl()
|
|
|
|
|
+ {
|
|
|
|
|
+ close();
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- ///
|
|
|
|
|
- virtual void close() override;
|
|
|
|
|
|
|
+ virtual bool open(const FilePath& _filePath, Error* _err) override
|
|
|
|
|
+ {
|
|
|
|
|
+ BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
|
|
|
|
- ///
|
|
|
|
|
- virtual int32_t read(void* _data, int32_t _size, Error* _err) override;
|
|
|
|
|
|
|
+ m_dir = opendir(_filePath.get() );
|
|
|
|
|
|
|
|
- private:
|
|
|
|
|
- FileInfo m_cache;
|
|
|
|
|
- DIR* m_dir;
|
|
|
|
|
- int32_t m_pos;
|
|
|
|
|
- };
|
|
|
|
|
|
|
+ if (NULL == m_dir)
|
|
|
|
|
+ {
|
|
|
|
|
+ BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "DirectoryReader: Failed to open directory.");
|
|
|
|
|
+ return false;
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- DirectoryReaderImpl::DirectoryReaderImpl()
|
|
|
|
|
- : m_dir(NULL)
|
|
|
|
|
- , m_pos(0)
|
|
|
|
|
- {
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ m_pos = 0;
|
|
|
|
|
|
|
|
- DirectoryReaderImpl::~DirectoryReaderImpl()
|
|
|
|
|
- {
|
|
|
|
|
- close();
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ return true;
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- static bool fetch(FileInfo& _out, DIR* _dir)
|
|
|
|
|
- {
|
|
|
|
|
- for (;;)
|
|
|
|
|
|
|
+ virtual void close() override
|
|
|
{
|
|
{
|
|
|
- const dirent* item = readdir(_dir);
|
|
|
|
|
-
|
|
|
|
|
- if (NULL == item)
|
|
|
|
|
|
|
+ if (NULL != m_dir)
|
|
|
{
|
|
{
|
|
|
- return false;
|
|
|
|
|
|
|
+ closedir(m_dir);
|
|
|
|
|
+ m_dir = NULL;
|
|
|
}
|
|
}
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- if (0 != (item->d_type & DT_DIR) )
|
|
|
|
|
- {
|
|
|
|
|
- _out.type = FileType::Dir;
|
|
|
|
|
- _out.size = UINT64_MAX;
|
|
|
|
|
- _out.filePath.set(item->d_name);
|
|
|
|
|
- return true;
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ virtual int32_t read(void* _data, int32_t _size, Error* _err) override
|
|
|
|
|
+ {
|
|
|
|
|
+ BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
|
+
|
|
|
|
|
+ int32_t total = 0;
|
|
|
|
|
+
|
|
|
|
|
+ uint8_t* out = (uint8_t*)_data;
|
|
|
|
|
|
|
|
- if (0 != (item->d_type & DT_REG) )
|
|
|
|
|
|
|
+ while (0 < _size)
|
|
|
{
|
|
{
|
|
|
- _out.type = FileType::File;
|
|
|
|
|
- _out.size = UINT64_MAX;
|
|
|
|
|
- _out.filePath.set(item->d_name);
|
|
|
|
|
- return true;
|
|
|
|
|
|
|
+ if (0 == m_pos)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (!fetch(m_cache, m_dir) )
|
|
|
|
|
+ {
|
|
|
|
|
+ BX_ERROR_SET(_err, BX_ERROR_READERWRITER_EOF, "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;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- return false;
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ static bool fetch(FileInfo& _out, DIR* _dir)
|
|
|
|
|
+ {
|
|
|
|
|
+ for (;;)
|
|
|
|
|
+ {
|
|
|
|
|
+ const dirent* item = readdir(_dir);
|
|
|
|
|
|
|
|
- bool DirectoryReaderImpl::open(const FilePath& _filePath, Error* _err)
|
|
|
|
|
- {
|
|
|
|
|
- BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
|
|
|
+ if (NULL == item)
|
|
|
|
|
+ {
|
|
|
|
|
+ return false;
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- m_dir = opendir(_filePath.get() );
|
|
|
|
|
|
|
+ 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;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- if (NULL == m_dir)
|
|
|
|
|
- {
|
|
|
|
|
- BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "DirectoryReader: Failed to open directory.");
|
|
|
|
|
return false;
|
|
return false;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- m_pos = 0;
|
|
|
|
|
|
|
+ FileInfo m_cache;
|
|
|
|
|
+ DIR* m_dir;
|
|
|
|
|
+ int32_t m_pos;
|
|
|
|
|
+ };
|
|
|
|
|
|
|
|
- return true;
|
|
|
|
|
- }
|
|
|
|
|
|
|
+#else
|
|
|
|
|
|
|
|
- void DirectoryReaderImpl::close()
|
|
|
|
|
|
|
+ class DirectoryReaderImpl : public ReaderOpenI, public CloserI, public ReaderI
|
|
|
{
|
|
{
|
|
|
- if (NULL != m_dir)
|
|
|
|
|
|
|
+ public:
|
|
|
|
|
+ DirectoryReaderImpl()
|
|
|
{
|
|
{
|
|
|
- closedir(m_dir);
|
|
|
|
|
- m_dir = NULL;
|
|
|
|
|
}
|
|
}
|
|
|
- }
|
|
|
|
|
|
|
|
|
|
- int32_t DirectoryReaderImpl::read(void* _data, int32_t _size, Error* _err)
|
|
|
|
|
- {
|
|
|
|
|
- BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
|
-
|
|
|
|
|
- int32_t total = 0;
|
|
|
|
|
-
|
|
|
|
|
- uint8_t* out = (uint8_t*)_data;
|
|
|
|
|
|
|
+ virtual ~DirectoryReaderImpl()
|
|
|
|
|
+ {
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- for (; 0 < _size;)
|
|
|
|
|
|
|
+ virtual bool open(const FilePath& _filePath, Error* _err) override
|
|
|
{
|
|
{
|
|
|
- if (0 == m_pos)
|
|
|
|
|
- {
|
|
|
|
|
- if (!fetch(m_cache, m_dir) )
|
|
|
|
|
- {
|
|
|
|
|
- BX_ERROR_SET(_err, BX_ERROR_READERWRITER_EOF, "DirectoryReader: EOF.");
|
|
|
|
|
- return total;
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ BX_UNUSED(_filePath);
|
|
|
|
|
+ BX_ERROR_SET(_err, BX_ERROR_READERWRITER_OPEN, "DirectoryReader: Failed to open directory.");
|
|
|
|
|
+ return false;
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- 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;
|
|
|
|
|
|
|
+ virtual void close() override
|
|
|
|
|
+ {
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
- m_pos += size;
|
|
|
|
|
- m_pos %= sizeof(m_cache);
|
|
|
|
|
|
|
+ virtual int32_t read(void* _data, int32_t _size, Error* _err) override
|
|
|
|
|
+ {
|
|
|
|
|
+ BX_UNUSED(_data, _size);
|
|
|
|
|
+ BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
|
|
|
|
+ BX_ERROR_SET(_err, BX_ERROR_READERWRITER_EOF, "DirectoryReader: EOF.");
|
|
|
|
|
+ return 0;
|
|
|
}
|
|
}
|
|
|
|
|
+ };
|
|
|
|
|
|
|
|
- return total;
|
|
|
|
|
- }
|
|
|
|
|
|
|
+#endif // BX_CONFIG_CRT_DIRECTORY_READER
|
|
|
|
|
|
|
|
DirectoryReader::DirectoryReader()
|
|
DirectoryReader::DirectoryReader()
|
|
|
{
|
|
{
|