mirror of https://github.com/AxioDL/kabufuda.git
FileHandle refactor
This commit is contained in:
parent
b5f5104877
commit
408ac735b4
|
@ -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
|
||||||
|
|
|
@ -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__
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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++)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue