mirror of https://github.com/PrimeDecomp/prime.git
Add CDvdFile, match and link dvdfs
This commit is contained in:
parent
b2a9c6a8f3
commit
112a7adec8
|
@ -654,7 +654,7 @@ LIBS = [
|
||||||
"Kyoto/DolphinCMemoryCardSys",
|
"Kyoto/DolphinCMemoryCardSys",
|
||||||
["Kyoto/Input/DolphinIController", True],
|
["Kyoto/Input/DolphinIController", True],
|
||||||
["Kyoto/Input/CDolphinController", True],
|
["Kyoto/Input/CDolphinController", True],
|
||||||
"Kyoto/DolphinCDvdFile",
|
["Kyoto/DolphinCDvdFile", False],
|
||||||
"Kyoto/Alloc/CMediumAllocPool",
|
"Kyoto/Alloc/CMediumAllocPool",
|
||||||
["Kyoto/Alloc/CSmallAllocPool", True],
|
["Kyoto/Alloc/CSmallAllocPool", True],
|
||||||
["Kyoto/Alloc/CGameAllocator", False],
|
["Kyoto/Alloc/CGameAllocator", False],
|
||||||
|
@ -752,7 +752,7 @@ LIBS = [
|
||||||
"host": False,
|
"host": False,
|
||||||
"objects": [
|
"objects": [
|
||||||
"Dolphin/dvd/dvdlow",
|
"Dolphin/dvd/dvdlow",
|
||||||
"Dolphin/dvd/dvdfs",
|
["Dolphin/dvd/dvdfs", True],
|
||||||
["Dolphin/dvd/dvd", False],
|
["Dolphin/dvd/dvd", False],
|
||||||
"Dolphin/dvd/dvdqueue",
|
"Dolphin/dvd/dvdqueue",
|
||||||
"Dolphin/dvd/dvderror",
|
"Dolphin/dvd/dvderror",
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
|
|
||||||
#include "Kyoto/IDvdRequest.hpp"
|
#include "Kyoto/IDvdRequest.hpp"
|
||||||
|
|
||||||
|
struct DVDFileInfo;
|
||||||
class CDvdFile {
|
class CDvdFile {
|
||||||
public:
|
public:
|
||||||
CDvdFile(const char* name);
|
CDvdFile(const char* name);
|
||||||
|
@ -15,6 +16,9 @@ public:
|
||||||
|
|
||||||
static bool FileExists(const char*);
|
static bool FileExists(const char*);
|
||||||
|
|
||||||
|
static void DVDARAMXferCallback(long, DVDFileInfo*);
|
||||||
|
void HandleDVDInterrupt();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uchar pad[0x14];
|
uchar pad[0x14];
|
||||||
uint x14_size;
|
uint x14_size;
|
||||||
|
|
|
@ -7,6 +7,32 @@
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define DVD_MIN_TRANSFER_SIZE 32
|
||||||
|
|
||||||
|
#define DVD_STATE_FATAL_ERROR -1
|
||||||
|
#define DVD_STATE_END 0
|
||||||
|
#define DVD_STATE_BUSY 1
|
||||||
|
#define DVD_STATE_WAITING 2
|
||||||
|
#define DVD_STATE_COVER_CLOSED 3
|
||||||
|
#define DVD_STATE_NO_DISK 4
|
||||||
|
#define DVD_STATE_COVER_OPEN 5
|
||||||
|
#define DVD_STATE_WRONG_DISK 6
|
||||||
|
#define DVD_STATE_MOTOR_STOPPED 7
|
||||||
|
#define DVD_STATE_PAUSING 8
|
||||||
|
#define DVD_STATE_IGNORED 9
|
||||||
|
#define DVD_STATE_CANCELED 10
|
||||||
|
#define DVD_STATE_RETRY 11
|
||||||
|
|
||||||
|
#define DVD_FILEINFO_READY 0
|
||||||
|
#define DVD_FILEINFO_BUSY 1
|
||||||
|
|
||||||
|
#define DVD_RESULT_GOOD 0
|
||||||
|
#define DVD_RESULT_FATAL_ERROR -1
|
||||||
|
#define DVD_RESULT_IGNORED -2
|
||||||
|
#define DVD_RESULT_CANCELED -3
|
||||||
|
|
||||||
|
#define DVD_AIS_SUCCESS 0x0
|
||||||
|
|
||||||
typedef struct DVDDiskID {
|
typedef struct DVDDiskID {
|
||||||
char gameName[4];
|
char gameName[4];
|
||||||
char company[2];
|
char company[2];
|
||||||
|
@ -41,15 +67,29 @@ typedef struct DVDFileInfo DVDFileInfo;
|
||||||
typedef void (*DVDCallback)(s32 result, DVDFileInfo* fileInfo);
|
typedef void (*DVDCallback)(s32 result, DVDFileInfo* fileInfo);
|
||||||
|
|
||||||
struct DVDFileInfo {
|
struct DVDFileInfo {
|
||||||
|
DVDCommandBlock cb;
|
||||||
u32 startAddr;
|
u32 startAddr;
|
||||||
u32 length;
|
u32 length;
|
||||||
DVDCallback callback;
|
DVDCallback callback;
|
||||||
};
|
};
|
||||||
|
|
||||||
void DVDInit();
|
typedef struct {
|
||||||
|
u32 entryNum;
|
||||||
|
u32 location;
|
||||||
|
u32 next;
|
||||||
|
} DVDDir;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u32 entryNum;
|
||||||
|
BOOL isDir;
|
||||||
|
char* name;
|
||||||
|
} DVDDirEntry;
|
||||||
|
|
||||||
|
void DVDInit();
|
||||||
|
BOOL DVDClose(DVDFileInfo* f);
|
||||||
void DVDSetAutoFatalMessaging(BOOL);
|
void DVDSetAutoFatalMessaging(BOOL);
|
||||||
void DVDReset();
|
void DVDReset();
|
||||||
|
s32 DVDCancel(DVDCommandBlock* block);
|
||||||
|
|
||||||
BOOL DVDPrepareStreamAsync(DVDFileInfo* fInfo, u32 length, u32 offset, DVDCallback callback);
|
BOOL DVDPrepareStreamAsync(DVDFileInfo* fInfo, u32 length, u32 offset, DVDCallback callback);
|
||||||
s32 DVDPrepareStream(DVDFileInfo* fInfo, u32 length, u32 offset);
|
s32 DVDPrepareStream(DVDFileInfo* fInfo, u32 length, u32 offset);
|
||||||
|
|
|
@ -0,0 +1,656 @@
|
||||||
|
#include "dolphin/DVDPriv.h"
|
||||||
|
#include "dolphin/os.h"
|
||||||
|
#include "dolphin/os/OSBootInfo.h"
|
||||||
|
|
||||||
|
typedef struct FSTEntry FSTEntry;
|
||||||
|
|
||||||
|
struct FSTEntry {
|
||||||
|
unsigned int isDirAndStringOff;
|
||||||
|
unsigned int parentOrPosition;
|
||||||
|
unsigned int nextEntryOrLength;
|
||||||
|
};
|
||||||
|
|
||||||
|
static OSBootInfo* BootInfo;
|
||||||
|
static FSTEntry* FstStart;
|
||||||
|
static char* FstStringStart;
|
||||||
|
static u32 MaxEntryNum;
|
||||||
|
static u32 currentDirectory = 0;
|
||||||
|
OSThreadQueue __DVDThreadQueue;
|
||||||
|
u32 __DVDLongFileNameFlag = 0;
|
||||||
|
|
||||||
|
static void cbForReadAsync(s32 result, DVDCommandBlock* block);
|
||||||
|
static void cbForReadSync(s32 result, DVDCommandBlock* block);
|
||||||
|
static void cbForSeekAsync(s32 result, DVDCommandBlock* block);
|
||||||
|
static void cbForSeekSync(s32 result, DVDCommandBlock* block);
|
||||||
|
static void cbForPrepareStreamAsync(s32 result, DVDCommandBlock* block);
|
||||||
|
static void cbForPrepareStreamSync(s32 result, DVDCommandBlock* block);
|
||||||
|
|
||||||
|
void __DVDFSInit() {
|
||||||
|
BootInfo = (OSBootInfo*)OSPhysicalToCached(0);
|
||||||
|
FstStart = (FSTEntry*)BootInfo->FSTLocation;
|
||||||
|
|
||||||
|
if (FstStart) {
|
||||||
|
MaxEntryNum = FstStart[0].nextEntryOrLength;
|
||||||
|
FstStringStart = (char*)&(FstStart[MaxEntryNum]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* For convenience */
|
||||||
|
#define entryIsDir(i) (((FstStart[i].isDirAndStringOff & 0xff000000) == 0) ? FALSE : TRUE)
|
||||||
|
#define stringOff(i) (FstStart[i].isDirAndStringOff & ~0xff000000)
|
||||||
|
#define parentDir(i) (FstStart[i].parentOrPosition)
|
||||||
|
#define nextDir(i) (FstStart[i].nextEntryOrLength)
|
||||||
|
#define filePosition(i) (FstStart[i].parentOrPosition)
|
||||||
|
#define fileLength(i) (FstStart[i].nextEntryOrLength)
|
||||||
|
|
||||||
|
static BOOL isSame(const char* path, const char* string) {
|
||||||
|
while (*string != '\0') {
|
||||||
|
if (tolower(*path++) != tolower(*string++)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((*path == '/') || (*path == '\0')) {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 DVDConvertPathToEntrynum(const char* pathPtr) {
|
||||||
|
const char* ptr;
|
||||||
|
char* stringPtr;
|
||||||
|
BOOL isDir;
|
||||||
|
u32 length;
|
||||||
|
u32 dirLookAt;
|
||||||
|
u32 i;
|
||||||
|
const char* origPathPtr = pathPtr;
|
||||||
|
const char* extentionStart;
|
||||||
|
BOOL illegal;
|
||||||
|
BOOL extention;
|
||||||
|
|
||||||
|
dirLookAt = currentDirectory;
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
|
||||||
|
if (*pathPtr == '\0') {
|
||||||
|
return (s32)dirLookAt;
|
||||||
|
} else if (*pathPtr == '/') {
|
||||||
|
dirLookAt = 0;
|
||||||
|
pathPtr++;
|
||||||
|
continue;
|
||||||
|
} else if (*pathPtr == '.') {
|
||||||
|
if (*(pathPtr + 1) == '.') {
|
||||||
|
if (*(pathPtr + 2) == '/') {
|
||||||
|
dirLookAt = parentDir(dirLookAt);
|
||||||
|
pathPtr += 3;
|
||||||
|
continue;
|
||||||
|
} else if (*(pathPtr + 2) == '\0') {
|
||||||
|
return (s32)parentDir(dirLookAt);
|
||||||
|
}
|
||||||
|
} else if (*(pathPtr + 1) == '/') {
|
||||||
|
pathPtr += 2;
|
||||||
|
continue;
|
||||||
|
} else if (*(pathPtr + 1) == '\0') {
|
||||||
|
return (s32)dirLookAt;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (__DVDLongFileNameFlag == 0) {
|
||||||
|
extention = FALSE;
|
||||||
|
illegal = FALSE;
|
||||||
|
|
||||||
|
for (ptr = pathPtr; (*ptr != '\0') && (*ptr != '/'); ptr++) {
|
||||||
|
if (*ptr == '.') {
|
||||||
|
if ((ptr - pathPtr > 8) || (extention == TRUE)) {
|
||||||
|
illegal = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
extention = TRUE;
|
||||||
|
extentionStart = ptr + 1;
|
||||||
|
|
||||||
|
} else if (*ptr == ' ')
|
||||||
|
illegal = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((extention == TRUE) && (ptr - extentionStart > 3))
|
||||||
|
illegal = TRUE;
|
||||||
|
|
||||||
|
if (illegal)
|
||||||
|
OSPanic(__FILE__, 379,
|
||||||
|
"DVDConvertEntrynumToPath(possibly DVDOpen or DVDChangeDir or DVDOpenDir): "
|
||||||
|
"specified directory or file (%s) doesn't match standard 8.3 format. This is a "
|
||||||
|
"temporary restriction and will be removed soon\n",
|
||||||
|
origPathPtr);
|
||||||
|
} else {
|
||||||
|
for (ptr = pathPtr; (*ptr != '\0') && (*ptr != '/'); ptr++)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
isDir = (*ptr == '\0') ? FALSE : TRUE;
|
||||||
|
length = (u32)(ptr - pathPtr);
|
||||||
|
|
||||||
|
ptr = pathPtr;
|
||||||
|
|
||||||
|
for (i = dirLookAt + 1; i < nextDir(dirLookAt); i = entryIsDir(i) ? nextDir(i) : (i + 1)) {
|
||||||
|
if ((entryIsDir(i) == FALSE) && (isDir == TRUE)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
stringPtr = FstStringStart + stringOff(i);
|
||||||
|
|
||||||
|
if (isSame(ptr, stringPtr) == TRUE) {
|
||||||
|
goto next_hier;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
next_hier:
|
||||||
|
if (!isDir) {
|
||||||
|
return (s32)i;
|
||||||
|
}
|
||||||
|
|
||||||
|
dirLookAt = i;
|
||||||
|
pathPtr += length + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDFastOpen(s32 entrynum, DVDFileInfo* fileInfo) {
|
||||||
|
if ((entrynum < 0) || (entrynum >= MaxEntryNum) || entryIsDir(entrynum)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo->startAddr = filePosition(entrynum);
|
||||||
|
fileInfo->length = fileLength(entrynum);
|
||||||
|
fileInfo->callback = (DVDCallback)NULL;
|
||||||
|
fileInfo->cb.state = DVD_STATE_END;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDOpen(const char* fileName, DVDFileInfo* fileInfo) {
|
||||||
|
s32 entry;
|
||||||
|
char currentDir[128];
|
||||||
|
|
||||||
|
entry = DVDConvertPathToEntrynum(fileName);
|
||||||
|
|
||||||
|
if (0 > entry) {
|
||||||
|
DVDGetCurrentDir(currentDir, 128);
|
||||||
|
OSReport("Warning: DVDOpen(): file '%s' was not found under %s.\n", fileName, currentDir);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entryIsDir(entry)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo->startAddr = filePosition(entry);
|
||||||
|
fileInfo->length = fileLength(entry);
|
||||||
|
fileInfo->callback = (DVDCallback)NULL;
|
||||||
|
fileInfo->cb.state = DVD_STATE_END;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if NONMATCHING
|
||||||
|
BOOL DVDClose(DVDFileInfo* fileInfo) {
|
||||||
|
DVDCancel(&(fileInfo->cb));
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
/* clang-format off */
|
||||||
|
#pragma push
|
||||||
|
#pragma optimization_level 0
|
||||||
|
#pragma optimizewithasm off
|
||||||
|
asm BOOL DVDClose(DVDFileInfo* fileInfo) {
|
||||||
|
nofralloc
|
||||||
|
mflr r0
|
||||||
|
stw r0, 4(r1)
|
||||||
|
stwu r1, -8(r1)
|
||||||
|
bl DVDCancel
|
||||||
|
li r3, 1
|
||||||
|
lwz r0, 0xc(r1)
|
||||||
|
addi r1, r1, 8
|
||||||
|
mtlr r0
|
||||||
|
blr
|
||||||
|
}
|
||||||
|
#pragma pop
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static u32 myStrncpy(char* dest, char* src, u32 maxlen) {
|
||||||
|
u32 i = maxlen;
|
||||||
|
|
||||||
|
while ((i > 0) && (*src != 0)) {
|
||||||
|
*dest++ = *src++;
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (maxlen - i);
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 entryToPath(u32 entry, char* path, u32 maxlen) {
|
||||||
|
char* name;
|
||||||
|
u32 loc;
|
||||||
|
|
||||||
|
if (entry == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
name = FstStringStart + stringOff(entry);
|
||||||
|
|
||||||
|
loc = entryToPath(parentDir(entry), path, maxlen);
|
||||||
|
|
||||||
|
if (loc == maxlen) {
|
||||||
|
return loc;
|
||||||
|
}
|
||||||
|
|
||||||
|
*(path + loc++) = '/';
|
||||||
|
|
||||||
|
loc += myStrncpy(path + loc, name, maxlen - loc);
|
||||||
|
|
||||||
|
return loc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static BOOL DVDConvertEntrynumToPath(s32 entrynum, char* path, u32 maxlen) {
|
||||||
|
u32 loc;
|
||||||
|
|
||||||
|
loc = entryToPath((u32)entrynum, path, maxlen);
|
||||||
|
|
||||||
|
if (loc == maxlen) {
|
||||||
|
path[maxlen - 1] = '\0';
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entryIsDir(entrynum)) {
|
||||||
|
if (loc == maxlen - 1) {
|
||||||
|
path[loc] = '\0';
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
path[loc++] = '/';
|
||||||
|
}
|
||||||
|
|
||||||
|
path[loc] = '\0';
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDGetCurrentDir(char* path, u32 maxlen) {
|
||||||
|
return DVDConvertEntrynumToPath((s32)currentDirectory, path, maxlen);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDChangeDir(const char* dirName) {
|
||||||
|
s32 entry;
|
||||||
|
entry = DVDConvertPathToEntrynum(dirName);
|
||||||
|
if ((entry < 0) || (entryIsDir(entry) == FALSE)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
currentDirectory = (u32)entry;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDReadAsyncPrio(DVDFileInfo* fileInfo, void* addr, s32 length, s32 offset,
|
||||||
|
DVDCallback callback, s32 prio) {
|
||||||
|
|
||||||
|
if (!((0 <= offset) && (offset < fileInfo->length))) {
|
||||||
|
OSPanic(__FILE__, 742, "DVDReadAsync(): specified area is out of the file ");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!((0 <= offset + length) && (offset + length < fileInfo->length + DVD_MIN_TRANSFER_SIZE))) {
|
||||||
|
OSPanic(__FILE__, 748, "DVDReadAsync(): specified area is out of the file ");
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo->callback = callback;
|
||||||
|
DVDReadAbsAsyncPrio(&(fileInfo->cb), addr, length, (s32)(fileInfo->startAddr + offset),
|
||||||
|
cbForReadAsync, prio);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
#ifndef offsetof
|
||||||
|
#define offsetof(type, memb) ((u32) & ((type*)0)->memb)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void cbForReadAsync(s32 result, DVDCommandBlock* block) {
|
||||||
|
DVDFileInfo* fileInfo;
|
||||||
|
|
||||||
|
fileInfo = (DVDFileInfo*)((char*)block - offsetof(DVDFileInfo, cb));
|
||||||
|
if (fileInfo->callback) {
|
||||||
|
(fileInfo->callback)(result, fileInfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases I have also left the line numbers at 0 */
|
||||||
|
s32 DVDReadPrio(DVDFileInfo* fileInfo, void* addr, s32 length, s32 offset, s32 prio) {
|
||||||
|
BOOL result;
|
||||||
|
DVDCommandBlock* block;
|
||||||
|
s32 state;
|
||||||
|
BOOL enabled;
|
||||||
|
s32 retVal;
|
||||||
|
|
||||||
|
if (!((0 <= offset) && (offset <= fileInfo->length))) {
|
||||||
|
OSPanic(__FILE__, 0, "DVDRead(): specified area is out of the file ");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!((0 <= offset + length) && (offset + length < fileInfo->length + DVD_MIN_TRANSFER_SIZE))) {
|
||||||
|
OSPanic(__FILE__, 0, "DVDRead(): specified area is out of the file ");
|
||||||
|
}
|
||||||
|
|
||||||
|
block = &(fileInfo->cb);
|
||||||
|
|
||||||
|
result = DVDReadAbsAsyncPrio(block, addr, length, (s32)(fileInfo->startAddr + offset),
|
||||||
|
cbForReadSync, prio);
|
||||||
|
|
||||||
|
if (result == FALSE) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
enabled = OSDisableInterrupts();
|
||||||
|
|
||||||
|
while(1) {
|
||||||
|
state = ((volatile DVDCommandBlock*)block)->state;
|
||||||
|
|
||||||
|
if (state == DVD_STATE_END) {
|
||||||
|
retVal = (s32)block->transferredSize;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_FATAL_ERROR) {
|
||||||
|
retVal = DVD_RESULT_FATAL_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_CANCELED) {
|
||||||
|
retVal = DVD_RESULT_CANCELED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
OSSleepThread(&__DVDThreadQueue);
|
||||||
|
}
|
||||||
|
|
||||||
|
OSRestoreInterrupts(enabled);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
static void cbForReadSync(s32 result, DVDCommandBlock* block) { OSWakeupThread(&__DVDThreadQueue); }
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
BOOL DVDSeekAsyncPrio(DVDFileInfo* fileInfo, s32 offset, DVDCallback callback, s32 prio) {
|
||||||
|
if (!((0 <= offset) && (offset <= fileInfo->length))) {
|
||||||
|
OSPanic(__FILE__, 0, "DVDSeek(): offset is out of the file ");
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo->callback = callback;
|
||||||
|
DVDSeekAbsAsyncPrio(&(fileInfo->cb), (s32)(fileInfo->startAddr + offset), cbForSeekAsync,
|
||||||
|
prio);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
static void cbForSeekAsync(s32 result, DVDCommandBlock* block) {
|
||||||
|
DVDFileInfo* fileInfo;
|
||||||
|
|
||||||
|
fileInfo = (DVDFileInfo*)((char*)block - offsetof(DVDFileInfo, cb));
|
||||||
|
|
||||||
|
if (fileInfo->callback) {
|
||||||
|
(fileInfo->callback)(result, fileInfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
s32 DVDSeekPrio(DVDFileInfo* fileInfo, s32 offset, s32 prio) {
|
||||||
|
BOOL result;
|
||||||
|
DVDCommandBlock* block;
|
||||||
|
s32 state;
|
||||||
|
BOOL enabled;
|
||||||
|
s32 retVal;
|
||||||
|
|
||||||
|
block = &(fileInfo->cb);
|
||||||
|
|
||||||
|
result =
|
||||||
|
DVDSeekAbsAsyncPrio(block, (s32)(fileInfo->startAddr + offset), cbForSeekSync, prio);
|
||||||
|
|
||||||
|
if (result == FALSE) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
enabled = OSDisableInterrupts();
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
state = ((volatile DVDCommandBlock*)block)->state;
|
||||||
|
|
||||||
|
if (state == DVD_STATE_END) {
|
||||||
|
retVal = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_FATAL_ERROR) {
|
||||||
|
retVal = DVD_RESULT_FATAL_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_CANCELED) {
|
||||||
|
retVal = DVD_RESULT_CANCELED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
OSSleepThread(&__DVDThreadQueue);
|
||||||
|
}
|
||||||
|
|
||||||
|
OSRestoreInterrupts(enabled);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
static void cbForSeekSync(s32 result, DVDCommandBlock* block) { OSWakeupThread(&__DVDThreadQueue); }
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
s32 DVDGetFileInfoStatus(const DVDFileInfo* fileInfo) {
|
||||||
|
return DVDGetCommandBlockStatus(&fileInfo->cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
BOOL DVDFastOpenDir(s32 entrynum, DVDDir* dir) {
|
||||||
|
|
||||||
|
if ((entrynum < 0) || (entrynum >= MaxEntryNum) || !entryIsDir(entrynum)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
dir->entryNum = (u32)entrynum;
|
||||||
|
dir->location = (u32)entrynum + 1;
|
||||||
|
dir->next = nextDir(entrynum);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
BOOL DVDOpenDir(const char* dirName, DVDDir* dir) {
|
||||||
|
s32 entry;
|
||||||
|
char currentDir[128];
|
||||||
|
entry = DVDConvertPathToEntrynum(dirName);
|
||||||
|
|
||||||
|
if (entry < 0) {
|
||||||
|
DVDGetCurrentDir(currentDir, 128);
|
||||||
|
OSReport("Warning: DVDOpenDir(): file '%s' was not found under %s.\n", dirName, currentDir);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!entryIsDir(entry)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
dir->entryNum = (u32)entry;
|
||||||
|
dir->location = (u32)entry + 1;
|
||||||
|
dir->next = nextDir(entry);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL DVDReadDir(DVDDir* dir, DVDDirEntry* dirent) {
|
||||||
|
u32 loc = dir->location;
|
||||||
|
if ((loc <= dir->entryNum) || (dir->next <= loc))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
dirent->entryNum = loc;
|
||||||
|
dirent->isDir = entryIsDir(loc);
|
||||||
|
dirent->name = FstStringStart + stringOff(loc);
|
||||||
|
|
||||||
|
dir->location = entryIsDir(loc) ? nextDir(loc) : (loc + 1);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
BOOL DVDCloseDir(DVDDir* dir) { return TRUE; }
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
void DVDRewindDir(DVDDir* dir) { dir->location = dir->entryNum + 1; }
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
void* DVDGetFSTLocation(void) { return BootInfo->FSTLocation; }
|
||||||
|
|
||||||
|
#define RoundUp32KB(x) (((u32)(x) + 32 * 1024 - 1) & ~(32 * 1024 - 1))
|
||||||
|
#define Is32KBAligned(x) (((u32)(x) & (32 * 1024 - 1)) == 0)
|
||||||
|
|
||||||
|
BOOL DVDPrepareStreamAsync(DVDFileInfo* fileInfo, u32 length, u32 offset, DVDCallback callback) {
|
||||||
|
u32 start;
|
||||||
|
|
||||||
|
start = fileInfo->startAddr + offset;
|
||||||
|
|
||||||
|
if (!Is32KBAligned(start)) {
|
||||||
|
OSPanic(__FILE__, 1189,
|
||||||
|
"DVDPrepareStreamAsync(): Specified start address (filestart(0x%x) + offset(0x%x)) is "
|
||||||
|
"not 32KB aligned",
|
||||||
|
fileInfo->startAddr, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length == 0)
|
||||||
|
length = fileInfo->length - offset;
|
||||||
|
|
||||||
|
if (!Is32KBAligned(length)) {
|
||||||
|
OSPanic(__FILE__, 1199,
|
||||||
|
"DVDPrepareStreamAsync(): Specified length (0x%x) is not a multiple of 32768(32*1024)",
|
||||||
|
length);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!((offset < fileInfo->length) && (offset + length <= fileInfo->length))) {
|
||||||
|
OSPanic(__FILE__, 1207,
|
||||||
|
"DVDPrepareStreamAsync(): The area specified (offset(0x%x), length(0x%x)) is out of "
|
||||||
|
"the file",
|
||||||
|
offset, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo->callback = callback;
|
||||||
|
return DVDPrepareStreamAbsAsync(&(fileInfo->cb), length, fileInfo->startAddr + offset,
|
||||||
|
cbForPrepareStreamAsync);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cbForPrepareStreamAsync(s32 result, DVDCommandBlock* block) {
|
||||||
|
DVDFileInfo* fileInfo;
|
||||||
|
|
||||||
|
fileInfo = (DVDFileInfo*)((char*)block - offsetof(DVDFileInfo, cb));
|
||||||
|
|
||||||
|
if (fileInfo->callback) {
|
||||||
|
(fileInfo->callback)(result, fileInfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
s32 DVDPrepareStream(DVDFileInfo* fileInfo, u32 length, u32 offset) {
|
||||||
|
BOOL result;
|
||||||
|
DVDCommandBlock* block;
|
||||||
|
s32 state;
|
||||||
|
BOOL enabled;
|
||||||
|
s32 retVal;
|
||||||
|
u32 start;
|
||||||
|
start = fileInfo->startAddr + offset;
|
||||||
|
|
||||||
|
if (!Is32KBAligned(start)) {
|
||||||
|
OSPanic(__FILE__, 0,
|
||||||
|
"DVDPrepareStream(): Specified start address (filestart(0x%x) + offset(0x%x)) is not "
|
||||||
|
"32KB aligned",
|
||||||
|
fileInfo->startAddr, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length == 0)
|
||||||
|
length = fileInfo->length - offset;
|
||||||
|
|
||||||
|
if (!Is32KBAligned(length)) {
|
||||||
|
OSPanic(__FILE__, 0,
|
||||||
|
"DVDPrepareStream(): Specified length (0x%x) is not a multiple of 32768(32*1024)",
|
||||||
|
length);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!((offset <= fileInfo->length) && (offset + length <= fileInfo->length))) {
|
||||||
|
OSPanic(
|
||||||
|
__FILE__, 0,
|
||||||
|
"DVDPrepareStream(): The area specified (offset(0x%x), length(0x%x)) is out of the file",
|
||||||
|
offset, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
block = &(fileInfo->cb);
|
||||||
|
result = DVDPrepareStreamAbsAsync(block, length, start, cbForPrepareStreamSync);
|
||||||
|
|
||||||
|
if (result == FALSE) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
enabled = OSDisableInterrupts();
|
||||||
|
|
||||||
|
while(1) {
|
||||||
|
state = ((volatile DVDCommandBlock*)block)->state;
|
||||||
|
|
||||||
|
if (state == DVD_STATE_END) {
|
||||||
|
retVal = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_FATAL_ERROR) {
|
||||||
|
retVal = DVD_RESULT_FATAL_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (state == DVD_STATE_CANCELED) {
|
||||||
|
retVal = DVD_RESULT_CANCELED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
OSSleepThread(&__DVDThreadQueue);
|
||||||
|
}
|
||||||
|
|
||||||
|
OSRestoreInterrupts(enabled);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
static void cbForPrepareStreamSync(s32 result, DVDCommandBlock* block) {
|
||||||
|
OSWakeupThread(&__DVDThreadQueue);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is based on the revolution SDK, these may not match in all cases */
|
||||||
|
s32 DVDGetTransferredSize(DVDFileInfo* fileinfo) {
|
||||||
|
s32 bytes;
|
||||||
|
DVDCommandBlock* cb;
|
||||||
|
|
||||||
|
cb = &(fileinfo->cb);
|
||||||
|
|
||||||
|
switch (cb->state) {
|
||||||
|
case DVD_STATE_END:
|
||||||
|
case DVD_STATE_COVER_CLOSED:
|
||||||
|
case DVD_STATE_NO_DISK:
|
||||||
|
case DVD_STATE_COVER_OPEN:
|
||||||
|
case DVD_STATE_WRONG_DISK:
|
||||||
|
case DVD_STATE_FATAL_ERROR:
|
||||||
|
case DVD_STATE_MOTOR_STOPPED:
|
||||||
|
case DVD_STATE_CANCELED:
|
||||||
|
case DVD_STATE_RETRY:
|
||||||
|
bytes = (s32)cb->transferredSize;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DVD_STATE_WAITING:
|
||||||
|
bytes = 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DVD_STATE_BUSY:
|
||||||
|
bytes = (s32)(cb->transferredSize + (cb->currTransferSize - DVDLowGetLength()));
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return bytes;
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
#include "Kyoto/CDvdFile.hpp"
|
||||||
|
#include "dolphin/dvd.h"
|
||||||
|
|
||||||
|
#include "string.h"
|
||||||
|
|
||||||
|
const char* DecodeARAMFile(const char* filename) {
|
||||||
|
if (!strncmp(filename, "aram:", 5)) {
|
||||||
|
return filename + 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
return filename;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CDvdFile::DVDARAMXferCallback(s32 result, DVDFileInfo* info) {
|
||||||
|
DVDClose(info);
|
||||||
|
reinterpret_cast<CDvdFile*>(info->cb.userData)->HandleDVDInterrupt();
|
||||||
|
}
|
Loading…
Reference in New Issue