FileHandle refactor

This commit is contained in:
Jack Andersen 2017-01-23 21:40:47 -10:00
parent b5f5104877
commit 408ac735b4
7 changed files with 229 additions and 196 deletions

View File

@ -17,15 +17,16 @@
namespace kabufuda namespace kabufuda
{ {
class IFileHandle class FileHandle
{ {
protected: friend class Card;
uint32_t idx; uint32_t idx = 0;
IFileHandle() = default; int32_t offset = 0;
IFileHandle(uint32_t idx) : idx(idx) {} FileHandle(uint32_t idx) : idx(idx) {}
public: public:
FileHandle() = default;
uint32_t getFileNo() const { return idx; } uint32_t getFileNo() const { return idx; }
virtual ~IFileHandle(); operator bool() const { return getFileNo() != 0; }
}; };
enum class ECardResult enum class ECardResult
@ -136,7 +137,7 @@ class Card
void _swapEndian(); void _swapEndian();
void _updateDirAndBat(); void _updateDirAndBat();
void _updateChecksum(); void _updateChecksum();
File* _fileFromHandle(const std::unique_ptr<IFileHandle>& fh) const; File* _fileFromHandle(const FileHandle& fh) const;
void _deleteFile(File& f); void _deleteFile(File& f);
public: public:
@ -147,8 +148,8 @@ public:
*/ */
Card(const Card& other) = delete; Card(const Card& other) = delete;
Card& operator=(const Card& other) = delete; Card& operator=(const Card& other) = delete;
Card(Card&& other) = default; Card(Card&& other);
Card& operator=(Card&& other) = default; Card& operator=(Card&& other);
/** /**
* @brief Card * @brief Card
@ -163,46 +164,53 @@ public:
* @brief openFile * @brief openFile
* @param filename * @param filename
*/ */
ECardResult openFile(const char* filename, std::unique_ptr<IFileHandle>& handleOut); ECardResult openFile(const char* filename, FileHandle& handleOut);
/** /**
* @brief openFile * @brief openFile
* @param fileno * @param fileno
*/ */
ECardResult openFile(uint32_t fileno, std::unique_ptr<IFileHandle>& handleOut); ECardResult openFile(uint32_t fileno, FileHandle& handleOut);
/** /**
* @brief createFile * @brief createFile
* @param filename * @param filename
* @return * @return
*/ */
ECardResult createFile(const char* filename, size_t size, std::unique_ptr<IFileHandle>& handleOut); ECardResult createFile(const char* filename, size_t size, FileHandle& handleOut);
/**
* @brief closeFile
* @param fh FileHandle to close
* @return
*/
ECardResult closeFile(FileHandle& fh);
/** /**
* @brief firstFile * @brief firstFile
* @return * @return
*/ */
std::unique_ptr<IFileHandle> firstFile(); FileHandle firstFile();
/** /**
* @brief nextFile * @brief nextFile
* @param cur * @param cur
* @return * @return
*/ */
std::unique_ptr<IFileHandle> nextFile(const std::unique_ptr<IFileHandle>& cur); FileHandle nextFile(const FileHandle& cur);
/** /**
* @brief getFilename * @brief getFilename
* @param fh * @param fh
* @return * @return
*/ */
const char* getFilename(const std::unique_ptr<IFileHandle>& fh); const char* getFilename(const FileHandle& fh);
/** /**
* @brief deleteFile * @brief deleteFile
* @param fh * @param fh
*/ */
void deleteFile(const std::unique_ptr<IFileHandle>& fh); void deleteFile(const FileHandle& fh);
/** /**
* @brief deleteFile * @brief deleteFile
@ -229,7 +237,7 @@ public:
* @param buf * @param buf
* @param size * @param size
*/ */
void write(const std::unique_ptr<IFileHandle>& fh, const void* buf, size_t size); ECardResult write(FileHandle& fh, const void* buf, size_t size);
/** /**
* @brief read * @brief read
@ -237,7 +245,7 @@ public:
* @param dst * @param dst
* @param size * @param size
*/ */
void read(const std::unique_ptr<IFileHandle>& fh, void* dst, size_t size); ECardResult read(FileHandle& fh, void* dst, size_t size);
/** /**
* @brief seek * @brief seek
@ -245,56 +253,56 @@ public:
* @param pos * @param pos
* @param whence * @param whence
*/ */
void seek(const std::unique_ptr<IFileHandle>& fh, int32_t pos, SeekOrigin whence); void seek(FileHandle& fh, int32_t pos, SeekOrigin whence);
/** /**
* @brief Returns the current offset of the specified file * @brief Returns the current offset of the specified file
* @param fh The file to retrieve the offset from * @param fh The file to retrieve the offset from
* @return The offset or -1 if an invalid handle is passed * @return The offset or -1 if an invalid handle is passed
*/ */
int32_t tell(const std::unique_ptr<IFileHandle>& fh); int32_t tell(const FileHandle& fh);
/** /**
* @brief setPublic * @brief setPublic
* @param fh * @param fh
* @param pub * @param pub
*/ */
void setPublic(const std::unique_ptr<IFileHandle>& fh, bool pub); void setPublic(const FileHandle& fh, bool pub);
/** /**
* @brief isPublic * @brief isPublic
* @param fh * @param fh
* @return * @return
*/ */
bool isPublic(const std::unique_ptr<IFileHandle>& fh) const; bool isPublic(const FileHandle& fh) const;
/** /**
* @brief setCanCopy * @brief setCanCopy
* @param fh * @param fh
* @param copy * @param copy
*/ */
void setCanCopy(const std::unique_ptr<IFileHandle>& fh, bool copy) const; void setCanCopy(const FileHandle& fh, bool copy) const;
/** /**
* @brief canCopy * @brief canCopy
* @param fh * @param fh
* @return * @return
*/ */
bool canCopy(const std::unique_ptr<IFileHandle>& fh) const; bool canCopy(const FileHandle& fh) const;
/** /**
* @brief setCanMove * @brief setCanMove
* @param fh * @param fh
* @param move * @param move
*/ */
void setCanMove(const std::unique_ptr<IFileHandle>& fh, bool move); void setCanMove(const FileHandle& fh, bool move);
/** /**
* @brief canMove * @brief canMove
* @param fh * @param fh
* @return * @return
*/ */
bool canMove(const std::unique_ptr<IFileHandle>& fh) const; bool canMove(const FileHandle& fh) const;
/** /**
* @brief getStatus * @brief getStatus
@ -302,7 +310,15 @@ public:
* @param statOut Structure to fill with file stat * @param statOut Structure to fill with file stat
* @return NOFILE or READY * @return NOFILE or READY
*/ */
ECardResult getStatus(const std::unique_ptr<IFileHandle>& fh, CardStat& statOut) const; ECardResult getStatus(const FileHandle& fh, CardStat& statOut) const;
/**
* @brief getStatus
* @param fileNo Number of requested file
* @param statOut Structure to fill with file stat
* @return NOFILE or READY
*/
ECardResult getStatus(uint32_t fileNo, CardStat& statOut) const;
/** /**
* @brief setStatus * @brief setStatus
@ -310,49 +326,57 @@ public:
* @param statOut Structure to access for file stat * @param statOut Structure to access for file stat
* @return NOFILE or READY * @return NOFILE or READY
*/ */
ECardResult setStatus(const std::unique_ptr<IFileHandle>& fh, const CardStat& stat); ECardResult setStatus(const FileHandle& fh, const CardStat& stat);
/**
* @brief setStatus
* @param fileNo Number of requested file
* @param statOut Structure to access for file stat
* @return NOFILE or READY
*/
ECardResult setStatus(uint32_t fileNo, const CardStat& stat);
/** /**
* @brief gameId * @brief gameId
* @param fh * @param fh
* @return * @return
*/ */
const char* gameId(const std::unique_ptr<IFileHandle>& fh) const; const char* gameId(const FileHandle& fh) const;
/** /**
* @brief maker * @brief maker
* @param fh * @param fh
* @return * @return
*/ */
const char* maker(const std::unique_ptr<IFileHandle>& fh) const; const char* maker(const FileHandle& fh) const;
/** /**
* @brief setBannerFormat * @brief setBannerFormat
* @param fh * @param fh
* @param fmt * @param fmt
*/ */
void setBannerFormat(const std::unique_ptr<IFileHandle>& fh, EImageFormat fmt); void setBannerFormat(const FileHandle& fh, EImageFormat fmt);
/** /**
* @brief bannerFormat * @brief bannerFormat
* @param fh * @param fh
* @return * @return
*/ */
EImageFormat bannerFormat(const std::unique_ptr<IFileHandle>& fh) const; EImageFormat bannerFormat(const FileHandle& fh) const;
/** /**
* @brief setIconAnimationType * @brief setIconAnimationType
* @param fh * @param fh
* @param type * @param type
*/ */
void setIconAnimationType(const std::unique_ptr<IFileHandle>& fh, EAnimationType type); void setIconAnimationType(const FileHandle& fh, EAnimationType type);
/** /**
* @brief iconAnimationType * @brief iconAnimationType
* @param fh * @param fh
* @return * @return
*/ */
EAnimationType iconAnimationType(const std::unique_ptr<IFileHandle>& fh) const; EAnimationType iconAnimationType(const FileHandle& fh) const;
/** /**
* @brief setIconFormat * @brief setIconFormat
@ -360,7 +384,7 @@ public:
* @param idx * @param idx
* @param fmt * @param fmt
*/ */
void setIconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, EImageFormat fmt); void setIconFormat(const FileHandle& fh, uint32_t idx, EImageFormat fmt);
/** /**
* @brief iconFormat * @brief iconFormat
@ -368,7 +392,7 @@ public:
* @param idx * @param idx
* @return * @return
*/ */
EImageFormat iconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t idx) const; EImageFormat iconFormat(const FileHandle& fh, uint32_t idx) const;
/** /**
* @brief setIconSpeed * @brief setIconSpeed
@ -376,7 +400,7 @@ public:
* @param idx * @param idx
* @param speed * @param speed
*/ */
void setIconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, EAnimationSpeed speed); void setIconSpeed(const FileHandle& fh, uint32_t idx, EAnimationSpeed speed);
/** /**
* @brief iconSpeed * @brief iconSpeed
@ -384,35 +408,35 @@ public:
* @param idx * @param idx
* @return * @return
*/ */
EAnimationSpeed iconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t idx) const; EAnimationSpeed iconSpeed(const FileHandle& fh, uint32_t idx) const;
/** /**
* @brief setImageAddress * @brief setImageAddress
* @param fh * @param fh
* @param addr * @param addr
*/ */
void setImageAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t addr); void setImageAddress(const FileHandle& fh, uint32_t addr);
/** /**
* @brief imageAddress * @brief imageAddress
* @param fh * @param fh
* @return * @return
*/ */
int32_t imageAddress(const std::unique_ptr<IFileHandle>& fh) const; int32_t imageAddress(const FileHandle& fh) const;
/** /**
* @brief setCommentAddress * @brief setCommentAddress
* @param fh * @param fh
* @param addr * @param addr
*/ */
void setCommentAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t addr); void setCommentAddress(const FileHandle& fh, uint32_t addr);
/** /**
* @brief commentAddress * @brief commentAddress
* @param fh * @param fh
* @return * @return
*/ */
int32_t commentAddress(const std::unique_ptr<IFileHandle>& fh) const; int32_t commentAddress(const FileHandle& fh) const;
/** /**
* @brief Copies a file from the current Card instance to a specified Card instance * @brief Copies a file from the current Card instance to a specified Card instance
@ -420,7 +444,7 @@ public:
* @param dest The destination Card instance * @param dest The destination Card instance
* @return True if successful, false otherwise * @return True if successful, false otherwise
*/ */
bool copyFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest); bool copyFileTo(FileHandle& fh, Card& dest);
/** /**
* @brief moveFileTo * @brief moveFileTo
@ -428,7 +452,7 @@ public:
* @param dest * @param dest
* @return * @return
*/ */
bool moveFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest); bool moveFileTo(FileHandle& fh, Card& dest);
/** /**
* @brief Sets the current game, if not null any openFile requests will only return files that match this game * @brief Sets the current game, if not null any openFile requests will only return files that match this game

View File

@ -358,7 +358,6 @@ static inline int Stat(const SystemChar* path, Sstat* statOut)
* @param checksumInv * @param checksumInv
*/ */
void calculateChecksumBE(const uint16_t* data, size_t len, uint16_t* checksum, uint16_t* checksumInv); void calculateChecksumBE(const uint16_t* data, size_t len, uint16_t* checksum, uint16_t* checksumInv);
void calculateChecksumLE(const uint16_t* data, size_t len, uint16_t* checksum, uint16_t* checksumInv);
} }
#endif // __KABU_UTIL_HPP__ #endif // __KABU_UTIL_HPP__

