/* ScummVM - Scumm Interpreter * Copyright (C) 2002-2006 The ScummVM project * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * $URL$ * $Id$ */ #if defined(__MORPHOS__) #include #include #include "base/engine.h" #include "backends/fs/abstract-fs.h" /* * Implementation of the ScummVM file system API based on the MorphOS A-Box API. */ class ABoxFilesystemNode : public AbstractFilesystemNode { protected: BPTR _lock; String _displayName; bool _isDirectory; bool _isValid; String _path; public: ABoxFilesystemNode(); ABoxFilesystemNode(BPTR lock, CONST_STRPTR display_name = NULL); ~ABoxFilesystemNode(); virtual String displayName() const { return _displayName; } virtual String name() const { return _displayName; } virtual bool isValid() const { return _isValid; } virtual bool isDirectory() const { return _isDirectory; } virtual String path() const { return _path; } virtual bool listDir(AbstractFSList &list, ListMode mode) const; static AbstractFSList listRoot(); virtual AbstractFilesystemNode *parent() const; virtual AbstractFilesystemNode *child(const String &n) const; }; AbstractFilesystemNode *AbstractFilesystemNode::getCurrentDirectory() { return AbstractFilesystemNode::getRoot(); } AbstractFilesystemNode *AbstractFilesystemNode::getRoot() { return new ABoxFilesystemNode(); } ABoxFilesystemNode::ABoxFilesystemNode() { _displayName = "Mounted Volumes"; _isValid = true; _isDirectory = true; _path = ""; _lock = NULL; } ABoxFilesystemNode::ABoxFilesystemNode(BPTR lock, CONST_STRPTR display_name) { int bufsize = 256; _lock = NULL; for (;;) { char n[bufsize]; if (NameFromLock(lock, n, bufsize) != DOSFALSE) { _path = n; _displayName = display_name ? display_name : FilePart(n); break; } if (IoErr() != ERROR_LINE_TOO_LONG) { _isValid = false; warning("Error while retrieving path name: %d", IoErr()); return; } bufsize *= 2; } _isValid = false; FileInfoBlock *fib = (FileInfoBlock*) AllocDosObject(DOS_FIB, NULL); if (fib == NULL) { warning("Failed to allocate memory for FileInfoBlock"); return; } if (Examine(lock, fib) != DOSFALSE) { _isDirectory = fib->fib_EntryType > 0; if (_isDirectory) { if (fib->fib_EntryType != ST_ROOT) _path += "/"; _lock = DupLock(lock); _isValid = (_lock != NULL); } else _isValid = true; } FreeDosObject(DOS_FIB, fib); } ABoxFilesystemNode::~ABoxFilesystemNode() { if (_lock) { UnLock(_lock); _lock = NULL; } } bool ABoxFilesystemNode::listDir(AbstractFSList &myList, ListMode mode) const { if (!_isValid) error("listDir() called on invalid node"); if (!_isDirectory) error("listDir() called on file node"); if (_lock == NULL) { /* This is the root node */ myList = listRoot(); return true; } /* "Normal" file system directory */ FileInfoBlock *fib = (FileInfoBlock*) AllocDosObject(DOS_FIB, NULL); if (fib == NULL) { warning("Failed to allocate memory for FileInfoBlock"); return false; } if (Examine(_lock, fib) != DOSFALSE) { while (ExNext(_lock, fib) != DOSFALSE) { ABoxFilesystemNode *entry; String full_path; BPTR lock; if ((fib->fib_EntryType > 0 && (mode & FilesystemNode::kListDirectoriesOnly)) || (fib->fib_EntryType < 0 && (mode & FilesystemNode::kListFilesOnly))) { full_path = _path; full_path += fib->fib_FileName; lock = Lock(full_path.c_str(), SHARED_LOCK); if (lock) { entry = new ABoxFilesystemNode(lock); if (entry) { if (entry->isValid()) myList.push_back(entry); else delete entry; } UnLock(lock); } } } if (IoErr() != ERROR_NO_MORE_ENTRIES) warning("Error while reading directory: %d", IoErr()); } FreeDosObject(DOS_FIB, fib); return tree; } AbstractFilesystemNode *ABoxFilesystemNode::parent() const { AbstractFilesystemNode *node = NULL; if (!_isDirectory) error("parent() called on file node"); if (_lock == NULL) { /* Parent of the root is the root itself */ node = 0; } else { BPTR parent_lock = ParentDir(_lock); if (parent_lock) { node = new ABoxFilesystemNode(parent_lock); UnLock(parent_lock); } else node = new ABoxFilesystemNode(); } return node; } AbstractFilesystemNode *ABoxFilesystemNode::child(const String &n) const { TODO } AbstractFSList ABoxFilesystemNode::listRoot() { AbstractFSList myList; DosList *dosList; CONST ULONG lockDosListFlags = LDF_READ | LDF_VOLUMES; char n[256]; dosList = LockDosList(lockDosListFlags); if (dosList == NULL) { warning("Could not lock dos list"); return myList; } dosList = NextDosEntry(dosList, LDF_VOLUMES); while (dosList) { if (dosList->dol_Type == DLT_VOLUME && // Should always be true, but ... dosList->dol_Name && // Same here dosList->dol_Task // Will be NULL if volume is removed from drive but still in use by some program ) { ABoxFilesystemNode *entry; CONST_STRPTR volume_name = (CONST_STRPTR)BADDR(dosList->dol_Name)+1; CONST_STRPTR device_name = (CONST_STRPTR)((struct Task *)dosList->dol_Task->mp_SigTask)->tc_Node.ln_Name; BPTR volume_lock; strcpy(n, volume_name); strcat(n, ":"); volume_lock = Lock(n, SHARED_LOCK); if (volume_lock) { sprintf(n, "%s (%s)", volume_name, device_name); entry = new ABoxFilesystemNode(volume_lock, n); if (entry) { if (entry->isValid()) myList.push_back(entry); else delete entry; } UnLock(volume_lock); } } dosList = NextDosEntry(dosList, LDF_VOLUMES); } UnLockDosList(lockDosListFlags); return myList; } #endif // defined(__MORPHOS__)