RASCSI/cpp/devices/cfilesystem.cpp
Daniel Markstedt 52c2aa474f
Rebrand project to PiSCSI (#1016)
* Rebrand project to PiSCSI
- rascsi ->piscsi
- rasctl -> scsictl
- rasdump -> scsidump
- ras* -> piscsi* (rasutil -> piscsi_util, etc.)

* Refined the formatting and wording of the app startup banner
* Kept some references to rascsi and rasctl where backwards compatibility is concerned
* Point to the new github repo URL

Co-authored-by: nucleogenic <nr@nucleogenic.com>
Co-authored-by: Uwe Seimet <Uwe.Seimet@seimet.de>
2022-12-05 09:58:23 -08:00

3942 lines
102 KiB
C++

//---------------------------------------------------------------------------
//
// SCSI Target Emulator PiSCSI
// for Raspberry Pi
//
// Powered by XM6 TypeG Technology.
// Copyright (C) 2016-2020 GIMONS
//
// Imported sava's bugfix patch(in RASDRV DOS edition).
//
// [ Host File System for the X68000 ]
//
// Note: This functionality is specific to the X68000
// operating system.
// It is highly unlikely that this will work for other
// platforms.
//
//---------------------------------------------------------------------------
#include "shared/log.h"
#include "cfilesystem.h"
#include <dirent.h>
#include <iconv.h>
#include <utime.h>
#define ARRAY_SIZE(x) (sizeof(x)/(sizeof(x[0])))
//---------------------------------------------------------------------------
//
// Kanji code conversion
//
//---------------------------------------------------------------------------
static const int IC_BUF_SIZE = 1024;
static char convert_buf[IC_BUF_SIZE];
#define CONVERT(src, dest, inbuf, outbuf, outsize) \
convert(src, dest, (char *)inbuf, outbuf, outsize)
static void convert(char const *src, char const *dest,
char *inbuf, char *outbuf, size_t outsize)
{
*outbuf = '\0';
size_t in = strlen(inbuf);
size_t out = outsize - 1;
iconv_t cd = iconv_open(dest, src);
if (cd == (iconv_t)-1) {
return;
}
if (const size_t ret = iconv(cd, &inbuf, &in, &outbuf, &out); ret == (size_t)-1) {
return;
}
iconv_close(cd);
*outbuf = '\0';
}
//---------------------------------------------------------------------------
//
// SJIS->UTF8 conversion
//
//---------------------------------------------------------------------------
static char* SJIS2UTF8(const char *sjis, char *utf8, size_t bufsize)
{
CONVERT("SJIS", "UTF-8", sjis, utf8, bufsize);
return convert_buf;
}
//---------------------------------------------------------------------------
//
// UTF8->SJIS conversion
//
//---------------------------------------------------------------------------
static char* UTF82SJIS(const char *utf8, char *sjis, size_t bufsize)
{
CONVERT("UTF-8", "SJIS", utf8, sjis, bufsize);
return convert_buf;
}
//---------------------------------------------------------------------------
//
// SJIS->UTF8 conversion (simplified versoin)
//
//---------------------------------------------------------------------------
static char* S2U(const char *sjis)
{
SJIS2UTF8(sjis, convert_buf, IC_BUF_SIZE);
return convert_buf;
}
//---------------------------------------------------------------------------
//
// UTF8->SJIS conversion (simplified version)
//
//---------------------------------------------------------------------------
static char* U2S(const char *utf8)
{
UTF82SJIS(utf8, convert_buf, IC_BUF_SIZE);
return convert_buf;
}
//---------------------------------------------------------------------------
//
/// Get path name
///
/// From the structure used in Human68k namests, get the Human68k path name.
/// A 66 byte buffer is required for writing.
//
//---------------------------------------------------------------------------
void Human68k::namests_t::GetCopyPath(uint8_t* szPath) const
{
assert(szPath);
uint8_t* p = szPath;
for (uint8_t c : path) {
if (c == '\0')
break;
if (c == 0x09) {
c = '/';
}
*p++ = c;
}
*p = '\0';
}
//---------------------------------------------------------------------------
//
/// Get file name
///
/// From the structure used in Human68k namests, get the Human68k file name.
/// A 23 byte buffer is required for writing.
//
//---------------------------------------------------------------------------
void Human68k::namests_t::GetCopyFilename(uint8_t* szFilename) const
{
assert(szFilename);
size_t i;
uint8_t* p = szFilename;
// Transfer the base file name
for (i = 0; i < 8; i++) {
uint8_t c = name[i];
if (c == ' ') {
// Check that the file name continues after a space is detected
/// TODO: Should change this function to be compatible with 8+3 chars and TwentyOne
// Continue if add[0] is a valid character
if (add[0] != '\0')
goto next_name;
// Continue if a non-space character exists after name[i]
for (size_t j = i + 1; j < 8; j++) {
if (name[j] != ' ')
goto next_name;
}
// Exit if the file name ends
break;
}
next_name:
*p++ = c;
}
// At this point, the number of read characters becomes i >= 8
// If the body of the file name exceeds 8 characters, add the extraneous part
if (i >= 8) {
// Transfer the extraneous part
for (i = 0; i < 10; i++) {
const uint8_t c = add[i];
if (c == '\0')
break;
*p++ = c;
}
// At this point, the number of read characters becomes i >= 10
}
// Transfer the file extension if it exists
if (ext[0] != ' ' || ext[1] != ' ' || ext[2] != ' ') {
*p++ = '.';
for (i = 0; i < 3; i++) {
const uint8_t c = ext[i];
if (c == ' ') {
// Check that the file extension continues after a space is detected
/// TODO: Should change this function to be compatible with 8+3 chars and TwentyOne
// Continue if a non-space character exists after ext[i]
for (size_t j = i + 1; j < 3; j++) {
if (ext[j] != ' ')
goto next_ext;
}
// If the extension ends, the transfer ends
break;
}
next_ext:
*p++ = c;
}
// When all the characters are read, here i >= 3
}
// Add sentinel
*p = '\0';
}
//===========================================================================
//
// Host side drive
//
//===========================================================================
CHostDrv::~CHostDrv()
{
CHostPath* p;
while ((p = (CHostPath*)m_cRing.Next()) != &m_cRing) {
delete p;
assert(m_nRing);
m_nRing--;
}
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
assert(m_nRing == 0);
}
//---------------------------------------------------------------------------
//
// Initialization (device boot and load)
//
//---------------------------------------------------------------------------
void CHostDrv::Init(const TCHAR* szBase, uint32_t nFlag)
{
assert(szBase);
assert(strlen(szBase) < FILEPATH_MAX);
assert(!m_bWriteProtect);
assert(!m_bEnable);
assert(m_capCache.sectors == 0);
assert(!m_bVolumeCache);
assert(m_szVolumeCache[0] == '\0');
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
assert(m_nRing == 0);
m_capCache.sectors = 0;
// Receive parameters
if (nFlag & FSFLAG_WRITE_PROTECT)
m_bWriteProtect = true;
strcpy(m_szBase, szBase);
// Remove the last path delimiter in the base path
// @warning needs to be modified when using Unicode
TCHAR* pClear = nullptr;
TCHAR* p = m_szBase;
for (;;) {
const TCHAR c = *p;
if (c == '\0')
break;
if (c == '/' || c == '\\') {
pClear = p;
} else {
pClear = nullptr;
}
if ((c <= (TCHAR)0x9F) || (TCHAR)0xE0 <= c) { // To be precise: 0x81~0x9F 0xE0~0xEF
p++;
if (*p == '\0')
break;
}
p++;
}
if (pClear)
*pClear = '\0';
// Status update
m_bEnable = true;
}
//---------------------------------------------------------------------------
//
// Media check
//
//---------------------------------------------------------------------------
bool CHostDrv::isMediaOffline() const
{
// Offline status check
return !m_bEnable;
}
//---------------------------------------------------------------------------
//
// Get media bytes
//
//---------------------------------------------------------------------------
uint8_t CHostDrv::GetMediaByte() const
{
return Human68k::MEDIA_REMOTE;
}
//---------------------------------------------------------------------------
//
// Get drive status
//
//---------------------------------------------------------------------------
uint32_t CHostDrv::GetStatus() const
{
return 0x40 | (m_bEnable ? (m_bWriteProtect ? 0x08 : 0) | 0x02 : 0);
}
//---------------------------------------------------------------------------
//
// Media status settings
//
//---------------------------------------------------------------------------
void CHostDrv::SetEnable(bool bEnable)
{
m_bEnable = bEnable;
if (!bEnable) {
// Clear cache
m_capCache.sectors = 0;
m_bVolumeCache = false;
m_szVolumeCache[0] = '\0';
}
}
//---------------------------------------------------------------------------
//
// Media change check
//
//---------------------------------------------------------------------------
bool CHostDrv::CheckMedia()
{
// Status update
Update();
if (!m_bEnable)
CleanCache();
return m_bEnable;
}
//---------------------------------------------------------------------------
//
// Media status update
//
//---------------------------------------------------------------------------
void CHostDrv::Update()
{
// Considered as media insertion
// Media status reflected
SetEnable(true);
}
//---------------------------------------------------------------------------
//
// Eject
//
//---------------------------------------------------------------------------
void CHostDrv::Eject()
{
// Media discharge
CleanCache();
SetEnable(false);
// Status update
Update();
}
//---------------------------------------------------------------------------
//
// Get volume label
//
//---------------------------------------------------------------------------
void CHostDrv::GetVolume(TCHAR* szLabel)
{
assert(szLabel);
assert(m_bEnable);
// Get volume label
strcpy(m_szVolumeCache, "RASDRV ");
if (m_szBase[0]) {
strcat(m_szVolumeCache, m_szBase);
} else {
strcat(m_szVolumeCache, "/");
}
// Cache update
m_bVolumeCache = true;
// Transfer content
strcpy(szLabel, m_szVolumeCache);
}
//---------------------------------------------------------------------------
//
/// Get volume label from cache
///
/// Transfer the cached volume label information.
/// Return true if the cache contents are valid.
//
//---------------------------------------------------------------------------
bool CHostDrv::GetVolumeCache(TCHAR* szLabel) const
{
assert(szLabel);
// Transfer contents
strcpy(szLabel, m_szVolumeCache);
return m_bVolumeCache;
}
uint32_t CHostDrv::GetCapacity(Human68k::capacity_t* pCapacity)
{
assert(pCapacity);
assert(m_bEnable);
const uint32_t nFree = 0x7FFF8000;
uint32_t freearea;
uint32_t clusters;
uint32_t sectors;
freearea = 0xFFFF;
clusters = 0xFFFF;
sectors = 64;
// Estimated parameter range
assert(freearea <= 0xFFFF);
assert(clusters <= 0xFFFF);
assert(sectors <= 64);
// Update cache
m_capCache.freearea = (uint16_t)freearea;
m_capCache.clusters = (uint16_t)clusters;
m_capCache.sectors = (uint16_t)sectors;
m_capCache.bytes = 512;
// Transfer contents
memcpy(pCapacity, &m_capCache, sizeof(m_capCache));
return nFree;
}
//---------------------------------------------------------------------------
//
/// Get capacity from the cache
///
/// Transfer the capacity data stored in cache.
/// Return true if the contents of the cache are valid.
//
//---------------------------------------------------------------------------
bool CHostDrv::GetCapacityCache(Human68k::capacity_t* pCapacity) const
{
assert(pCapacity);
// Transfer contents
memcpy(pCapacity, &m_capCache, sizeof(m_capCache));
return m_capCache.sectors != 0;
}
//---------------------------------------------------------------------------
//
/// Update all cache
//
//---------------------------------------------------------------------------
void CHostDrv::CleanCache() const
{
for (auto p = (CHostPath*)m_cRing.Next(); p != &m_cRing;) {
p->Release();
p = (CHostPath*)p->Next();
}
}
//---------------------------------------------------------------------------
//
/// Update the cache for the specified path
//
//---------------------------------------------------------------------------
void CHostDrv::CleanCache(const uint8_t* szHumanPath)
{
assert(szHumanPath);
CHostPath* p = FindCache(szHumanPath);
if (p) {
p->Restore();
p->Release();
}
}
//---------------------------------------------------------------------------
//
/// Update the cache below and including the specified path
//
//---------------------------------------------------------------------------
void CHostDrv::CleanCacheChild(const uint8_t* szHumanPath) const
{
assert(szHumanPath);
auto p = (CHostPath*)m_cRing.Next();
while (p != &m_cRing) {
if (p->isSameChild(szHumanPath))
p->Release();
p = (CHostPath*)p->Next();
}
}
//---------------------------------------------------------------------------
//
/// Delete the cache for the specified path
//
//---------------------------------------------------------------------------
void CHostDrv::DeleteCache(const uint8_t* szHumanPath)
{
auto p = FindCache(szHumanPath);
if (p) {
delete p;
assert(m_nRing);
m_nRing--;
}
}
//---------------------------------------------------------------------------
//
/// Check if the specified path is cached
///
/// Check if whether it is a perfect match with the cache buffer, and return the name if found.
/// File names are excempted.
/// Make sure to lock from the top.
//
//---------------------------------------------------------------------------
CHostPath* CHostDrv::FindCache(const uint8_t* szHuman)
{
assert(szHuman);
// Find something that matches perfectly with either of the stored file names
for (auto p = (CHostPath*)m_cRing.Next(); p != &m_cRing;) {
if (p->isSameHuman(szHuman))
return p;
p = (CHostPath*)p->Next();
}
return nullptr;
}
//---------------------------------------------------------------------------
//
/// Get the host side name from cached data.
///
/// Confirm if the path is cached. If not, throw an error.
/// Carry out an update check on found cache. If an update is needed, throw an error.
/// Make sure to lock from the top.
//
//---------------------------------------------------------------------------
CHostPath* CHostDrv::CopyCache(CHostFiles* pFiles)
{
assert(pFiles);
assert(strlen((const char*)pFiles->GetHumanPath()) < HUMAN68K_PATH_MAX);
// Find in cache
CHostPath* pPath = FindCache(pFiles->GetHumanPath());
if (pPath == nullptr) {
return nullptr; // Error: No cache
}
// Move to the beginning of the ring
pPath->Insert(&m_cRing);
// Cache update check
if (pPath->isRefresh()) {
return nullptr; // Error: Cache update is required
}
// Store the host side path
pFiles->SetResult(pPath->GetHost());
return pPath;
}
//---------------------------------------------------------------------------
//
/// Get all the data required for a host side name structure
///
/// File names can be abbreviated. (Normally not selected)
/// Make sure to lock from the top.
/// Be careful not to append the path separator char to the end of the base path.
/// Initiate VM threads when there is a chance of multiple file accesses.
///
/// How to use:
/// When CopyCache() throws an error execute MakeCache(). It ensures you get a correct host side path.
///
/// Split all file names and path names.
/// Confirm that it's cached from the top level directory down.
/// If it's cached, do a destruction check. If it was destroyed, treat it as uncached.
/// If it isn't cached, build cache.
/// Exit after processing all directory and file names in order.
/// Make it nullptr is an error is thrown.
//
//---------------------------------------------------------------------------
CHostPath* CHostDrv::MakeCache(CHostFiles* pFiles)
{
assert(pFiles);
assert(strlen((const char*)pFiles->GetHumanPath()) < HUMAN68K_PATH_MAX);
assert(m_szBase);
assert(strlen(m_szBase) < FILEPATH_MAX);
uint8_t szHumanPath[HUMAN68K_PATH_MAX]; // Path names are entered in order from the route
szHumanPath[0] = '\0';
size_t nHumanPath = 0;
TCHAR szHostPath[FILEPATH_MAX];
strcpy(szHostPath, m_szBase);
size_t nHostPath = strlen(szHostPath);
CHostPath* pPath;
const uint8_t* p = pFiles->GetHumanPath();
for (;;) {
// Add path separators
if (nHumanPath + 1 >= HUMAN68K_PATH_MAX)
return nullptr; // Error: The Human68k path is too long
szHumanPath[nHumanPath++] = '/';
szHumanPath[nHumanPath] = '\0';
if (nHostPath + 1 >= FILEPATH_MAX)
return nullptr; // Error: The host side path is too long
szHostPath[nHostPath++] = '/';
szHostPath[nHostPath] = '\0';
// Insert one file
uint8_t szHumanFilename[24]; // File name part
p = SeparateCopyFilename(p, szHumanFilename);
if (p == nullptr)
return nullptr; // Error: Failed to read file name
size_t n = strlen((const char*)szHumanFilename);
if (nHumanPath + n >= HUMAN68K_PATH_MAX)
return nullptr; // Error: The Human68k path is too long
// Is the relevant path cached?
pPath = FindCache(szHumanPath);
if (pPath == nullptr) {
// Check for max number of cache
if (m_nRing >= XM6_HOST_DIRENTRY_CACHE_MAX) {
// Destroy the oldest cache and reuse it
pPath = (CHostPath*)m_cRing.Prev();
pPath->Clean(); // Release all files. Release update check handlers.
} else {
// Register new
pPath = new CHostPath;
assert(pPath);
m_nRing++;
}
pPath->SetHuman(szHumanPath);
pPath->SetHost(szHostPath);
// Update status
pPath->Refresh();
}
// Cache update check
if (pPath->isRefresh()) {
Update();
// Update status
pPath->Refresh();
}
// Into the beginning of the ring
pPath->Insert(&m_cRing);
// Exit if there is not file name
if (n == 0)
break;
// Find the next path
// Confirm if directory from the middle of the path
const CHostFilename* pFilename;
if (*p != '\0')
pFilename = pPath->FindFilename(szHumanFilename, Human68k::AT_DIRECTORY);
else
pFilename = pPath->FindFilename(szHumanFilename);
if (pFilename == nullptr)
return nullptr; // Error: Could not find path or file names in the middle
// Link path name
strcpy((char*)szHumanPath + nHumanPath, (const char*)szHumanFilename);
nHumanPath += n;
n = strlen(pFilename->GetHost());
if (nHostPath + n >= FILEPATH_MAX)
return nullptr; // Error: Host side path is too long
strcpy(szHostPath + nHostPath, pFilename->GetHost());
nHostPath += n;
// PLEASE CONTINUE
if (*p == '\0')
break;
}
// Store the host side path name
pFiles->SetResult(szHostPath);
return pPath;
}
//---------------------------------------------------------------------------
//
/// Find host side name (path name + file name (can be abbeviated) + attribute)
///
/// Set all Human68k parameters once more.
//
//---------------------------------------------------------------------------
bool CHostDrv::Find(CHostFiles* pFiles)
{
assert(pFiles);
// Get path name and build cache
const CHostPath* pPath = CopyCache(pFiles);
if (pPath == nullptr) {
pPath = MakeCache(pFiles);
if (pPath == nullptr) {
CleanCache();
return false; // Error: Failed to build cache
}
}
// Store host side path
pFiles->SetResult(pPath->GetHost());
// Exit if only path name
if (pFiles->isPathOnly()) {
return true; // Normal exit: only path name
}
// Find file name
const CHostFilename* pFilename = pFiles->Find(pPath);
if (pFilename == nullptr) {
return false; // Error: Could not get file name
}
// Store the Human68k side search results
pFiles->SetEntry(pFilename);
// Store the host side full path name
pFiles->AddResult(pFilename->GetHost());
return true;
}
//===========================================================================
//
// Directory entry: File name
//
//===========================================================================
//---------------------------------------------------------------------------
//
/// Set host side name
//
//---------------------------------------------------------------------------
void CHostFilename::SetHost(const TCHAR* szHost)
{
assert(szHost);
assert(strlen(szHost) < FILEPATH_MAX);
strcpy(m_szHost, szHost);
}
//---------------------------------------------------------------------------
//
/// Copy the Human68k file name elements
//
//---------------------------------------------------------------------------
uint8_t* CHostFilename::CopyName(uint8_t* pWrite, const uint8_t* pFirst, const uint8_t* pLast) // static
{
assert(pWrite);
assert(pFirst);
assert(pLast);
for (const uint8_t* p = pFirst; p < pLast; p++) {
*pWrite++ = *p;
}
return pWrite;
}
//---------------------------------------------------------------------------
//
/// Convert the Human68k side name
///
/// Once more, execute SetHost().
/// Carry out name conversion to the 18+3 standard.
/// Automatically delete spaces in the beginning and end of the names, since Human68k can't handle them.
/// The directory entry name segment is created at the time of conversion using knowledge of the location of the file name extension.
/// Afterwards, a file name validity check is performed. (Ex. file names consisting of 8 spaces only.)
/// No file name duplication check is performed so be careful. Such validation is carried out in classes higher up.
/// Adhers to the naming standards of: TwentyOne version 1.36c modified +14 patchlevel9 or later
//
//---------------------------------------------------------------------------
void CHostFilename::ConvertHuman(int nCount)
{
// Don't do conversion for special directory names
if (m_szHost[0] == '.' &&
(m_szHost[1] == '\0' || (m_szHost[1] == '.' && m_szHost[2] == '\0'))) {
strcpy((char*)m_szHuman, m_szHost);
m_bCorrect = true;
m_pszHumanLast = m_szHuman + strlen((const char*)m_szHuman);
m_pszHumanExt = m_pszHumanLast;
return;
}
size_t nMax = 18; // Number of bytes for the base segment (base name and extension)
uint32_t nOption = CFileSys::GetFileOption();
if (nOption & WINDRV_OPT_CONVERT_LENGTH)
nMax = 8;
// Preparations to adjust the base name segment
uint8_t szNumber[8];
uint8_t* pNumber = nullptr;
if (nCount >= 0) {
pNumber = &szNumber[8];
for (uint32_t i = 0; i < 5; i++) { // Max 5+1 digits (always leave the first 2 bytes of the base name)
int n = nCount % 36;
nMax--;
pNumber--;
*pNumber = (uint8_t)(n + (n < 10 ? '0' : 'A' - 10));
nCount /= 36;
if (nCount == 0)
break;
}
nMax--;
pNumber--;
auto c = (uint8_t)((nOption >> 24) & 0x7F);
if (c == 0)
c = XM6_HOST_FILENAME_MARK;
*pNumber = c;
}
// Char conversion
uint8_t szHuman[FILEPATH_MAX];
const uint8_t* pFirst = szHuman;
uint8_t* pLast;
uint8_t* pExt = nullptr;
{
char szHost[FILEPATH_MAX];
strcpy(szHost, m_szHost);
auto pRead = (const uint8_t*)szHost;
uint8_t* pWrite = szHuman;
const auto pPeriod = SeparateExt(pRead);
for (bool bFirst = true;; bFirst = false) {
uint8_t c = *pRead++;
switch (c) {
case ' ':
if (nOption & WINDRV_OPT_REDUCED_SPACE)
continue;
if (nOption & WINDRV_OPT_CONVERT_SPACE)
c = '_';
else if (pWrite == szHuman)
continue; // Ignore spaces in the beginning
break;
case '=':
case '+':
if (nOption & WINDRV_OPT_REDUCED_BADCHAR)
continue;
if (nOption & WINDRV_OPT_CONVERT_BADCHAR)
c = '_';
break;
case '-':
if (bFirst) {
if (nOption & WINDRV_OPT_REDUCED_HYPHEN)
continue;
if (nOption & WINDRV_OPT_CONVERT_HYPHEN)
c = '_';
break;
}
if (nOption & WINDRV_OPT_REDUCED_HYPHENS)
continue;
if (nOption & WINDRV_OPT_CONVERT_HYPHENS)
c = '_';
break;
case '.':
if (pRead - 1 == pPeriod) { // Make exception for Human68k extensions
pExt = pWrite;
break;
}
if (bFirst) {
if (nOption & WINDRV_OPT_REDUCED_PERIOD)
continue;
if (nOption & WINDRV_OPT_CONVERT_PERIOD)
c = '_';
break;
}
if (nOption & WINDRV_OPT_REDUCED_PERIODS)
continue;
if (nOption & WINDRV_OPT_CONVERT_PERIODS)
c = '_';
break;
default:
break;
}
*pWrite++ = c;
if (c == '\0')
break;
}
pLast = pWrite - 1;
}
// Adjust extensions
if (pExt) {
// Delete spaces at the end
while (pExt < pLast - 1 && *(pLast - 1) == ' ') {
pLast--;
auto p = pLast;
*p = '\0';
}
// Delete if the file name disappeared after conversion
if (pExt + 1 >= pLast) {
pLast = pExt;
*pLast = '\0'; // Just in case
}
} else {
pExt = pLast;
}
// Introducing the cast of characters
//
// pFirst: I'm the glorious leader. The start of the file name.
// pCut: A.k.a. Phase. Location of the initial period. Afterwards becomes the end of the base name.
// pSecond: Hello there! I'm the incredible Murdock. The start of the file name extension. What's it to you?
// pExt: B.A. Baracus. The Human68k extension genius. But don't you dare giving me more than 3 chars, fool.
// The location of the final period. If not applicable, gets the same value as pLast.
//
// ↓pFirst ↓pStop ↓pSecond ← ↓pExt
// T h i s _ i s _ a . V e r y . L o n g . F i l e n a m e . t x t \0
// ↑pCut ← ↑pCut initial location ↑pLast
//
// The above example becomes "This.Long.Filename.txt" after conversion
// Evaluate first char
const uint8_t* pCut = pFirst;
const uint8_t* pStop = pExt - nMax; // Allow for up to 17 bytes for extension (leave base name)
if (pFirst < pExt) {
pCut++; // 1 byte always uses the base name
uint8_t c = *pFirst;
if ((0x80 <= c && c <= 0x9F) || 0xE0 <= c) { // Specifically 0x81~0x9F 0xE0~0xEF
pCut++; // Base name. At least 2 bytes.
pStop++; // File extension. Max 16 bytes.
}
}
if (pStop < pFirst)
pStop = pFirst;
// Evaluate base name
pCut = (const uint8_t*)strchr((const char*)pCut, '.'); // The 2nd byte of Shift-JIS is always 0x40 or higher, so this is ok
if (pCut == nullptr)
pCut = pLast;
if ((size_t)(pCut - pFirst) > nMax)
pCut = pFirst + nMax; // Execute Shift-JIS 2 byte evaluation/adjustment later. Not allowed to do it here.
// Evaluate extension
const uint8_t* pSecond = pExt;
const uint8_t* p;
for (p = pExt - 1; pStop < p; p--) {
if (*p == '.')
pSecond = p; // The 2nd byte of Shift-JIS is always 0x40 or higher, so this is ok
}
// Shorten base name
// Length of extension segment
if (size_t nExt = pExt - pSecond; (size_t)(pCut - pFirst) + nExt > nMax)
pCut = pFirst + nMax - nExt;
// If in the middle of a 2 byte char, shorten even further
for (p = pFirst; p < pCut; p++) {
uint8_t c = *p;
if ((0x80 <= c && c <= 0x9F) || 0xE0 <= c) { // Specifically 0x81~0x9F 0xE0~0xEF
p++;
if (p >= pCut) {
pCut--;
break;
}
}
}
// Joining the name
uint8_t* pWrite = m_szHuman;
pWrite = CopyName(pWrite, pFirst, pCut); // Transfer the base name
if (pNumber)
pWrite = CopyName(pWrite, pNumber, &szNumber[8]); // Transfer the adjustment char
pWrite = CopyName(pWrite, pSecond, pExt); // Transfer the extension name
m_pszHumanExt = pWrite; // Store the extention position
pWrite = CopyName(pWrite, pExt, pLast); // Transfer the Human68k extension
m_pszHumanLast = pWrite; // Store the end position
*pWrite = '\0';
// Confirm the conversion results
m_bCorrect = true;
// Fail if the base file name does not exist
if (m_pszHumanExt <= m_szHuman)
m_bCorrect = false;
// Fail if the base file name is more than 1 char and ends with a space
// While it is theoretically valid to have a base file name exceed 8 chars,
// Human68k is unable to handle it, so failing this case too.
else if (m_pszHumanExt[-1] == ' ')
m_bCorrect = false;
// Fail if the conversion result is the same as a special directory name
if (m_szHuman[0] == '.' &&
(m_szHuman[1] == '\0' || (m_szHuman[1] == '.' && m_szHuman[2] == '\0')))
m_bCorrect = false;
}
//---------------------------------------------------------------------------
//
/// Human68k side name duplication
///
/// Duplicates the file name segment data, then executes the correspoding initialization with ConvertHuman().
//
//---------------------------------------------------------------------------
void CHostFilename::CopyHuman(const uint8_t* szHuman)
{
assert(szHuman);
assert(strlen((const char*)szHuman) < 23);
strcpy((char*)m_szHuman, (const char*)szHuman);
m_bCorrect = true;
m_pszHumanLast = m_szHuman + strlen((const char*)m_szHuman);
m_pszHumanExt = SeparateExt(m_szHuman);
}
//---------------------------------------------------------------------------
//
/// Set Human68k directory entry
///
/// Apply the set file name to the directory entry with ConvertHuman().
//
//---------------------------------------------------------------------------
void CHostFilename::SetEntryName()
{
// Set file name
const uint8_t* p = m_szHuman;
size_t i;
for (i = 0; i < 8; i++) {
if (p < m_pszHumanExt)
m_dirHuman.name[i] = *p++;
else
m_dirHuman.name[i] = ' ';
}
for (i = 0; i < 10; i++) {
if (p < m_pszHumanExt)
m_dirHuman.add[i] = *p++;
else
m_dirHuman.add[i] = '\0';
}
if (*p == '.')
p++;
for (i = 0; i < 3; i++) {
uint8_t c = *p;
if (c)
p++;
m_dirHuman.ext[i] = c;
}
}
//---------------------------------------------------------------------------
//
/// Investigate if the Human68k side name has been processed
//
//---------------------------------------------------------------------------
bool CHostFilename::isReduce() const
{
return strcmp((const char *)m_szHost, (const char*)m_szHuman) != 0;
}
//---------------------------------------------------------------------------
//
/// Evaluate Human68k directory entry attribute
//
//---------------------------------------------------------------------------
int CHostFilename::CheckAttribute(uint32_t nHumanAttribute) const
{
uint8_t nAttribute = m_dirHuman.attr;
if ((nAttribute & (Human68k::AT_ARCHIVE | Human68k::AT_DIRECTORY | Human68k::AT_VOLUME)) == 0)
nAttribute |= Human68k::AT_ARCHIVE;
return nAttribute & nHumanAttribute;
}
//---------------------------------------------------------------------------
//
/// Split the extension from Human68k file name
//
//---------------------------------------------------------------------------
const uint8_t* CHostFilename::SeparateExt(const uint8_t* szHuman) // static
{
// Obtain the file name length
const size_t nLength = strlen((const char*)szHuman);
const uint8_t* pFirst = szHuman;
const uint8_t* pLast = pFirst + nLength;
// Confirm the position of the Human68k extension
auto pExt = (const uint8_t*)strrchr((const char*)pFirst, '.'); // The 2nd byte of Shift-JIS is always 0x40 or higher, so this is ok
if (pExt == nullptr)
pExt = pLast;
// Special handling of the pattern where the file name is 20~22 chars, and the 19th char is '.' or ends with '.'
if (20 <= nLength && nLength <= 22 && pFirst[18] == '.' && pFirst[nLength - 1] == '.')
pExt = pFirst + 18;
// Calculate the number of chars in the extension (-1:None 0:Only period 1~3:Human68k extension 4 or above:extension name)
// Consider it an extension only when '.' is anywhere except the beginning of the string, and between 1~3 chars long
if (size_t nExt = pLast - pExt - 1; pExt == pFirst || nExt < 1 || nExt > 3)
pExt = pLast;
return pExt;
}
//===========================================================================
//
// Directory entry: path name
//
//===========================================================================
uint32_t CHostPath::g_nId; ///< Identifier creation counter
CHostPath::~CHostPath()
{
Clean();
}
//---------------------------------------------------------------------------
//
/// File name memory allocation
///
/// In most cases, the length of the host side file name is way shorter
/// than the size of the buffer. In addition, file names may be created in huge volumes.
/// Therefore, allocate variable lengths that correspond to the number of chars.
//
//---------------------------------------------------------------------------
CHostPath::ring_t* CHostPath::Alloc(size_t nLength) // static
{
assert(nLength < FILEPATH_MAX);
const size_t n = offsetof(ring_t, f) + CHostFilename::Offset() + (nLength + 1) * sizeof(TCHAR);
auto p = (ring_t*)malloc(n);
assert(p);
p->r.Init(); // This is nothing to worry about!
return p;
}
//---------------------------------------------------------------------------
//
// Release file name allocations
//
//---------------------------------------------------------------------------
void CHostPath::Free(ring_t* pRing) // static
{
assert(pRing);
pRing->~ring_t();
free(pRing);
}
//---------------------------------------------------------------------------
//
/// Initialize for reuse
//
//---------------------------------------------------------------------------
void CHostPath::Clean()
{
Release();
// Release all file names
ring_t* p;
while ((p = (ring_t*)m_cRing.Next()) != (ring_t*)&m_cRing) {
Free(p);
}
}
//---------------------------------------------------------------------------
//
/// Specify Human68k side names directly
//
//---------------------------------------------------------------------------
void CHostPath::SetHuman(const uint8_t* szHuman)
{
assert(szHuman);
assert(strlen((const char*)szHuman) < HUMAN68K_PATH_MAX);
strcpy((char*)m_szHuman, (const char*)szHuman);
}
//---------------------------------------------------------------------------
//
/// Specify host side names directly
//
//---------------------------------------------------------------------------
void CHostPath::SetHost(const TCHAR* szHost)
{
assert(szHost);
assert(strlen(szHost) < FILEPATH_MAX);
strcpy(m_szHost, szHost);
}
//---------------------------------------------------------------------------
//
/// Compare arrays (supports wildcards)
//
//---------------------------------------------------------------------------
int CHostPath::Compare(const uint8_t* pFirst, const uint8_t* pLast, const uint8_t* pBufFirst, const uint8_t* pBufLast)
{
assert(pFirst);
assert(pLast);
assert(pBufFirst);
assert(pBufLast);
// Compare chars
bool bSkip0 = false;
bool bSkip1 = false;
for (const uint8_t* p = pFirst; p < pLast; p++) {
// Read 1 char
uint8_t c = *p;
uint8_t d = '\0';
if (pBufFirst < pBufLast)
d = *pBufFirst++;
// Ajust char for comparison
if (!bSkip0) {
if (!bSkip1) { // First byte for both c and d
if ((0x80 <= c && c <= 0x9F) || 0xE0 <= c) { // Specifically 0x81~0x9F 0xE0~0xEF
bSkip0 = true;
}
if ((0x80 <= d && d <= 0x9F) || 0xE0 <= d) { // Specifically 0x81~0x9F 0xE0~0xEF
bSkip1 = true;
}
if (c == d)
continue; // Finishes the evaluation here with high probability
if ((CFileSys::GetFileOption() & WINDRV_OPT_ALPHABET) == 0) {
if ('A' <= c && c <= 'Z')
c += 'a' - 'A'; // To lower case
if ('A' <= d && d <= 'Z')
d += 'a' - 'A'; // To lower case
}
// Unify slashes and backslashes for comparison
if (c == '\\') {
c = '/';
}
if (d == '\\') {
d = '/';
}
} else { // Only c is first byte
if ((0x80 <= c && c <= 0x9F) || 0xE0 <= c) { // Specifically 0x81~0x9F 0xE0~0xEF
bSkip0 = true;
}
bSkip1 = false;
}
} else {
if (!bSkip1) { // Only d is first byte
bSkip0 = false;
if ((0x80 <= d && d <= 0x9F) || 0xE0 <= d) { // Specifically 0x81~0x9F 0xE0~0xEF
bSkip1 = true;
}
} else { // Second byte for both c and d
bSkip0 = false;
bSkip1 = false;
}
}
// Compare
if (c == d)
continue;
if (c == '?')
continue;
return 1;
}
if (pBufFirst < pBufLast)
return 2;
return 0;
}
//---------------------------------------------------------------------------
//
/// Compare Human68k side name
//
//---------------------------------------------------------------------------
bool CHostPath::isSameHuman(const uint8_t* szHuman) const
{
assert(szHuman);
// Calulate number of chars
const size_t nLength = strlen((const char*)m_szHuman);
const size_t n = strlen((const char*)szHuman);
// Check number of chars
if (nLength != n)
return false;
// Compare Human68k path name
return Compare(m_szHuman, m_szHuman + nLength, szHuman, szHuman + n) == 0;
}
bool CHostPath::isSameChild(const uint8_t* szHuman) const
{
assert(szHuman);
// Calulate number of chars
size_t nLength = strlen((const char*)m_szHuman);
size_t n = strlen((const char*)szHuman);
// Check number of chars
if (nLength < n)
return false;
// Compare Human68k path name
return Compare(m_szHuman, m_szHuman + n, szHuman, szHuman + n) == 0;
}
//---------------------------------------------------------------------------
//
/// Find file name
///
/// Check if whether it is a perfect match with the cache buffer, and return the name if found.
/// Path names are excempted.
/// Make sure to lock from the top.
//
//---------------------------------------------------------------------------
const CHostFilename* CHostPath::FindFilename(const uint8_t* szHuman, uint32_t nHumanAttribute) const
{
assert(szHuman);
// Calulate number of chars
const uint8_t* pFirst = szHuman;
size_t nLength = strlen((const char*)pFirst);
const uint8_t* pLast = pFirst + nLength;
// Find something that matches perfectly with either of the stored file names
const ring_t* p = (ring_t*)m_cRing.Next();
for (; p != (const ring_t*)&m_cRing; p = (ring_t*)p->r.Next()) {
if (p->f.CheckAttribute(nHumanAttribute) == 0)
continue;
// Calulate number of chars
const uint8_t* pBufFirst = p->f.GetHuman();
const uint8_t* pBufLast = p->f.GetHumanLast();
// Check number of chars
if (size_t nBufLength = pBufLast - pBufFirst; nLength != nBufLength)
continue;
// File name check
if (Compare(pFirst, pLast, pBufFirst, pBufLast) == 0)
return &p->f;
}
return nullptr;
}
//---------------------------------------------------------------------------
//
/// Find file name (with wildcard support)
///
/// Check if whether it is a perfect match with the cache buffer, and return the name if found.
/// Path names are excempted.
/// Make sure to lock from the top.
//
//---------------------------------------------------------------------------
const CHostFilename* CHostPath::FindFilenameWildcard(const uint8_t* szHuman, uint32_t nHumanAttribute, find_t* pFind) const
{
assert(szHuman);
assert(pFind);
// Split the base file name and Human68k file extension
const uint8_t* pFirst = szHuman;
const uint8_t* pLast = pFirst + strlen((const char*)pFirst);
const uint8_t* pExt = CHostFilename::SeparateExt(pFirst);
// Move to the start position
auto p = (const ring_t*)m_cRing.Next();
if (pFind->count > 0) {
if (pFind->id == m_nId) {
// If the same directory entry, continue right away from the previous position
p = pFind->pos;
} else {
// Find the start position in the directory entry contents
uint32_t n = 0;
for (;; p = (const ring_t*)p->r.Next()) {
if (p == (const ring_t*)&m_cRing) {
// Extrapolate from the count when the same entry isn't found (just in case)
p = (const ring_t*)m_cRing.Next();
n = 0;
for (; p != (const ring_t*)&m_cRing; p = (const ring_t*)p->r.Next()) {
if (n >= pFind->count)
break;
n++;
}
break;
}
if (p->f.isSameEntry(&pFind->entry)) {
// Same entry is found
pFind->count = n;
break;
}
n++;
}
}
}
// Find files
for (; p != (const ring_t*)&m_cRing; p = (const ring_t*)p->r.Next()) {
pFind->count++;
if (p->f.CheckAttribute(nHumanAttribute) == 0)
continue;
// Split the base file name and Human68k file extension
const uint8_t* pBufFirst = p->f.GetHuman();
const uint8_t* pBufLast = p->f.GetHumanLast();
const uint8_t* pBufExt = p->f.GetHumanExt();
// Compare base file name
if (Compare(pFirst, pExt, pBufFirst, pBufExt))
continue;
// Compare Human68k extension
// In the case of a '.???' extension, match the Human68k extension without period.
if (strcmp((const char*)pExt, ".???") == 0 ||
Compare(pExt, pLast, pBufExt, pBufLast) == 0) {
// Store the contents of the next candidate's directory entry
const auto pNext = (const ring_t*)p->r.Next();
pFind->id = m_nId;
pFind->pos = pNext;
if (pNext != (const ring_t*)&m_cRing)
memcpy(&pFind->entry, pNext->f.GetEntry(), sizeof(pFind->entry));
else
memset(&pFind->entry, 0, sizeof(pFind->entry));
return &p->f;
}
}
pFind->id = m_nId;
pFind->pos = p;
memset(&pFind->entry, 0, sizeof(pFind->entry));
return nullptr;
}
//---------------------------------------------------------------------------
//
/// Confirm that the file update has been carried out
//
//---------------------------------------------------------------------------
bool CHostPath::isRefresh() const
{
return m_bRefresh;
}
//---------------------------------------------------------------------------
//
/// ASCII sort function
//
//---------------------------------------------------------------------------
int AsciiSort(const dirent **a, const dirent **b)
{
return strcmp((*a)->d_name, (*b)->d_name);
}
//---------------------------------------------------------------------------
//
/// Reconstruct the file
///
/// Here we carry out the first host side file system observation.
/// Always lock from the top.
//
//---------------------------------------------------------------------------
void CHostPath::Refresh()
{
assert(strlen(m_szHost) + 22 < FILEPATH_MAX);
// Store time stamp
Backup();
TCHAR szPath[FILEPATH_MAX];
strcpy(szPath, m_szHost);
// Update refresh flag
m_bRefresh = false;
// Store previous cache contents
CRing cRingBackup;
m_cRing.InsertRing(&cRingBackup);
// Register file name
bool bUpdate = false;
dirent **pd = nullptr;
int nument = 0;
int maxent = XM6_HOST_DIRENTRY_FILE_MAX;
for (int i = 0; i < maxent; i++) {
TCHAR szFilename[FILEPATH_MAX];
if (pd == nullptr) {
nument = scandir(S2U(szPath), &pd, nullptr, AsciiSort);
if (nument == -1) {
pd = nullptr;
break;
}
maxent = nument;
}
// When at the top level directory, exclude current and parent
const dirent* pe = pd[i];
if (m_szHuman[0] == '/' && m_szHuman[1] == 0) {
if (strcmp(pe->d_name, ".") == 0 || strcmp(pe->d_name, "..") == 0) {
continue;
}
}
// Get file name
strcpy(szFilename, U2S(pe->d_name));
// Allocate file name memory
ring_t* pRing = Alloc(strlen(szFilename));
CHostFilename* pFilename = &pRing->f;
pFilename->SetHost(szFilename);
// If there is a relevant file name in the previous cache, prioritize that for the Human68k name
auto pCache = (ring_t*)cRingBackup.Next();
for (;;) {
if (pCache == (ring_t*)&cRingBackup) {
pCache = nullptr; // No relevant entry
bUpdate = true; // Confirm new entry
pFilename->ConvertHuman();
break;
}
if (strcmp(pFilename->GetHost(), pCache->f.GetHost()) == 0) {
pFilename->CopyHuman(pCache->f.GetHuman()); // Copy Human68k name
break;
}
pCache = (ring_t*)pCache->r.Next();
}
// If there is a new entry, carry out file name duplication check.
// If the host side file name changed, or if Human68k cannot express the file name,
// generate a new file name that passes all the below checks:
// - File name correctness
// - No duplicated names in previous entries
// - No entity with the same name exists
if (pFilename->isReduce() || !pFilename->isCorrect()) { // Confirm that file name update is required
for (uint32_t n = 0; n < XM6_HOST_FILENAME_PATTERN_MAX; n++) {
// Confirm file name validity
if (pFilename->isCorrect()) {
// Confirm match with previous entry
const CHostFilename* pCheck = FindFilename(pFilename->GetHuman());
if (pCheck == nullptr) {
// If no match, confirm existence of real file
strcpy(szPath, m_szHost);
strcat(szPath, (const char*)pFilename->GetHuman());
if (struct stat sb; stat(S2U(szPath), &sb))
break; // Discover available patterns
}
}
// Generate new name
pFilename->ConvertHuman(n);
}
}
// Directory entry name
pFilename->SetEntryName();
// Get data
strcpy(szPath, m_szHost);
strcat(szPath, U2S(pe->d_name));
struct stat sb;
if (stat(S2U(szPath), &sb))
continue;
uint8_t nHumanAttribute = Human68k::AT_ARCHIVE;
if (S_ISDIR(sb.st_mode))
nHumanAttribute = Human68k::AT_DIRECTORY;
if ((sb.st_mode & 0200) == 0)
nHumanAttribute |= Human68k::AT_READONLY;
pFilename->SetEntryAttribute(nHumanAttribute);
auto nHumanSize = (uint32_t)sb.st_size;
pFilename->SetEntrySize(nHumanSize);
uint16_t nHumanDate = 0;
uint16_t nHumanTime = 0;
if (tm pt = {}; localtime_r(&sb.st_mtime, &pt) != nullptr) {
nHumanDate = (uint16_t)(((pt.tm_year - 80) << 9) | ((pt.tm_mon + 1) << 5) | pt.tm_mday);
nHumanTime = (uint16_t)((pt.tm_hour << 11) | (pt.tm_min << 5) | (pt.tm_sec >> 1));
}
pFilename->SetEntryDate(nHumanDate);
pFilename->SetEntryTime(nHumanTime);
pFilename->SetEntryCluster(0);
// Compare with previous cached contents
if (pCache) {
if (pCache->f.isSameEntry(pFilename->GetEntry())) {
Free(pRing); // Destroy entry that was created here
pRing = pCache; // Use previous cache
} else {
Free(pCache); // Remove from the next search target
bUpdate = true; // Flag for update if no match
}
}
// Add to end of ring
pRing->r.InsertTail(&m_cRing);
}
// Release directory entry
if (pd) {
for (int i = 0; i < nument; i++) {
free(pd[i]);
}
free(pd);
}
// Delete remaining cache
ring_t* p;
while ((p = (ring_t*)cRingBackup.Next()) != (ring_t*)&cRingBackup) {
bUpdate = true; // Confirms the decrease in entries due to deletion
Free(p);
}
// Update the identifier if the update has been carried out
if (bUpdate)
m_nId = ++g_nId;
}
//---------------------------------------------------------------------------
//
/// Store the host side time stamp
//
//---------------------------------------------------------------------------
void CHostPath::Backup()
{
assert(m_szHost);
assert(strlen(m_szHost) < FILEPATH_MAX);
TCHAR szPath[FILEPATH_MAX];
strcpy(szPath, m_szHost);
size_t len = strlen(szPath);
m_tBackup = 0;
if (len > 1) { // Don't do anything if it is the root directory
len--;
assert(szPath[len] == '/');
szPath[len] = '\0';
if (struct stat sb; stat(S2U(szPath), &sb) == 0)
m_tBackup = sb.st_mtime;
}
}
//---------------------------------------------------------------------------
//
/// Restore the host side time stamp
//
//---------------------------------------------------------------------------
void CHostPath::Restore() const
{
assert(m_szHost);
assert(strlen(m_szHost) < FILEPATH_MAX);
TCHAR szPath[FILEPATH_MAX];
strcpy(szPath, m_szHost);
size_t len = strlen(szPath);
if (m_tBackup) {
assert(len);
len--;
assert(szPath[len] == '/');
szPath[len] = '\0';
utimbuf ut;
ut.actime = m_tBackup;
ut.modtime = m_tBackup;
utime(szPath, &ut);
}
}
//---------------------------------------------------------------------------
//
/// Update
//
//---------------------------------------------------------------------------
void CHostPath::Release()
{
m_bRefresh = true;
}
//===========================================================================
//
// Manage directory entries
//
//===========================================================================
CHostEntry::~CHostEntry()
{
Clean();
#ifdef DEBUG
// Confirm object
for (const auto& d : m_pDrv) {
assert(d == nullptr);
}
#endif
}
//---------------------------------------------------------------------------
//
/// Initialize (when the driver is installed)
//
//---------------------------------------------------------------------------
void CHostEntry::Init() const
{
#ifdef DEBUG
// Confirm object
for (const auto& d : m_pDrv) {
assert(d == nullptr);
}
#endif
}
//---------------------------------------------------------------------------
//
/// Release (at startup and reset)
//
//---------------------------------------------------------------------------
void CHostEntry::Clean()
{
// Delete object
for (auto& d: m_pDrv) {
delete d;
d = nullptr;
}
}
//---------------------------------------------------------------------------
//
/// Update all cache
//
//---------------------------------------------------------------------------
void CHostEntry::CleanCache() const
{
for (const auto& d : m_pDrv) {
if (d)
d->CleanCache();
}
CHostPath::InitId();
}
//---------------------------------------------------------------------------
//
/// Update the cache for the specified unit
//
//---------------------------------------------------------------------------
void CHostEntry::CleanCache(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->CleanCache();
}
//---------------------------------------------------------------------------
//
/// Update the cache for the specified path
//
//---------------------------------------------------------------------------
void CHostEntry::CleanCache(uint32_t nUnit, const uint8_t* szHumanPath) const
{
assert(szHumanPath);
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->CleanCache(szHumanPath);
}
//---------------------------------------------------------------------------
//
/// Update all cache for the specified path and below
//
//---------------------------------------------------------------------------
void CHostEntry::CleanCacheChild(uint32_t nUnit, const uint8_t* szHumanPath) const
{
assert(szHumanPath);
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->CleanCacheChild(szHumanPath);
}
//---------------------------------------------------------------------------
//
/// Delete cache for the specified path
//
//---------------------------------------------------------------------------
void CHostEntry::DeleteCache(uint32_t nUnit, const uint8_t* szHumanPath) const
{
assert(szHumanPath);
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->DeleteCache(szHumanPath);
}
//---------------------------------------------------------------------------
//
/// Find host side names (path name + file name (can be abbreviated) + attribute)
//
//---------------------------------------------------------------------------
bool CHostEntry::Find(uint32_t nUnit, CHostFiles* pFiles) const
{
assert(pFiles);
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->Find(pFiles);
}
//---------------------------------------------------------------------------
//
/// Drive settings
//
//---------------------------------------------------------------------------
void CHostEntry::SetDrv(uint32_t nUnit, CHostDrv* pDrv)
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit] == nullptr);
m_pDrv[nUnit] = pDrv;
}
//---------------------------------------------------------------------------
//
/// Is it write-protected?
//
//---------------------------------------------------------------------------
bool CHostEntry::isWriteProtect(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->isWriteProtect();
}
//---------------------------------------------------------------------------
//
/// Is it accessible?
//
//---------------------------------------------------------------------------
bool CHostEntry::isEnable(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->isEnable();
}
//---------------------------------------------------------------------------
//
/// Media check
//
//---------------------------------------------------------------------------
bool CHostEntry::isMediaOffline(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->isMediaOffline();
}
//---------------------------------------------------------------------------
//
/// Get media byte
//
//---------------------------------------------------------------------------
uint8_t CHostEntry::GetMediaByte(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->GetMediaByte();
}
//---------------------------------------------------------------------------
//
/// Get drive status
//
//---------------------------------------------------------------------------
uint32_t CHostEntry::GetStatus(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->GetStatus();
}
//---------------------------------------------------------------------------
//
/// Media change check
//
//---------------------------------------------------------------------------
bool CHostEntry::CheckMedia(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->CheckMedia();
}
//---------------------------------------------------------------------------
//
/// Eject
//
//---------------------------------------------------------------------------
void CHostEntry::Eject(uint32_t nUnit) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->Eject();
}
//---------------------------------------------------------------------------
//
/// Get volume label
//
//---------------------------------------------------------------------------
void CHostEntry::GetVolume(uint32_t nUnit, TCHAR* szLabel) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
m_pDrv[nUnit]->GetVolume(szLabel);
}
//---------------------------------------------------------------------------
//
/// Get volume label from cache
//
//---------------------------------------------------------------------------
bool CHostEntry::GetVolumeCache(uint32_t nUnit, TCHAR* szLabel) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->GetVolumeCache(szLabel);
}
//---------------------------------------------------------------------------
//
/// Get capacity
//
//---------------------------------------------------------------------------
uint32_t CHostEntry::GetCapacity(uint32_t nUnit, Human68k::capacity_t* pCapacity) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->GetCapacity(pCapacity);
}
//---------------------------------------------------------------------------
//
/// Get cluster size from cache
//
//---------------------------------------------------------------------------
bool CHostEntry::GetCapacityCache(uint32_t nUnit, Human68k::capacity_t* pCapacity) const
{
assert(nUnit < DRIVE_MAX);
assert(m_pDrv[nUnit]);
return m_pDrv[nUnit]->GetCapacityCache(pCapacity);
}
//---------------------------------------------------------------------------
//
/// Split and copy the first element from the Human68k full path name
///
/// Get the first element from the Human68k full path name and delete the path separator char.
/// 23 bytes is required in the buffer to write to.
/// A Human68k path always starts with a '/'.
/// Throw an error if 2 '/' appears in sequence.
/// If the array ends with a '/' treat it as an empty array and don't trow an error.
//
//---------------------------------------------------------------------------
const uint8_t* CHostDrv::SeparateCopyFilename(const uint8_t* szHuman, uint8_t* szBuffer) // static
{
assert(szHuman);
assert(szBuffer);
const size_t nMax = 22;
const uint8_t* p = szHuman;
uint8_t c = *p++; // Read
if (c != '/' && c != '\\')
return nullptr; // Error: Invalid path name
// Insert one file
size_t i = 0;
for (;;) {
c = *p; // Read
if (c == '\0')
break; // Exit if at the end of an array (return the end position)
if (c == '/' || c == '\\') {
if (i == 0)
return nullptr; // Error: Two separator chars appear in sequence
break; // Exit after reading the separator (return the char position)
}
p++;
if (i >= nMax)
return nullptr; // Error: The first byte hits the end of the buffer
szBuffer[i++] = c; // Read
if ((0x80 <= c && c <= 0x9F) || 0xE0 <= c) { // Specifically 0x81~0x9F and 0xE0~0xEF
c = *p++; // Read
if (c < 0x40)
return nullptr; // Error: Invalid Shift-JIS 2nd byte
if (i >= nMax)
return nullptr; // Error: The second byte hits the end of the buffer
szBuffer[i++] = c; // Read
}
}
szBuffer[i] = '\0'; // Read
return p;
}
//---------------------------------------------------------------------------
//
/// Generate path and file name internally
//
//---------------------------------------------------------------------------
void CHostFiles::SetPath(const Human68k::namests_t* pNamests)
{
assert(pNamests);
pNamests->GetCopyPath(m_szHumanPath);
pNamests->GetCopyFilename(m_szHumanFilename);
m_nHumanWildcard = 0;
m_nHumanAttribute = Human68k::AT_ARCHIVE;
m_findNext.Clear();
}
//---------------------------------------------------------------------------
//
/// Find file on the Human68k side and create data on the host side
//
//---------------------------------------------------------------------------
bool CHostFiles::Find(uint32_t nUnit, const CHostEntry* pEntry)
{
assert(pEntry);
return pEntry->Find(nUnit, this);
}
//---------------------------------------------------------------------------
//
/// Find file name
//
//---------------------------------------------------------------------------
const CHostFilename* CHostFiles::Find(const CHostPath* pPath)
{
assert(pPath);
if (m_nHumanWildcard)
return pPath->FindFilenameWildcard(m_szHumanFilename, m_nHumanAttribute, &m_findNext);
return pPath->FindFilename(m_szHumanFilename, m_nHumanAttribute);
}
//---------------------------------------------------------------------------
//
/// Store the Human68k side search results
//
//---------------------------------------------------------------------------
void CHostFiles::SetEntry(const CHostFilename* pFilename)
{
assert(pFilename);
// Store Human68k directory entry
memcpy(&m_dirHuman, pFilename->GetEntry(), sizeof(m_dirHuman));
// Stire Human68k file name
strcpy((char*)m_szHumanResult, (const char*)pFilename->GetHuman());
}
//---------------------------------------------------------------------------
//
/// Set host side name
//
//---------------------------------------------------------------------------
void CHostFiles::SetResult(const TCHAR* szPath)
{
assert(szPath);
assert(strlen(szPath) < FILEPATH_MAX);
strcpy(m_szHostResult, szPath);
}
//---------------------------------------------------------------------------
//
/// Add file name to the host side name
//
//---------------------------------------------------------------------------
void CHostFiles::AddResult(const TCHAR* szPath)
{
assert(szPath);
assert(strlen(m_szHostResult) + strlen(szPath) < FILEPATH_MAX);
strcat(m_szHostResult, szPath);
}
//---------------------------------------------------------------------------
//
/// Add a new Human68k file name to the host side name
//
//---------------------------------------------------------------------------
void CHostFiles::AddFilename()
{
assert(strlen(m_szHostResult) + strlen((const char*)m_szHumanFilename) < FILEPATH_MAX);
strncat(m_szHostResult, (const char*)m_szHumanFilename, ARRAY_SIZE(m_szHumanFilename));
}
//===========================================================================
//
// File search memory manager
//
//===========================================================================
#ifdef _DEBUG
CHostFilesManager::~CHostFilesManager()
{
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
}
#endif // _DEBUG
//---------------------------------------------------------------------------
//
/// Initialization (when the driver is installed)
//
//---------------------------------------------------------------------------
void CHostFilesManager::Init()
{
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
// Allocate memory
for (uint32_t i = 0; i < XM6_HOST_FILES_MAX; i++) {
auto p = new ring_t();
p->r.Insert(&m_cRing);
}
}
//---------------------------------------------------------------------------
//
/// Release (at startup and reset)
//
//---------------------------------------------------------------------------
void CHostFilesManager::Clean()
{
// Release memory
CRing* p;
while ((p = m_cRing.Next()) != &m_cRing) {
delete (ring_t*)p;
}
}
CHostFiles* CHostFilesManager::Alloc(uint32_t nKey)
{
assert(nKey);
// Select from the end
auto p = (ring_t*)m_cRing.Prev();
// Move to the start of the ring
p->r.Insert(&m_cRing);
p->f.SetKey(nKey);
return &p->f;
}
CHostFiles* CHostFilesManager::Search(uint32_t nKey)
{
// assert(nKey); // The search key may become 0 due to DPB damage
// Find the relevant object
auto p = (ring_t*)m_cRing.Next();
for (; p != (ring_t*)&m_cRing; p = (ring_t*)p->r.Next()) {
if (p->f.isSameKey(nKey)) {
// Move to the start of the ring
p->r.Insert(&m_cRing);
return &p->f;
}
}
return nullptr;
}
void CHostFilesManager::Free(CHostFiles* pFiles)
{
assert(pFiles);
// Release
pFiles->SetKey(0);
// Move to the end of the ring
auto p = (ring_t*)((size_t)pFiles - offsetof(ring_t, f));
p->r.InsertTail(&m_cRing);
}
//===========================================================================
//
// FCB processing
//
//===========================================================================
//---------------------------------------------------------------------------
//
/// Set file open mode
//
//---------------------------------------------------------------------------
bool CHostFcb::SetMode(uint32_t nHumanMode)
{
switch (nHumanMode & Human68k::OP_MASK) {
case Human68k::OP_READ:
m_pszMode = "rb";
break;
case Human68k::OP_WRITE:
m_pszMode = "wb";
break;
case Human68k::OP_FULL:
m_pszMode = "r+b";
break;
default:
return false;
}
m_bFlag = (nHumanMode & Human68k::OP_SPECIAL) != 0;
return true;
}
void CHostFcb::SetFilename(const TCHAR* szFilename)
{
assert(szFilename);
assert(strlen(szFilename) < FILEPATH_MAX);
strcpy(m_szFilename, szFilename);
}
void CHostFcb::SetHumanPath(const uint8_t* szHumanPath)
{
assert(szHumanPath);
assert(strlen((const char*)szHumanPath) < HUMAN68K_PATH_MAX);
strcpy((char*)m_szHumanPath, (const char*)szHumanPath);
}
//---------------------------------------------------------------------------
//
/// Create file
///
/// Return false if error is thrown.
//
//---------------------------------------------------------------------------
bool CHostFcb::Create(uint32_t, bool bForce)
{
assert((Human68k::AT_DIRECTORY | Human68k::AT_VOLUME) == 0);
assert(strlen(m_szFilename) > 0);
assert(m_pFile == nullptr);
// Duplication check
if (!bForce) {
if (struct stat sb; stat(S2U(m_szFilename), &sb) == 0)
return false;
}
// Create file
m_pFile = fopen(S2U(m_szFilename), "w+b"); /// @warning The ideal operation is to overwrite each attribute
return m_pFile != nullptr;
}
//---------------------------------------------------------------------------
//
/// File open
///
/// Return false if error is thrown.
//
//---------------------------------------------------------------------------
bool CHostFcb::Open()
{
assert(strlen(m_szFilename) > 0);
// Fail if directory
if (struct stat st; stat(S2U(m_szFilename), &st) == 0 && ((st.st_mode & S_IFMT) == S_IFDIR)) {
return false || m_bFlag;
}
// File open
if (m_pFile == nullptr)
m_pFile = fopen(S2U(m_szFilename), m_pszMode);
return m_pFile != nullptr || m_bFlag;
}
//---------------------------------------------------------------------------
//
/// Read file
///
/// Handle a 0 byte read as normal operation too.
/// Return -1 if error is thrown.
//
//---------------------------------------------------------------------------
uint32_t CHostFcb::Read(uint8_t* pBuffer, uint32_t nSize)
{
assert(pBuffer);
assert(m_pFile);
size_t nResult = fread(pBuffer, sizeof(uint8_t), nSize, m_pFile);
if (ferror(m_pFile))
nResult = (size_t)-1;
return (uint32_t)nResult;
}
//---------------------------------------------------------------------------
//
/// Write file
///
/// Handle a 0 byte read as normal operation too.
/// Return -1 if error is thrown.
//
//---------------------------------------------------------------------------
uint32_t CHostFcb::Write(const uint8_t* pBuffer, uint32_t nSize)
{
assert(pBuffer);
assert(m_pFile);
size_t nResult = fwrite(pBuffer, sizeof(uint8_t), nSize, m_pFile);
if (ferror(m_pFile))
nResult = (size_t)-1;
return (uint32_t)nResult;
}
//---------------------------------------------------------------------------
//
/// Truncate file
///
/// Return false if error is thrown.
//
//---------------------------------------------------------------------------
bool CHostFcb::Truncate() const
{
assert(m_pFile);
return ftruncate(fileno(m_pFile), ftell(m_pFile)) == 0;
}
//---------------------------------------------------------------------------
//
/// File seek
///
/// Return -1 if error is thrown.
//
//---------------------------------------------------------------------------
uint32_t CHostFcb::Seek(uint32_t nOffset, Human68k::seek_t nHumanSeek)
{
assert(m_pFile);
int nSeek;
switch (nHumanSeek) {
case Human68k::seek_t::SK_BEGIN:
nSeek = SEEK_SET;
break;
case Human68k::seek_t::SK_CURRENT:
nSeek = SEEK_CUR;
break;
// case SK_END:
default:
nSeek = SEEK_END;
break;
}
if (fseek(m_pFile, nOffset, nSeek))
return (uint32_t)-1;
return (uint32_t)ftell(m_pFile);
}
//---------------------------------------------------------------------------
//
/// Set file time stamp
///
/// Return false if error is thrown.
//
//---------------------------------------------------------------------------
bool CHostFcb::TimeStamp(uint32_t nHumanTime) const
{
assert(m_pFile || m_bFlag);
tm t = { };
t.tm_year = (nHumanTime >> 25) + 80;
t.tm_mon = ((nHumanTime >> 21) - 1) & 15;
t.tm_mday = (nHumanTime >> 16) & 31;
t.tm_hour = (nHumanTime >> 11) & 31;
t.tm_min = (nHumanTime >> 5) & 63;
t.tm_sec = (nHumanTime & 31) << 1;
time_t ti = mktime(&t);
if (ti == (time_t)-1)
return false;
utimbuf ut;
ut.actime = ti;
ut.modtime = ti;
// This is for preventing the last updated time stamp to be overwritten upon closing.
// Flush and synchronize before updating the time stamp.
fflush(m_pFile);
return utime(S2U(m_szFilename), &ut) == 0 || m_bFlag;
}
//---------------------------------------------------------------------------
//
/// File close
///
/// Return false if error is thrown.
//
//---------------------------------------------------------------------------
void CHostFcb::Close()
{
// File close
// Always initialize because of the Close→Free (internally one more Close) flow.
if (m_pFile) {
fclose(m_pFile);
m_pFile = nullptr;
}
}
//===========================================================================
//
// FCB processing manager
//
//===========================================================================
#ifdef _DEBUG
CHostFcbManager::~CHostFcbManager()
{
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
}
#endif // _DEBUG
//---------------------------------------------------------------------------
//
// Initialization (when the driver is installed)
//
//---------------------------------------------------------------------------
void CHostFcbManager::Init()
{
// Confirm that the entity does not exist (just in case)
assert(m_cRing.Next() == &m_cRing);
assert(m_cRing.Prev() == &m_cRing);
// Memory allocation
for (uint32_t i = 0; i < XM6_HOST_FCB_MAX; i++) {
auto p = new ring_t;
assert(p);
p->r.Insert(&m_cRing);
}
}
//---------------------------------------------------------------------------
//
// Clean (at startup/reset)
//
//---------------------------------------------------------------------------
void CHostFcbManager::Clean() const
{
// Fast task killer
CRing* p;
while ((p = m_cRing.Next()) != &m_cRing) {
delete (ring_t*)p;
}
}
CHostFcb* CHostFcbManager::Alloc(uint32_t nKey)
{
assert(nKey);
// Select from the end
auto p = (ring_t*)m_cRing.Prev();
// Error if in use (just in case)
if (!p->f.isSameKey(0)) {
assert(false);
return nullptr;
}
// Move to the top of the ring
p->r.Insert(&m_cRing);
// Set key
p->f.SetKey(nKey);
return &p->f;
}
CHostFcb* CHostFcbManager::Search(uint32_t nKey)
{
assert(nKey);
// Search for applicable objects
auto p = (ring_t*)m_cRing.Next();
while (p != (ring_t*)&m_cRing) {
if (p->f.isSameKey(nKey)) {
// Move to the top of the ring
p->r.Insert(&m_cRing);
return &p->f;
}
p = (ring_t*)p->r.Next();
}
return nullptr;
}
void CHostFcbManager::Free(CHostFcb* pFcb)
{
assert(pFcb);
// Free
pFcb->SetKey(0);
pFcb->Close();
// Move to the end of the ring
auto p = (ring_t*)((size_t)pFcb - offsetof(ring_t, f));
p->r.InsertTail(&m_cRing);
}
//===========================================================================
//
// Host side file system
//
//===========================================================================
uint32_t CFileSys::g_nOption; // File name conversion flag
//---------------------------------------------------------------------------
//
/// Reset (close all)
//
//---------------------------------------------------------------------------
void CFileSys::Reset()
{
// Initialize virtual sectors
m_nHostSectorCount = 0;
memset(m_nHostSectorBuffer, 0, sizeof(m_nHostSectorBuffer));
// File search memory - release (on startup and reset)
m_cFiles.Clean();
// FCB operation memory (on startup and reset)
m_cFcb.Clean();
// Directory entry - release (on startup and reset)
m_cEntry.Clean();
// Initialize TwentyOne option monitoring
m_nKernel = 0;
m_nKernelSearch = 0;
// Initialize operational flags
SetOption(m_nOptionDefault);
}
//---------------------------------------------------------------------------
//
/// Initialize (device startup and load)
//
//---------------------------------------------------------------------------
void CFileSys::Init()
{
// Initialize file search memory (device startup and load)
m_cFiles.Init();
// Initialize FCB operation memory (device startup and load)
m_cFcb.Init();
// Initialize directory entries (device startup and load)
m_cEntry.Init();
// Evaluate per-path setting validity
uint32_t nDrives = m_nDrives;
if (nDrives == 0) {
// Use root directory instead of per-path settings
strcpy(m_szBase[0], "/");
m_nFlag[0] = 0;
nDrives++;
}
// Register file system
uint32_t nUnit = 0;
for (uint32_t n = 0; n < nDrives; n++) {
// Don't register is base path do not exist
if (m_szBase[n][0] == '\0')
continue;
// Create 1 unit file system
auto p = new CHostDrv;
if (p) {
m_cEntry.SetDrv(nUnit, p);
p->Init(m_szBase[n], m_nFlag[n]);
// To the next unit
nUnit++;
}
}
// Store the registered number of drives
m_nUnits = nUnit;
}
//---------------------------------------------------------------------------
//
/// $40 - Device startup
//
//---------------------------------------------------------------------------
uint32_t CFileSys::InitDevice(const Human68k::argument_t* pArgument)
{
InitOption(pArgument);
// File system initialization
Init();
return m_nUnits;
}
//---------------------------------------------------------------------------
//
/// $41 - Directory check
//
//---------------------------------------------------------------------------
int CFileSys::CheckDir(uint32_t nUnit, const Human68k::namests_t* pNamests) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_INVALIDFUNC; // Avoid triggering a fatal error in mint when resuming with an invalid drive
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
if (f.isRootPath())
return 0;
f.SetPathOnly();
if (!f.Find(nUnit, &m_cEntry))
return FS_DIRNOTFND;
return 0;
}
//---------------------------------------------------------------------------
//
/// $42 - Create directory
//
//---------------------------------------------------------------------------
int CFileSys::MakeDir(uint32_t nUnit, const Human68k::namests_t* pNamests) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetPathOnly();
if (!f.Find(nUnit, &m_cEntry))
return FS_INVALIDPATH;
f.AddFilename();
// Create directory
if (mkdir(S2U(f.GetPath()), 0777))
return FS_INVALIDPATH;
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $43 - Delete directory
//
//---------------------------------------------------------------------------
int CFileSys::RemoveDir(uint32_t nUnit, const Human68k::namests_t* pNamests) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetAttribute(Human68k::AT_DIRECTORY);
if (!f.Find(nUnit, &m_cEntry))
return FS_DIRNOTFND;
// Delete cache
uint8_t szHuman[HUMAN68K_PATH_MAX + 24];
assert(strlen((const char*)f.GetHumanPath()) +
strlen((const char*)f.GetHumanFilename()) < HUMAN68K_PATH_MAX + 24);
strcpy((char*)szHuman, (const char*)f.GetHumanPath());
strcat((char*)szHuman, (const char*)f.GetHumanFilename());
strcat((char*)szHuman, "/");
m_cEntry.DeleteCache(nUnit, szHuman);
// Delete directory
if (rmdir(S2U(f.GetPath())))
return FS_CANTDELETE;
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $44 - Change file name
//
//---------------------------------------------------------------------------
int CFileSys::Rename(uint32_t nUnit, const Human68k::namests_t* pNamests, const Human68k::namests_t* pNamestsNew) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetAttribute(Human68k::AT_ALL);
if (!f.Find(nUnit, &m_cEntry))
return FS_FILENOTFND;
CHostFiles fNew;
fNew.SetPath(pNamestsNew);
fNew.SetPathOnly();
if (!fNew.Find(nUnit, &m_cEntry))
return FS_INVALIDPATH;
fNew.AddFilename();
// Update cache
if (f.GetAttribute() & Human68k::AT_DIRECTORY)
m_cEntry.CleanCacheChild(nUnit, f.GetHumanPath());
// Change file name
char szFrom[FILENAME_MAX];
char szTo[FILENAME_MAX];
SJIS2UTF8(f.GetPath(), szFrom, FILENAME_MAX);
SJIS2UTF8(fNew.GetPath(), szTo, FILENAME_MAX);
if (rename(szFrom, szTo)) {
return FS_FILENOTFND;
}
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
m_cEntry.CleanCache(nUnit, fNew.GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $45 - Delete file
//
//---------------------------------------------------------------------------
int CFileSys::Delete(uint32_t nUnit, const Human68k::namests_t* pNamests) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
if (!f.Find(nUnit, &m_cEntry))
return FS_FILENOTFND;
// Delete file
if (unlink(S2U(f.GetPath())))
return FS_CANTDELETE;
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $46 - Get/set file attribute
//
//---------------------------------------------------------------------------
int CFileSys::Attribute(uint32_t nUnit, const Human68k::namests_t* pNamests, uint32_t nHumanAttribute) const
{
assert(pNamests);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetAttribute(Human68k::AT_ALL);
if (!f.Find(nUnit, &m_cEntry))
return FS_FILENOTFND;
// Exit if attribute is acquired
if (nHumanAttribute == 0xFF)
return f.GetAttribute();
// Attribute check
if (nHumanAttribute & Human68k::AT_VOLUME)
return FS_CANTACCESS;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Generate attribute
if (uint32_t nAttribute = (nHumanAttribute & Human68k::AT_READONLY) | (f.GetAttribute() & ~Human68k::AT_READONLY);
f.GetAttribute() != nAttribute) {
struct stat sb;
if (stat(S2U(f.GetPath()), &sb))
return FS_FILENOTFND;
mode_t m = sb.st_mode & 0777;
if (nAttribute & Human68k::AT_READONLY)
m &= 0555; // ugo-w
else
m |= 0200; // u+w
// Set attribute
if (chmod(S2U(f.GetPath()), m))
return FS_FILENOTFND;
}
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
// Get attribute after changing
if (!f.Find(nUnit, &m_cEntry))
return FS_FILENOTFND;
return f.GetAttribute();
}
//---------------------------------------------------------------------------
//
/// $47 - File search
//
//---------------------------------------------------------------------------
int CFileSys::Files(uint32_t nUnit, uint32_t nKey, const Human68k::namests_t* pNamests, Human68k::files_t* pFiles)
{
assert(pNamests);
assert(nKey);
assert(pFiles);
// Release if memory with the same key already exists
CHostFiles* pHostFiles = m_cFiles.Search(nKey);
if (pHostFiles != nullptr) {
m_cFiles.Free(pHostFiles);
}
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Get volume label
/** @note
This skips the media check when getting the volume label to avoid triggering a fatal error
with host side removable media (CD-ROM, etc.) Some poorly coded applications will attempt to
get the volume label even though a proper error was thrown doing a media change check just before.
*/
if ((pFiles->fatr & (Human68k::AT_ARCHIVE | Human68k::AT_DIRECTORY | Human68k::AT_VOLUME))
== Human68k::AT_VOLUME) {
// Path check
CHostFiles f;
f.SetPath(pNamests);
if (!f.isRootPath())
return FS_FILENOTFND;
// Immediately return the results without allocating buffer
if (!FilesVolume(nUnit, pFiles))
return FS_FILENOTFND;
return 0;
}
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Allocate buffer
pHostFiles = m_cFiles.Alloc(nKey);
if (pHostFiles == nullptr)
return FS_OUTOFMEM;
// Directory check
pHostFiles->SetPath(pNamests);
if (!pHostFiles->isRootPath()) {
pHostFiles->SetPathOnly();
if (!pHostFiles->Find(nUnit, &m_cEntry)) {
m_cFiles.Free(pHostFiles);
return FS_DIRNOTFND;
}
}
// Enable wildcards
pHostFiles->SetPathWildcard();
pHostFiles->SetAttribute(pFiles->fatr);
// Find file
if (!pHostFiles->Find(nUnit, &m_cEntry)) {
m_cFiles.Free(pHostFiles);
return FS_FILENOTFND;
}
// Store search results
pFiles->attr = (uint8_t)pHostFiles->GetAttribute();
pFiles->date = pHostFiles->GetDate();
pFiles->time = pHostFiles->GetTime();
pFiles->size = pHostFiles->GetSize();
strcpy((char*)pFiles->full, (const char*)pHostFiles->GetHumanResult());
// Specify pseudo-directory entry
pFiles->sector = nKey;
pFiles->offset = 0;
return 0;
}
//---------------------------------------------------------------------------
//
/// $48 - Search next file
//
//---------------------------------------------------------------------------
int CFileSys::NFiles(uint32_t nUnit, uint32_t nKey, Human68k::files_t* pFiles)
{
assert(nKey);
assert(pFiles);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Find buffer
CHostFiles* pHostFiles = m_cFiles.Search(nKey);
if (pHostFiles == nullptr)
return FS_INVALIDPTR;
// Find file
if (!pHostFiles->Find(nUnit, &m_cEntry)) {
m_cFiles.Free(pHostFiles);
return FS_FILENOTFND;
}
assert(pFiles->sector == nKey);
assert(pFiles->offset == 0);
// Store search results
pFiles->attr = (uint8_t)pHostFiles->GetAttribute();
pFiles->date = pHostFiles->GetDate();
pFiles->time = pHostFiles->GetTime();
pFiles->size = pHostFiles->GetSize();
strcpy((char*)pFiles->full, (const char*)pHostFiles->GetHumanResult());
return 0;
}
//---------------------------------------------------------------------------
//
/// $49 - Create new file
//
//---------------------------------------------------------------------------
int CFileSys::Create(uint32_t nUnit, uint32_t nKey, const Human68k::namests_t* pNamests, Human68k::fcb_t* pFcb, uint32_t nHumanAttribute, bool bForce)
{
assert(pNamests);
assert(nKey);
assert(pFcb);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Release if memory with the same key already exists
if (m_cFcb.Search(nKey) != nullptr)
return FS_INVALIDPTR;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetPathOnly();
if (!f.Find(nUnit, &m_cEntry))
return FS_INVALIDPATH;
f.AddFilename();
// Attribute check
if (nHumanAttribute & (Human68k::AT_DIRECTORY | Human68k::AT_VOLUME))
return FS_CANTACCESS;
// Store path name
CHostFcb* pHostFcb = m_cFcb.Alloc(nKey);
if (pHostFcb == nullptr)
return FS_OUTOFMEM;
pHostFcb->SetFilename(f.GetPath());
pHostFcb->SetHumanPath(f.GetHumanPath());
// Set open mode
pFcb->mode = (uint16_t)((pFcb->mode & ~Human68k::OP_MASK) | Human68k::OP_FULL);
if (!pHostFcb->SetMode(pFcb->mode)) {
m_cFcb.Free(pHostFcb);
return FS_ILLEGALMOD;
}
// Create file
if (!pHostFcb->Create(nHumanAttribute, bForce)) {
m_cFcb.Free(pHostFcb);
return FS_FILEEXIST;
}
// Update cache
m_cEntry.CleanCache(nUnit, f.GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $4A - File open
//
//---------------------------------------------------------------------------
int CFileSys::Open(uint32_t nUnit, uint32_t nKey, const Human68k::namests_t* pNamests, Human68k::fcb_t* pFcb)
{
assert(pNamests);
assert(nKey);
assert(pFcb);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
switch (pFcb->mode & Human68k::OP_MASK) {
case Human68k::OP_WRITE:
case Human68k::OP_FULL:
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
break;
default: break;
}
// Release if memory with the same key already exists
if (m_cFcb.Search(nKey) != nullptr)
return FS_INVALIDPRM;
// Generate path name
CHostFiles f;
f.SetPath(pNamests);
f.SetAttribute(Human68k::AT_ALL);
if (!f.Find(nUnit, &m_cEntry))
return FS_FILENOTFND;
// Time stamp
pFcb->date = f.GetDate();
pFcb->time = f.GetTime();
// File size
pFcb->size = f.GetSize();
// Store path name
CHostFcb* pHostFcb = m_cFcb.Alloc(nKey);
if (pHostFcb == nullptr)
return FS_OUTOFMEM;
pHostFcb->SetFilename(f.GetPath());
pHostFcb->SetHumanPath(f.GetHumanPath());
// Set open mode
if (!pHostFcb->SetMode(pFcb->mode)) {
m_cFcb.Free(pHostFcb);
return FS_ILLEGALMOD;
}
// File open
if (!pHostFcb->Open()) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDPATH;
}
return 0;
}
//---------------------------------------------------------------------------
//
/// $4B - File close
//
//---------------------------------------------------------------------------
int CFileSys::Close(uint32_t nUnit, uint32_t nKey, const Human68k::fcb_t* /* pFcb */)
{
assert(nKey);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Throw error if memory with the same key does not exist
CHostFcb* pHostFcb = m_cFcb.Search(nKey);
if (pHostFcb == nullptr)
return FS_INVALIDPRM;
// File close and release memory
m_cFcb.Free(pHostFcb);
// Update cache
if (pHostFcb->isUpdate())
m_cEntry.CleanCache(nUnit, pHostFcb->GetHumanPath());
/// TODO: Match the FCB status on close with other devices
return 0;
}
//---------------------------------------------------------------------------
//
/// $4C - Read file
///
/// Clean exit when 0 bytes are read.
//
//---------------------------------------------------------------------------
int CFileSys::Read(uint32_t nKey, Human68k::fcb_t* pFcb, uint8_t* pBuffer, uint32_t nSize)
{
assert(nKey);
assert(pFcb);
// assert(pBuffer); // Evaluate only when needed
assert(nSize <= 0xFFFFFF); // Clipped
// Throw error if memory with the same key does not exist
CHostFcb* pHostFcb = m_cFcb.Search(nKey);
if (pHostFcb == nullptr)
return FS_NOTOPENED;
// Confirm the existence of the buffer
if (pBuffer == nullptr) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDFUNC;
}
// Read
const uint32_t nResult = pHostFcb->Read(pBuffer, nSize);
if (nResult == (uint32_t)-1) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDFUNC; // TODO: Should return error code 10 (read error) as well here
}
assert(nResult <= nSize);
// Update file pointer
pFcb->fileptr += nResult; /// TODO: Maybe an overflow check is needed here?
return nResult;
}
//---------------------------------------------------------------------------
//
/// $4D - Write file
///
/// Truncate file if 0 bytes are written.
//
//---------------------------------------------------------------------------
int CFileSys::Write(uint32_t nKey, Human68k::fcb_t* pFcb, const uint8_t* pBuffer, uint32_t nSize)
{
assert(nKey);
assert(pFcb);
// assert(pBuffer); // Evaluate only when needed
assert(nSize <= 0xFFFFFF); // Clipped
// Throw error if memory with the same key does not exist
CHostFcb* pHostFcb = m_cFcb.Search(nKey);
if (pHostFcb == nullptr)
return FS_NOTOPENED;
uint32_t nResult;
if (nSize == 0) {
// Truncate
if (!pHostFcb->Truncate()) {
m_cFcb.Free(pHostFcb);
return FS_CANTSEEK;
}
// Update file size
pFcb->size = pFcb->fileptr;
nResult = 0;
} else {
// Confirm the existence of the buffer
if (pBuffer == nullptr) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDFUNC;
}
// Write
nResult = pHostFcb->Write(pBuffer, nSize);
if (nResult == (uint32_t)-1) {
m_cFcb.Free(pHostFcb);
return FS_CANTWRITE; /// TODO: Should return error code 11 (write error) as well here
}
assert(nResult <= nSize);
// Update file pointer
pFcb->fileptr += nResult; /// TODO: Do we need an overflow check here?
// Update file size
if (pFcb->size < pFcb->fileptr)
pFcb->size = pFcb->fileptr;
}
// Update flag
pHostFcb->SetUpdate();
return nResult;
}
//---------------------------------------------------------------------------
//
/// $4E - File seek
//
//---------------------------------------------------------------------------
int CFileSys::Seek(uint32_t nKey, Human68k::fcb_t* pFcb, uint32_t nSeek, int nOffset)
{
assert(pFcb);
// Throw error if memory with the same key does not exist
CHostFcb* pHostFcb = m_cFcb.Search(nKey);
if (pHostFcb == nullptr)
return FS_NOTOPENED;
// Parameter check
if (nSeek > (uint32_t)Human68k::seek_t::SK_END) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDPRM;
}
// File seek
uint32_t nResult = pHostFcb->Seek(nOffset, (Human68k::seek_t)nSeek);
if (nResult == (uint32_t)-1) {
m_cFcb.Free(pHostFcb);
return FS_CANTSEEK;
}
// Update file pointer
pFcb->fileptr = nResult;
return nResult;
}
//---------------------------------------------------------------------------
//
/// $4F - Get/set file time stamp
///
/// Throw error when the top 16 bits are $FFFF.
//
//---------------------------------------------------------------------------
uint32_t CFileSys::TimeStamp(uint32_t nUnit, uint32_t nKey, Human68k::fcb_t* pFcb, uint32_t nHumanTime)
{
assert(nKey);
assert(pFcb);
// Get only
if (nHumanTime == 0)
return ((uint32_t)pFcb->date << 16) | pFcb->time;
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Throw error if memory with the same key does not exist
CHostFcb* pHostFcb = m_cFcb.Search(nKey);
if (pHostFcb == nullptr)
return FS_NOTOPENED;
// Set time stamp
if (!pHostFcb->TimeStamp(nHumanTime)) {
m_cFcb.Free(pHostFcb);
return FS_INVALIDPRM;
}
pFcb->date = (uint16_t)(nHumanTime >> 16);
pFcb->time = (uint16_t)nHumanTime;
// Update cache
m_cEntry.CleanCache(nUnit, pHostFcb->GetHumanPath());
return 0;
}
//---------------------------------------------------------------------------
//
/// $50 - Get capacity
//
//---------------------------------------------------------------------------
int CFileSys::GetCapacity(uint32_t nUnit, Human68k::capacity_t* pCapacity) const
{
assert(pCapacity);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Get capacity
return m_cEntry.GetCapacity(nUnit, pCapacity);
}
//---------------------------------------------------------------------------
//
/// $51 - Inspect/control drive status
//
//---------------------------------------------------------------------------
int CFileSys::CtrlDrive(uint32_t nUnit, Human68k::ctrldrive_t* pCtrlDrive) const
{
assert(pCtrlDrive);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_INVALIDFUNC; // Avoid triggering a fatal error in mint when resuming with an invalid drive
switch (pCtrlDrive->status) {
case 0: // Inspect status
case 9: // Inspect status 2
pCtrlDrive->status = (uint8_t)m_cEntry.GetStatus(nUnit);
return pCtrlDrive->status;
case 1: // Eject
case 2: // Eject forbidden 1 (not implemented)
case 3: // Eject allowed 1 (not implemented)
case 4: // Flash LED when media is not inserted (not implemented)
case 5: // Turn off LED when media is not inserted (not implemented)
case 6: // Eject forbidden 2 (not implemented)
case 7: // Eject allowed 2 (not implemented)
return 0;
case 8: // Eject inspection
return 1;
default:
break;
}
return FS_INVALIDFUNC;
}
//---------------------------------------------------------------------------
//
/// $52 - Get DPB
///
/// If DPB cannot be acquired after resuming, the drive will be torn down internally in Human68k.
/// Therefore, treat even a unit out of bounds as normal operation.
//
//---------------------------------------------------------------------------
int CFileSys::GetDPB(uint32_t nUnit, Human68k::dpb_t* pDpb) const
{
assert(pDpb);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
Human68k::capacity_t cap;
uint8_t media = Human68k::MEDIA_REMOTE;
if (nUnit < m_nUnits) {
media = m_cEntry.GetMediaByte(nUnit);
// Acquire sector data
if (!m_cEntry.GetCapacityCache(nUnit, &cap)) {
// Carry out an extra media check here because it may be skipped when doing a manual eject
if (!m_cEntry.isEnable(nUnit))
goto none;
// Media check
if (m_cEntry.isMediaOffline(nUnit))
goto none;
// Get drive status
m_cEntry.GetCapacity(nUnit, &cap);
}
} else {
none:
cap.clusters = 4; // This is totally fine, right?
cap.sectors = 64;
cap.bytes = 512;
}
// Calculate number of shifts
uint32_t nSize = 1;
uint32_t nShift = 0;
for (;;) {
if (nSize >= cap.sectors)
break;
nSize <<= 1;
nShift++;
}
// Sector number calculation
//
// In the following order:
// Cluster 0: Unused
// Cluster 1: FAT
// Cluster 2: Root directory
// Cluster 3: Data memory (pseudo-sector)
const uint32_t nFat = cap.sectors;
const uint32_t nRoot = cap.sectors * 2;
const uint32_t nData = cap.sectors * 3;
// Set DPB
pDpb->sector_size = cap.bytes; // Bytes per sector
pDpb->cluster_size =
(uint8_t)(cap.sectors - 1); // Sectors per cluster - 1
pDpb->shift = (uint8_t)nShift; // Number of cluster → sector shifts
pDpb->fat_sector = (uint16_t)nFat; // First FAT sector number
pDpb->fat_max = 1; // Number of FAT memory spaces
pDpb->fat_size = (uint8_t)cap.sectors; // Number of sectors controlled by FAT (excluding copies)
pDpb->file_max =
(uint16_t)(cap.sectors * cap.bytes / 0x20); // Number of files in the root directory
pDpb->data_sector = (uint16_t)nData; // First sector number of data memory
pDpb->cluster_max = cap.clusters; // Total number of clusters + 1
pDpb->root_sector = (uint16_t)nRoot; // First sector number of the root directory
pDpb->media = media; // Media byte
return 0;
}
//---------------------------------------------------------------------------
//
/// $53 - Read sector
///
/// We use pseudo-sectors.
/// Buffer size is hard coded to $200 byte.
//
//---------------------------------------------------------------------------
int CFileSys::DiskRead(uint32_t nUnit, uint8_t* pBuffer, uint32_t nSector, uint32_t nSize)
{
assert(pBuffer);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // This occurs when resuming with an invalid drive
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Throw error if number of sectors exceed 1
if (nSize != 1)
return FS_INVALIDPRM;
// Acquire sector data
Human68k::capacity_t cap;
if (!m_cEntry.GetCapacityCache(nUnit, &cap)) {
// Get drive status
m_cEntry.GetCapacity(nUnit, &cap);
}
// Access pseudo-directory entry
const CHostFiles* pHostFiles = m_cFiles.Search(nSector);
if (pHostFiles) {
// Generate pseudo-directory entry
auto dir = (Human68k::dirent_t*)pBuffer;
memcpy(pBuffer, pHostFiles->GetEntry(), sizeof(*dir));
memset(pBuffer + sizeof(*dir), 0xE5, 0x200 - sizeof(*dir));
// Register the pseudo-sector number that points to the file entity inside the pseudo-directory.
// Note that in lzdsys the sector number to read is calculated by the following formula:
// (dirent.cluster - 2) * (dpb.cluster_size + 1) + dpb.data_sector
/// @warning little endian only
dir->cluster = (uint16_t)(m_nHostSectorCount + 2); // Pseudo-sector number
m_nHostSectorBuffer[m_nHostSectorCount] = nSector; // Entity that points to the pseudo-sector
m_nHostSectorCount++;
m_nHostSectorCount %= XM6_HOST_PSEUDO_CLUSTER_MAX;
return 0;
}
// Calculate the sector number from the cluster number
uint32_t n = nSector - (3 * cap.sectors);
uint32_t nMod = 1;
if (cap.sectors) {
// Beware that cap.sectors becomes 0 when media does not exist
nMod = n % cap.sectors;
n /= cap.sectors;
}
// Access the file entity
if (nMod == 0 && n < XM6_HOST_PSEUDO_CLUSTER_MAX) {
pHostFiles = m_cFiles.Search(m_nHostSectorBuffer[n]); // Find entity
if (pHostFiles) {
// Generate pseudo-sector
CHostFcb f;
f.SetFilename(pHostFiles->GetPath());
f.SetMode(Human68k::OP_READ);
if (!f.Open())
return FS_INVALIDPRM;
memset(pBuffer, 0, 0x200);
uint32_t nResult = f.Read(pBuffer, 0x200);
f.Close();
if (nResult == (uint32_t)-1)
return FS_INVALIDPRM;
return 0;
}
}
return FS_INVALIDPRM;
}
//---------------------------------------------------------------------------
//
/// $54 - Write sector
//
//---------------------------------------------------------------------------
int CFileSys::DiskWrite(uint32_t nUnit) const
{
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Write-protect check
if (m_cEntry.isWriteProtect(nUnit))
return FS_FATAL_WRITEPROTECT;
// Thrust at reality
return FS_INVALIDPRM;
}
//---------------------------------------------------------------------------
//
/// $55 - IOCTRL
//
//---------------------------------------------------------------------------
int CFileSys::Ioctrl(uint32_t nUnit, uint32_t nFunction, Human68k::ioctrl_t* pIoctrl)
{
assert(pIoctrl);
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_INVALIDFUNC; // Avoid triggering a fatal error in mint when resuming with an invalid drive
switch (nFunction) {
case 0:
// Acquire media byte
pIoctrl->media = m_cEntry.GetMediaByte(nUnit);
return 0;
case 1:
// Dummy for Human68k compatibility
pIoctrl->param = -1;
return 0;
case 2:
switch (pIoctrl->param) {
case (uint32_t)-1:
// Re-identify media
m_cEntry.isMediaOffline(nUnit);
return 0;
case 0:
case 1:
// Dummy for Human68k compatibility
return 0;
default:
return FS_NOTIOCTRL;
}
break;
case (uint32_t)-1:
// Resident evaluation
memcpy(pIoctrl->buffer, "WindrvXM", 8);
return 0;
case (uint32_t)-2:
// Set options
SetOption(pIoctrl->param);
return 0;
case (uint32_t)-3:
// Get options
pIoctrl->param = GetOption();
return 0;
default:
break;
}
return FS_NOTIOCTRL;
}
//---------------------------------------------------------------------------
//
/// $56 - Flush
//
//---------------------------------------------------------------------------
int CFileSys::Flush(uint32_t nUnit) const
{
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_INVALIDFUNC; // Avoid triggering a fatal error returning from a mint command when resuming with an invalid drive
// Always succeed
return 0;
}
//---------------------------------------------------------------------------
//
/// $57 - Media change check
//
//---------------------------------------------------------------------------
int CFileSys::CheckMedia(uint32_t nUnit) const
{
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
if (nUnit >= m_nUnits)
return FS_INVALIDFUNC; // Avoid triggering a fatal error in mint when resuming with an invalid drive
// Media change check
// Throw error when media is not inserted
if (!m_cEntry.CheckMedia(nUnit)) {
return FS_INVALIDFUNC;
}
return 0;
}
//---------------------------------------------------------------------------
//
/// $58 - Lock
//
//---------------------------------------------------------------------------
int CFileSys::Lock(uint32_t nUnit) const
{
// Unit check
if (nUnit >= CHostEntry::DRIVE_MAX)
return FS_FATAL_INVALIDUNIT;
assert(nUnit < m_nUnits);
if (nUnit >= m_nUnits)
return FS_FATAL_MEDIAOFFLINE; // Just in case
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return FS_FATAL_MEDIAOFFLINE;
// Always succeed
return 0;
}
//---------------------------------------------------------------------------
//
/// Set options
//
//---------------------------------------------------------------------------
void CFileSys::SetOption(uint32_t nOption)
{
// Clear cache when option settings change
if (m_nOption ^ nOption)
m_cEntry.CleanCache();
m_nOption = nOption;
g_nOption = nOption;
}
//---------------------------------------------------------------------------
//
/// Initialize options
//
//---------------------------------------------------------------------------
void CFileSys::InitOption(const Human68k::argument_t* pArgument)
{
assert(pArgument);
// Initialize number of drives
m_nDrives = 0;
const uint8_t* pp = pArgument->buf;
pp += strlen((const char*)pp) + 1;
uint32_t nOption = m_nOptionDefault;
for (;;) {
assert(pp < pArgument->buf + sizeof(*pArgument));
const uint8_t* p = pp;
uint8_t c = *p++;
if (c == '\0')
break;
uint32_t nMode;
if (c == '+') {
nMode = 1;
} else if (c == '-') {
nMode = 0;
} else if (c == '/') {
// Specify default base path
if (m_nDrives < CHostEntry::DRIVE_MAX) {
p--;
strcpy(m_szBase[m_nDrives], (const char *)p);
m_nDrives++;
}
pp += strlen((const char*)pp) + 1;
continue;
} else {
// Continue since no option is specified
pp += strlen((const char*)pp) + 1;
continue;
}
for (;;) {
c = *p++;
if (c == '\0')
break;
uint32_t nBit = 0;
switch (c) {
case 'A': case 'a': nBit = WINDRV_OPT_CONVERT_LENGTH; break;
case 'T': case 't': nBit = WINDRV_OPT_COMPARE_LENGTH; nMode ^= 1; break;
case 'C': case 'c': nBit = WINDRV_OPT_ALPHABET; break;
case 'E': nBit = WINDRV_OPT_CONVERT_PERIOD; break;
case 'P': nBit = WINDRV_OPT_CONVERT_PERIODS; break;
case 'N': nBit = WINDRV_OPT_CONVERT_HYPHEN; break;
case 'H': nBit = WINDRV_OPT_CONVERT_HYPHENS; break;
case 'X': nBit = WINDRV_OPT_CONVERT_BADCHAR; break;
case 'S': nBit = WINDRV_OPT_CONVERT_SPACE; break;
case 'e': nBit = WINDRV_OPT_REDUCED_PERIOD; break;
case 'p': nBit = WINDRV_OPT_REDUCED_PERIODS; break;
case 'n': nBit = WINDRV_OPT_REDUCED_HYPHEN; break;
case 'h': nBit = WINDRV_OPT_REDUCED_HYPHENS; break;
case 'x': nBit = WINDRV_OPT_REDUCED_BADCHAR; break;
case 's': nBit = WINDRV_OPT_REDUCED_SPACE; break;
default: break;
}
if (nMode)
nOption |= nBit;
else
nOption &= ~nBit;
}
pp = p;
}
// Set options
if (nOption != m_nOption) {
SetOption(nOption);
}
}
//---------------------------------------------------------------------------
//
/// Get volume label
//
//---------------------------------------------------------------------------
bool CFileSys::FilesVolume(uint32_t nUnit, Human68k::files_t* pFiles) const
{
assert(pFiles);
// Get volume label
TCHAR szVolume[32];
if (bool bResult = m_cEntry.GetVolumeCache(nUnit, szVolume); !bResult) {
// Carry out an extra media check here because it may be skipped when doing a manual eject
if (!m_cEntry.isEnable(nUnit))
return false;
// Media check
if (m_cEntry.isMediaOffline(nUnit))
return false;
// Get volume label
m_cEntry.GetVolume(nUnit, szVolume);
}
if (szVolume[0] == '\0')
return false;
pFiles->attr = Human68k::AT_VOLUME;
pFiles->time = 0;
pFiles->date = 0;
pFiles->size = 0;
CHostFilename fname;
fname.SetHost(szVolume);
fname.ConvertHuman();
strcpy((char*)pFiles->full, (const char*)fname.GetHuman());
return true;
}