View File

@ -16,14 +16,19 @@ void BlockAllocationTable::swapEndian()
void BlockAllocationTable::updateChecksum() void BlockAllocationTable::updateChecksum()
{ {
calculateChecksumLE(reinterpret_cast<uint16_t*>(__raw + 4), 0xFFE, &m_checksum, &m_checksumInv); swapEndian();
calculateChecksumBE(reinterpret_cast<uint16_t*>(__raw + 4), 0xFFE, &m_checksum, &m_checksumInv);
swapEndian();
} }
bool BlockAllocationTable::valid() const bool BlockAllocationTable::valid() const
{ {
uint16_t ckSum, ckSumInv; uint16_t ckSum, ckSumInv;
calculateChecksumLE(reinterpret_cast<const uint16_t*>(__raw + 4), 0xFFE, &ckSum, &ckSumInv); const_cast<BlockAllocationTable&>(*this).swapEndian();
return (ckSum == m_checksum && ckSumInv == m_checksumInv); calculateChecksumBE(reinterpret_cast<const uint16_t*>(__raw + 4), 0xFFE, &ckSum, &ckSumInv);
bool res = (ckSum == m_checksum && ckSumInv == m_checksumInv);
const_cast<BlockAllocationTable&>(*this).swapEndian();
return res;
} }
BlockAllocationTable::BlockAllocationTable(uint32_t blockCount) BlockAllocationTable::BlockAllocationTable(uint32_t blockCount)

View File

@ -10,21 +10,6 @@ namespace kabufuda
#define ROUND_UP_8192(val) (((val) + 8191) & ~8191) #define ROUND_UP_8192(val) (((val) + 8191) & ~8191)
IFileHandle::~IFileHandle() {}
class FileHandle : public IFileHandle
{
friend class Card;
int32_t offset = 0;
public:
FileHandle() = default;
FileHandle(uint32_t idx) : IFileHandle(idx) {}
virtual ~FileHandle();
};
FileHandle::~FileHandle() {}
void Card::_swapEndian() void Card::_swapEndian()
{ {
m_formatTime = SBig(m_formatTime); m_formatTime = SBig(m_formatTime);
@ -41,6 +26,44 @@ void Card::_swapEndian()
Card::Card() { memset(__raw, 0xFF, BlockSize); } Card::Card() { memset(__raw, 0xFF, BlockSize); }
Card::Card(Card&& other)
{
memmove(__raw, other.__raw, BlockSize);
m_filename = std::move(other.m_filename);
m_fileHandle = other.m_fileHandle;
other.m_fileHandle = nullptr;
m_dirs[0] = std::move(other.m_dirs[0]);
m_dirs[1] = std::move(other.m_dirs[1]);
m_bats[0] = std::move(other.m_bats[0]);
m_bats[1] = std::move(other.m_bats[1]);
m_currentDir = other.m_currentDir;
m_currentBat = other.m_currentBat;
m_maxBlock = other.m_maxBlock;
memmove(m_game, other.m_game, 5);
memmove(m_maker, other.m_maker, 3);
}
Card& Card::operator=(Card&& other)
{
memmove(__raw, other.__raw, BlockSize);
m_filename = std::move(other.m_filename);
m_fileHandle = other.m_fileHandle;
other.m_fileHandle = nullptr;
m_dirs[0] = std::move(other.m_dirs[0]);
m_dirs[1] = std::move(other.m_dirs[1]);
m_bats[0] = std::move(other.m_bats[0]);
m_bats[1] = std::move(other.m_bats[1]);
m_currentDir = other.m_currentDir;
m_currentBat = other.m_currentBat;
m_maxBlock = other.m_maxBlock;
memmove(m_game, other.m_game, 5);
memmove(m_maker, other.m_maker, 3);
return *this;
}
Card::Card(const SystemString& filename, const char* game, const char* maker) : m_filename(filename) Card::Card(const SystemString& filename, const char* game, const char* maker) : m_filename(filename)
{ {
memset(__raw, 0xFF, BlockSize); memset(__raw, 0xFF, BlockSize);
@ -97,31 +120,30 @@ Card::~Card()
commit(); commit();
if (m_fileHandle) if (m_fileHandle)
fclose(m_fileHandle); fclose(m_fileHandle);
m_fileHandle = nullptr;
} }
ECardResult Card::openFile(const char* filename, std::unique_ptr<IFileHandle>& handleOut) ECardResult Card::openFile(const char* filename, FileHandle& handleOut)
{ {
handleOut.reset(); handleOut = {};
File* f = m_dirs[m_currentDir].getFile(m_game, m_maker, filename); File* f = m_dirs[m_currentDir].getFile(m_game, m_maker, filename);
if (!f || f->m_game[0] == 0xFF) if (!f || f->m_game[0] == 0xFF)
return ECardResult::NOFILE; return ECardResult::NOFILE;
int32_t idx = m_dirs[m_currentDir].indexForFile(f); int32_t idx = m_dirs[m_currentDir].indexForFile(f);
if (idx != -1) if (idx != -1)
{ {
handleOut = std::make_unique<FileHandle>(idx); handleOut = FileHandle(idx);
return ECardResult::READY; return ECardResult::READY;
} }
return ECardResult::FATAL_ERROR; return ECardResult::FATAL_ERROR;
} }
ECardResult Card::openFile(uint32_t fileno, std::unique_ptr<IFileHandle>& handleOut) ECardResult Card::openFile(uint32_t fileno, FileHandle& handleOut)
{ {
handleOut.reset(); handleOut = {};
File* f = m_dirs[m_currentDir].getFile(fileno); File* f = m_dirs[m_currentDir].getFile(fileno);
if (!f || f->m_game[0] == 0xFF) if (!f || f->m_game[0] == 0xFF)
return ECardResult::NOFILE; return ECardResult::NOFILE;
handleOut = std::make_unique<FileHandle>(fileno); handleOut = FileHandle(fileno);
return ECardResult::READY; return ECardResult::READY;
} }
@ -140,26 +162,20 @@ void Card::_updateDirAndBat()
void Card::_updateChecksum() void Card::_updateChecksum()
{ {
calculateChecksumLE(reinterpret_cast<uint16_t*>(__raw), 0xFE, &m_checksum, &m_checksumInv); _swapEndian();
calculateChecksumBE(reinterpret_cast<uint16_t*>(__raw), 0xFE, &m_checksum, &m_checksumInv);
_swapEndian();
} }
File* Card::_fileFromHandle(const std::unique_ptr<IFileHandle>& fh) const File* Card::_fileFromHandle(const FileHandle& fh) const
{ {
if (!fh) return const_cast<Directory&>(m_dirs[m_currentDir]).getFile(fh.idx);
return nullptr;
FileHandle* handle = dynamic_cast<FileHandle*>(fh.get());
if (!handle)
return nullptr;
File* file = const_cast<Directory&>(m_dirs[m_currentDir]).getFile(handle->idx);
return file;
} }
ECardResult Card::createFile(const char* filename, size_t size, ECardResult Card::createFile(const char* filename, size_t size,
std::unique_ptr<IFileHandle>& handleOut) FileHandle& handleOut)
{ {
handleOut.reset(); handleOut = {};
if (size <= 0) if (size <= 0)
return ECardResult::FATAL_ERROR; return ECardResult::FATAL_ERROR;
@ -182,35 +198,37 @@ ECardResult Card::createFile(const char* filename, size_t size,
f->m_firstBlock = block; f->m_firstBlock = block;
f->m_blockCount = neededBlocks; f->m_blockCount = neededBlocks;
handleOut = std::make_unique<FileHandle>(m_dirs[m_currentDir].indexForFile(f)); handleOut = FileHandle(m_dirs[m_currentDir].indexForFile(f));
return ECardResult::READY; return ECardResult::READY;
} }
return ECardResult::FATAL_ERROR; return ECardResult::FATAL_ERROR;
} }
std::unique_ptr<IFileHandle> Card::firstFile() ECardResult Card::closeFile(FileHandle& fh)
{
fh.offset = 0;
return ECardResult::READY;
}
FileHandle Card::firstFile()
{ {
File* f = m_dirs[m_currentDir].getFirstNonFreeFile(0, m_game, m_maker); File* f = m_dirs[m_currentDir].getFirstNonFreeFile(0, m_game, m_maker);
if (f) if (f)
return std::make_unique<FileHandle>(m_dirs[m_currentDir].indexForFile(f)); return FileHandle(m_dirs[m_currentDir].indexForFile(f));
return nullptr; return {};
} }
std::unique_ptr<IFileHandle> Card::nextFile(const std::unique_ptr<IFileHandle>& cur) FileHandle Card::nextFile(const FileHandle& cur)
{ {
FileHandle* handle = dynamic_cast<FileHandle*>(cur.get()); File* next = m_dirs[m_currentDir].getFirstNonFreeFile(cur.idx + 1, m_game, m_maker);
if (!handle)
return nullptr;
File* next = m_dirs[m_currentDir].getFirstNonFreeFile(handle->idx + 1, m_game, m_maker);
if (!next) if (!next)
return nullptr; return {};
return std::make_unique<FileHandle>(m_dirs[m_currentDir].indexForFile(next)); return FileHandle(m_dirs[m_currentDir].indexForFile(next));
} }
const char* Card::getFilename(const std::unique_ptr<IFileHandle>& fh) const char* Card::getFilename(const FileHandle& fh)
{ {
File* f = _fileFromHandle(fh); File* f = _fileFromHandle(fh);
if (!f) if (!f)
@ -231,13 +249,10 @@ void Card::_deleteFile(File& f)
f = File(); f = File();
} }
void Card::deleteFile(const std::unique_ptr<IFileHandle>& fh) void Card::deleteFile(const FileHandle& fh)
{ {
_updateDirAndBat(); _updateDirAndBat();
if (!fh) _deleteFile(*m_dirs[m_currentDir].getFile(fh.idx));
return;
FileHandle* f = dynamic_cast<FileHandle*>(fh.get());
_deleteFile(*m_dirs[m_currentDir].getFile(f->idx));
} }
ECardResult Card::deleteFile(const char* filename) ECardResult Card::deleteFile(const char* filename)
@ -276,42 +291,39 @@ ECardResult Card::renameFile(const char* oldName, const char* newName)
return ECardResult::READY; return ECardResult::READY;
} }
void Card::write(const std::unique_ptr<IFileHandle>& fh, const void* buf, size_t size) ECardResult Card::write(FileHandle& fh, const void* buf, size_t size)
{ {
if (!fh)
return;
if (m_fileHandle) if (m_fileHandle)
{ {
FileHandle* f = dynamic_cast<FileHandle*>(fh.get()); File* file = m_dirs[m_currentDir].getFile(fh.idx);
File* file = m_dirs[m_currentDir].getFile(f->idx);
if (!file) if (!file)
return; return ECardResult::NOFILE;
/* Block handling is a little different from cache handling, /* Block handling is a little different from cache handling,
* since each block can be in an arbitrary location we must * since each block can be in an arbitrary location we must
* first find our starting block. * first find our starting block.
*/ */
const uint16_t blockId = uint16_t(f->offset / BlockSize); const uint16_t blockId = uint16_t(fh.offset / BlockSize);
uint16_t block = file->m_firstBlock; uint16_t block = file->m_firstBlock;
for (uint16_t i = 0; i < blockId; i++) for (uint16_t i = 0; i < blockId; i++)
block = m_bats[m_currentBat].getNextBlock(block); block = m_bats[m_currentBat].getNextBlock(block);
const uint8_t* tmpBuf = reinterpret_cast<const uint8_t*>(buf); const uint8_t* tmpBuf = reinterpret_cast<const uint8_t*>(buf);
uint16_t curBlock = block; uint16_t curBlock = block;
uint32_t blockOffset = f->offset % BlockSize; uint32_t blockOffset = fh.offset % BlockSize;
size_t rem = size; size_t rem = size;
while (rem) while (rem)
{ {
if (curBlock == 0xFFFF) if (curBlock == 0xFFFF)
return; return ECardResult::NOFILE;
size_t cacheSize = rem; size_t cacheSize = rem;
if (cacheSize + blockOffset > BlockSize) if (cacheSize + blockOffset > BlockSize)
cacheSize = BlockSize - blockOffset; cacheSize = BlockSize - blockOffset;
uint32_t offset = (curBlock * BlockSize) + blockOffset; uint32_t offset = (curBlock * BlockSize) + blockOffset;
fseek(m_fileHandle, offset, SEEK_SET); fseek(m_fileHandle, offset, SEEK_SET);
fwrite(tmpBuf, 1, cacheSize, m_fileHandle); if (fwrite(tmpBuf, 1, cacheSize, m_fileHandle) != cacheSize)
return ECardResult::IOERROR;
tmpBuf += cacheSize; tmpBuf += cacheSize;
rem -= cacheSize; rem -= cacheSize;
blockOffset += cacheSize; blockOffset += cacheSize;
@ -321,45 +333,44 @@ void Card::write(const std::unique_ptr<IFileHandle>& fh, const void* buf, size_t
blockOffset = 0; blockOffset = 0;
} }
} }
f->offset += size; fh.offset += size;
} }
return ECardResult::READY;
} }
void Card::read(const std::unique_ptr<IFileHandle>& fh, void* dst, size_t size) ECardResult Card::read(FileHandle& fh, void* dst, size_t size)
{ {
if (!fh)
return;
if (m_fileHandle) if (m_fileHandle)
{ {
FileHandle* f = dynamic_cast<FileHandle*>(fh.get()); File* file = m_dirs[m_currentDir].getFile(fh.idx);
File* file = m_dirs[m_currentDir].getFile(f->idx);
if (!file) if (!file)
return; return ECardResult::NOFILE;
/* Block handling is a little different from cache handling, /* Block handling is a little different from cache handling,
* since each block can be in an arbitrary location we must * since each block can be in an arbitrary location we must
* first find our starting block. * first find our starting block.
*/ */
const uint16_t blockId = uint16_t(f->offset / BlockSize); const uint16_t blockId = uint16_t(fh.offset / BlockSize);
uint16_t block = file->m_firstBlock; uint16_t block = file->m_firstBlock;
for (uint16_t i = 0; i < blockId; i++) for (uint16_t i = 0; i < blockId; i++)
block = m_bats[m_currentBat].getNextBlock(block); block = m_bats[m_currentBat].getNextBlock(block);
uint8_t* tmpBuf = reinterpret_cast<uint8_t*>(dst); uint8_t* tmpBuf = reinterpret_cast<uint8_t*>(dst);
uint16_t curBlock = block; uint16_t curBlock = block;
uint32_t blockOffset = f->offset % BlockSize; uint32_t blockOffset = fh.offset % BlockSize;
size_t rem = size; size_t rem = size;
while (rem) while (rem)
{ {
if (curBlock == 0xFFFF) if (curBlock == 0xFFFF)
return; return ECardResult::NOFILE;
size_t cacheSize = rem; size_t cacheSize = rem;
if (cacheSize + blockOffset > BlockSize) if (cacheSize + blockOffset > BlockSize)
cacheSize = BlockSize - blockOffset; cacheSize = BlockSize - blockOffset;
uint32_t offset = (curBlock * BlockSize) + blockOffset; uint32_t offset = (curBlock * BlockSize) + blockOffset;
fseek(m_fileHandle, offset, SEEK_SET); fseek(m_fileHandle, offset, SEEK_SET);
fread(tmpBuf, 1, cacheSize, m_fileHandle); if (fread(tmpBuf, 1, cacheSize, m_fileHandle) != cacheSize)
return ECardResult::IOERROR;
tmpBuf += cacheSize; tmpBuf += cacheSize;
rem -= cacheSize; rem -= cacheSize;
blockOffset += cacheSize; blockOffset += cacheSize;
@ -369,45 +380,38 @@ void Card::read(const std::unique_ptr<IFileHandle>& fh, void* dst, size_t size)
blockOffset = 0; blockOffset = 0;
} }
} }
f->offset += size; fh.offset += size;
} }
return ECardResult::READY;
} }
void Card::seek(const std::unique_ptr<IFileHandle>& fh, int32_t pos, SeekOrigin whence) void Card::seek(FileHandle& fh, int32_t pos, SeekOrigin whence)
{ {
if (!fh) File* file = m_dirs[m_currentDir].getFile(fh.idx);
return;
FileHandle* f = dynamic_cast<FileHandle*>(fh.get());
File* file = m_dirs[m_currentDir].getFile(f->idx);
if (!file) if (!file)
return; return;
switch (whence) switch (whence)
{ {
case SeekOrigin::Begin: case SeekOrigin::Begin:
f->offset = pos; fh.offset = pos;
break; break;
case SeekOrigin::Current: case SeekOrigin::Current:
f->offset += pos; fh.offset += pos;
break; break;
case SeekOrigin::End: case SeekOrigin::End:
f->offset = int32_t(file->m_blockCount * BlockSize) - pos; fh.offset = int32_t(file->m_blockCount * BlockSize) - pos;
break; break;
} }
} }
int32_t Card::tell(const std::unique_ptr<IFileHandle>& fh) int32_t Card::tell(const FileHandle& fh)
{ {
if (!fh) return fh.offset;
return -1;
FileHandle* handle = dynamic_cast<FileHandle*>(fh.get());
if (!handle)
return -1;
return handle->offset;
} }
void Card::setPublic(const std::unique_ptr<IFileHandle>& fh, bool pub) void Card::setPublic(const FileHandle& fh, bool pub)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -419,7 +423,7 @@ void Card::setPublic(const std::unique_ptr<IFileHandle>& fh, bool pub)
file->m_permissions &= ~EPermissions::Public; file->m_permissions &= ~EPermissions::Public;
} }
bool Card::isPublic(const std::unique_ptr<IFileHandle>& fh) const bool Card::isPublic(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -428,7 +432,7 @@ bool Card::isPublic(const std::unique_ptr<IFileHandle>& fh) const
return bool(file->m_permissions & EPermissions::Public); return bool(file->m_permissions & EPermissions::Public);
} }
void Card::setCanCopy(const std::unique_ptr<IFileHandle>& fh, bool copy) const void Card::setCanCopy(const FileHandle& fh, bool copy) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -440,7 +444,7 @@ void Card::setCanCopy(const std::unique_ptr<IFileHandle>& fh, bool copy) const
file->m_permissions |= EPermissions::NoCopy; file->m_permissions |= EPermissions::NoCopy;
} }
bool Card::canCopy(const std::unique_ptr<IFileHandle>& fh) const bool Card::canCopy(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -449,7 +453,7 @@ bool Card::canCopy(const std::unique_ptr<IFileHandle>& fh) const
return !bool(file->m_permissions & EPermissions::NoCopy); return !bool(file->m_permissions & EPermissions::NoCopy);
} }
void Card::setCanMove(const std::unique_ptr<IFileHandle>& fh, bool move) void Card::setCanMove(const FileHandle& fh, bool move)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -461,7 +465,7 @@ void Card::setCanMove(const std::unique_ptr<IFileHandle>& fh, bool move)
file->m_permissions |= EPermissions::NoMove; file->m_permissions |= EPermissions::NoMove;
} }
bool Card::canMove(const std::unique_ptr<IFileHandle>& fh) const bool Card::canMove(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -511,9 +515,14 @@ static uint32_t TlutSize(EImageFormat fmt)
} }
} }
ECardResult Card::getStatus(const std::unique_ptr<IFileHandle>& fh, CardStat& statOut) const ECardResult Card::getStatus(const FileHandle& fh, CardStat& statOut) const
{ {
File* file = _fileFromHandle(fh); return getStatus(fh.idx, statOut);
}
ECardResult Card::getStatus(uint32_t fileNo, CardStat& statOut) const
{
File* file = const_cast<Directory&>(m_dirs[m_currentDir]).getFile(fileNo);
if (!file || file->m_game[0] == 0xFF) if (!file || file->m_game[0] == 0xFF)
return ECardResult::NOFILE; return ECardResult::NOFILE;
@ -567,9 +576,14 @@ ECardResult Card::getStatus(const std::unique_ptr<IFileHandle>& fh, CardStat& st
return ECardResult::READY; return ECardResult::READY;
} }
ECardResult Card::setStatus(const std::unique_ptr<IFileHandle>& fh, const CardStat& stat) ECardResult Card::setStatus(const FileHandle& fh, const CardStat& stat)
{ {
File* file = _fileFromHandle(fh); return setStatus(fh.idx, stat);
}
ECardResult Card::setStatus(uint32_t fileNo, const CardStat& stat)
{
File* file = m_dirs[m_currentDir].getFile(fileNo);
if (!file || file->m_game[0] == 0xFF) if (!file || file->m_game[0] == 0xFF)
return ECardResult::NOFILE; return ECardResult::NOFILE;
@ -582,7 +596,7 @@ ECardResult Card::setStatus(const std::unique_ptr<IFileHandle>& fh, const CardSt
return ECardResult::READY; return ECardResult::READY;
} }
const char* Card::gameId(const std::unique_ptr<IFileHandle>& fh) const const char* Card::gameId(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -590,7 +604,7 @@ const char* Card::gameId(const std::unique_ptr<IFileHandle>& fh) const
return reinterpret_cast<const char*>(file->m_game); return reinterpret_cast<const char*>(file->m_game);
} }
const char* Card::maker(const std::unique_ptr<IFileHandle>& fh) const const char* Card::maker(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -598,7 +612,7 @@ const char* Card::maker(const std::unique_ptr<IFileHandle>& fh) const
return reinterpret_cast<const char*>(file->m_maker); return reinterpret_cast<const char*>(file->m_maker);
} }
void Card::setBannerFormat(const std::unique_ptr<IFileHandle>& fh, EImageFormat fmt) void Card::setBannerFormat(const FileHandle& fh, EImageFormat fmt)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -606,7 +620,7 @@ void Card::setBannerFormat(const std::unique_ptr<IFileHandle>& fh, EImageFormat
file->m_bannerFlags = (file->m_bannerFlags & ~3) | (uint8_t(fmt)); file->m_bannerFlags = (file->m_bannerFlags & ~3) | (uint8_t(fmt));
} }
EImageFormat Card::bannerFormat(const std::unique_ptr<IFileHandle>& fh) const EImageFormat Card::bannerFormat(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -614,7 +628,7 @@ EImageFormat Card::bannerFormat(const std::unique_ptr<IFileHandle>& fh) const
return EImageFormat(file->m_bannerFlags & 3); return EImageFormat(file->m_bannerFlags & 3);
} }
void Card::setIconAnimationType(const std::unique_ptr<IFileHandle>& fh, EAnimationType type) void Card::setIconAnimationType(const FileHandle& fh, EAnimationType type)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -622,7 +636,7 @@ void Card::setIconAnimationType(const std::unique_ptr<IFileHandle>& fh, EAnimati
file->m_bannerFlags = (file->m_bannerFlags & ~4) | uint8_t(type); file->m_bannerFlags = (file->m_bannerFlags & ~4) | uint8_t(type);
} }
EAnimationType Card::iconAnimationType(const std::unique_ptr<IFileHandle>& fh) const EAnimationType Card::iconAnimationType(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -631,7 +645,7 @@ EAnimationType Card::iconAnimationType(const std::unique_ptr<IFileHandle>& fh) c
return EAnimationType(file->m_bannerFlags & 4); return EAnimationType(file->m_bannerFlags & 4);
} }
void Card::setIconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, EImageFormat fmt) void Card::setIconFormat(const FileHandle& fh, uint32_t idx, EImageFormat fmt)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -640,7 +654,7 @@ void Card::setIconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, E
file->m_iconFmt = (file->m_iconFmt & ~(3 << (2 * idx))) | (uint16_t(fmt) << (2 * idx)); file->m_iconFmt = (file->m_iconFmt & ~(3 << (2 * idx))) | (uint16_t(fmt) << (2 * idx));
} }
EImageFormat Card::iconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t idx) const EImageFormat Card::iconFormat(const FileHandle& fh, uint32_t idx) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -649,7 +663,7 @@ EImageFormat Card::iconFormat(const std::unique_ptr<IFileHandle>& fh, uint32_t i
return EImageFormat(file->m_iconFmt >> (2 * (idx)) & 3); return EImageFormat(file->m_iconFmt >> (2 * (idx)) & 3);
} }
void Card::setIconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, EAnimationSpeed speed) void Card::setIconSpeed(const FileHandle& fh, uint32_t idx, EAnimationSpeed speed)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -658,7 +672,7 @@ void Card::setIconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t idx, EA
file->m_animSpeed = (file->m_animSpeed & ~(3 << (2 * idx))) | (uint16_t(speed) << (2 * idx)); file->m_animSpeed = (file->m_animSpeed & ~(3 << (2 * idx))) | (uint16_t(speed) << (2 * idx));
} }
EAnimationSpeed Card::iconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t idx) const EAnimationSpeed Card::iconSpeed(const FileHandle& fh, uint32_t idx) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -667,7 +681,7 @@ EAnimationSpeed Card::iconSpeed(const std::unique_ptr<IFileHandle>& fh, uint32_t
return EAnimationSpeed((file->m_animSpeed >> (2 * (idx))) & 3); return EAnimationSpeed((file->m_animSpeed >> (2 * (idx))) & 3);
} }
void Card::setImageAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t addr) void Card::setImageAddress(const FileHandle& fh, uint32_t addr)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -675,7 +689,7 @@ void Card::setImageAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t addr
file->m_iconAddress = addr; file->m_iconAddress = addr;
} }
int32_t Card::imageAddress(const std::unique_ptr<IFileHandle>& fh) const int32_t Card::imageAddress(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -683,7 +697,7 @@ int32_t Card::imageAddress(const std::unique_ptr<IFileHandle>& fh) const
return file->m_iconAddress; return file->m_iconAddress;
} }
void Card::setCommentAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t addr) void Card::setCommentAddress(const FileHandle& fh, uint32_t addr)
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -691,7 +705,7 @@ void Card::setCommentAddress(const std::unique_ptr<IFileHandle>& fh, uint32_t ad
file->m_commentAddr = addr; file->m_commentAddr = addr;
} }
int32_t Card::commentAddress(const std::unique_ptr<IFileHandle>& fh) const int32_t Card::commentAddress(const FileHandle& fh) const
{ {
File* file = _fileFromHandle(fh); File* file = _fileFromHandle(fh);
if (!file) if (!file)
@ -699,11 +713,8 @@ int32_t Card::commentAddress(const std::unique_ptr<IFileHandle>& fh) const
return file->m_commentAddr; return file->m_commentAddr;
} }
bool Card::copyFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest) bool Card::copyFileTo(FileHandle& fh, Card& dest)
{ {
if (!fh)
return false;
if (!canCopy(fh)) if (!canCopy(fh))
return false; return false;
@ -717,7 +728,7 @@ bool Card::copyFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest)
return false; return false;
/* Check to make sure dest does not already contain fh */ /* Check to make sure dest does not already contain fh */
std::unique_ptr<IFileHandle> tmpHandle; FileHandle tmpHandle;
dest.openFile(toCopy->m_filename, tmpHandle); dest.openFile(toCopy->m_filename, tmpHandle);
if (tmpHandle) if (tmpHandle)
return false; return false;
@ -750,7 +761,7 @@ bool Card::copyFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest)
return true; return true;
} }
bool Card::moveFileTo(const std::unique_ptr<IFileHandle>& fh, Card& dest) bool Card::moveFileTo(FileHandle& fh, Card& dest)
{ {
if (copyFileTo(fh, dest) && canMove(fh)) if (copyFileTo(fh, dest) && canMove(fh))
{ {
@ -928,8 +939,12 @@ ECardResult Card::getError() const
return ECardResult::NOCARD; return ECardResult::NOCARD;
uint16_t ckSum, ckSumInv; uint16_t ckSum, ckSumInv;
calculateChecksumLE(reinterpret_cast<const uint16_t*>(__raw), 0xFE, &ckSum, &ckSumInv); const_cast<Card&>(*this)._swapEndian();
if (ckSum != m_checksum || ckSumInv != m_checksumInv) calculateChecksumBE(reinterpret_cast<const uint16_t*>(__raw), 0xFE, &ckSum, &ckSumInv);
bool res = (ckSum == m_checksum && ckSumInv == m_checksumInv);
const_cast<Card&>(*this)._swapEndian();
if (!res)
return ECardResult::BROKEN; return ECardResult::BROKEN;
if (!m_dirs[0].valid() && !m_dirs[1].valid()) if (!m_dirs[0].valid() && !m_dirs[1].valid())
return ECardResult::BROKEN; return ECardResult::BROKEN;

View File

@ -15,14 +15,19 @@ void Directory::swapEndian()
void Directory::updateChecksum() void Directory::updateChecksum()
{ {
calculateChecksumLE(reinterpret_cast<uint16_t*>(__raw), 0xFFE, &m_checksum, &m_checksumInv); swapEndian();
calculateChecksumBE(reinterpret_cast<uint16_t*>(__raw), 0xFFE, &m_checksum, &m_checksumInv);
swapEndian();
} }
bool Directory::valid() const bool Directory::valid() const
{ {
uint16_t ckSum, ckSumInv; uint16_t ckSum, ckSumInv;
calculateChecksumLE(reinterpret_cast<const uint16_t*>(__raw), 0xFFE, &ckSum, &ckSumInv); const_cast<Directory&>(*this).swapEndian();
return (ckSum == m_checksum && ckSumInv == m_checksumInv); calculateChecksumBE(reinterpret_cast<const uint16_t*>(__raw), 0xFFE, &ckSum, &ckSumInv);
bool res = (ckSum == m_checksum && ckSumInv == m_checksumInv);
const_cast<Directory&>(*this).swapEndian();
return res;
} }
Directory::Directory() Directory::Directory()
@ -36,7 +41,7 @@ Directory::Directory(uint8_t data[]) { memcpy(__raw, data, BlockSize); }
bool Directory::hasFreeFile() const bool Directory::hasFreeFile() const
{ {
for (uint16_t i = 0; i < 127; i++) for (uint16_t i = 1; i < 127; i++)
if (m_files[i].m_game[0] == 0xFF) if (m_files[i].m_game[0] == 0xFF)
return true; return true;
return false; return false;
@ -45,7 +50,7 @@ bool Directory::hasFreeFile() const
int32_t Directory::numFreeFiles() const int32_t Directory::numFreeFiles() const
{ {
int32_t ret = 0; int32_t ret = 0;
for (uint16_t i = 0; i < 127; i++) for (uint16_t i = 1; i < 127; i++)
if (m_files[i].m_game[0] == 0xFF) if (m_files[i].m_game[0] == 0xFF)
++ret; ++ret;
return ret; return ret;
@ -53,7 +58,7 @@ int32_t Directory::numFreeFiles() const
File* Directory::getFirstFreeFile(const char* game, const char* maker, const char* filename) File* Directory::getFirstFreeFile(const char* game, const char* maker, const char* filename)
{ {
for (uint16_t i = 0; i < 127; i++) for (uint16_t i = 1; i < 127; i++)
{ {
if (m_files[i].m_game[0] == 0xFF) if (m_files[i].m_game[0] == 0xFF)
{ {
@ -92,7 +97,7 @@ File* Directory::getFirstNonFreeFile(uint32_t start, const char* game, const cha
File* Directory::getFile(const char* game, const char* maker, const char* filename) File* Directory::getFile(const char* game, const char* maker, const char* filename)
{ {
for (uint16_t i = 0; i < 127; i++) for (uint16_t i = 1; i < 127; i++)
{ {
if (game && strlen(game) == 4 && memcmp(m_files[i].m_game, game, 4)) if (game && strlen(game) == 4 && memcmp(m_files[i].m_game, game, 4))
continue; continue;

View File

@ -42,19 +42,4 @@ void calculateChecksumBE(const uint16_t* data, size_t len, uint16_t* checksum, u
*checksumInv = 0; *checksumInv = 0;
} }
void calculateChecksumLE(const uint16_t* data, size_t len, uint16_t* checksum, uint16_t* checksumInv)
{
*checksum = 0;
*checksumInv = 0;
for (size_t i = 0; i < len; ++i)
{
*checksum += data[i];
*checksumInv += uint16_t(data[i] ^ 0xFFFF);
}
if (*checksum == 0xFFFF)
*checksum = 0;
if (*checksumInv == 0xFFFF)
*checksumInv = 0;
}
} }

View File

@ -8,7 +8,7 @@ int main()
uint64_t a = 0; uint64_t a = 0;
mc.getSerial(a); mc.getSerial(a);
std::unique_ptr<kabufuda::IFileHandle> f; kabufuda::FileHandle f;
mc.openFile("MetroidPrime A", f); mc.openFile("MetroidPrime A", f);
for (uint32_t i = 0; i < 127; i++) for (uint32_t i = 0; i < 127; i++)
{ {