2005-05-09 18:58:19 +00:00
|
|
|
//============================================================================
|
|
|
|
//
|
2016-12-30 00:00:30 +00:00
|
|
|
// SSSS tt lll lll
|
|
|
|
// SS SS tt ll ll
|
|
|
|
// SS tttttt eeee ll ll aaaa
|
2005-05-09 18:58:19 +00:00
|
|
|
// SSSS tt ee ee ll ll aa
|
|
|
|
// SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator"
|
|
|
|
// SS SS tt ee ll ll aa aa
|
|
|
|
// SSSS ttt eeeee llll llll aaaaa
|
|
|
|
//
|
2019-01-01 15:05:51 +00:00
|
|
|
// Copyright (c) 1995-2019 by Bradford W. Mott, Stephen Anthony
|
2010-04-10 21:37:23 +00:00
|
|
|
// and the Stella Team
|
2005-05-09 18:58:19 +00:00
|
|
|
//
|
2010-01-10 03:23:32 +00:00
|
|
|
// See the file "License.txt" for information on usage and redistribution of
|
2005-05-09 18:58:19 +00:00
|
|
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
|
|
|
//
|
|
|
|
// Based on code from ScummVM - Scumm Interpreter
|
|
|
|
// Copyright (C) 2002-2004 The ScummVM project
|
|
|
|
//============================================================================
|
|
|
|
|
2013-02-10 21:47:47 +00:00
|
|
|
#include "FSNodeFactory.hxx"
|
2007-09-03 18:37:24 +00:00
|
|
|
#include "FSNode.hxx"
|
2005-05-09 18:58:19 +00:00
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2019-12-25 01:56:44 +00:00
|
|
|
FilesystemNode::FilesystemNode(const AbstractFSNodePtr& realNode)
|
|
|
|
: _realNode(realNode)
|
2005-05-09 18:58:19 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2009-01-11 19:10:40 +00:00
|
|
|
FilesystemNode::FilesystemNode(const string& p)
|
2005-05-09 18:58:19 +00:00
|
|
|
{
|
2013-02-10 21:47:47 +00:00
|
|
|
// Is this potentially a ZIP archive?
|
2019-04-24 18:06:20 +00:00
|
|
|
#if defined(ZIP_SUPPORT)
|
2019-06-22 23:39:42 +00:00
|
|
|
if (BSPF::containsIgnoreCase(p, ".zip"))
|
2019-04-13 23:27:46 +00:00
|
|
|
_realNode = FilesystemNodeFactory::create(p, FilesystemNodeFactory::Type::ZIP);
|
2013-02-10 21:47:47 +00:00
|
|
|
else
|
2019-04-24 18:06:20 +00:00
|
|
|
#endif
|
2019-04-13 23:27:46 +00:00
|
|
|
_realNode = FilesystemNodeFactory::create(p, FilesystemNodeFactory::Type::SYSTEM);
|
2005-05-09 18:58:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2009-01-11 19:10:40 +00:00
|
|
|
bool FilesystemNode::exists() const
|
2005-05-09 18:58:19 +00:00
|
|
|
{
|
2013-02-02 18:34:25 +00:00
|
|
|
return _realNode ? _realNode->exists() : false;
|
2005-05-09 18:58:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2019-08-17 23:43:15 +00:00
|
|
|
bool FilesystemNode::getChildren(FSList& fslist, ListMode mode,
|
|
|
|
const NameFilter& filter) const
|
2005-05-09 18:58:19 +00:00
|
|
|
{
|
2009-01-11 19:10:40 +00:00
|
|
|
if (!_realNode || !_realNode->isDirectory())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
AbstractFSList tmp;
|
2012-04-15 21:39:15 +00:00
|
|
|
tmp.reserve(fslist.capacity());
|
2009-01-11 19:10:40 +00:00
|
|
|
|
2019-06-22 23:39:42 +00:00
|
|
|
if (!_realNode->getChildren(tmp, mode))
|
2009-01-11 19:10:40 +00:00
|
|
|
return false;
|
|
|
|
|
2019-06-22 23:39:42 +00:00
|
|
|
std::sort(tmp.begin(), tmp.end(),
|
|
|
|
[](const AbstractFSNodePtr& node1, const AbstractFSNodePtr& node2)
|
|
|
|
{
|
|
|
|
if (node1->isDirectory() != node2->isDirectory())
|
|
|
|
return node1->isDirectory();
|
|
|
|
else
|
|
|
|
return BSPF::compareIgnoreCase(node1->getName(), node2->getName()) < 0;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// Add parent node, if it is valid to do so
|
|
|
|
if (hasParent())
|
|
|
|
{
|
|
|
|
FilesystemNode parent = getParent();
|
|
|
|
parent.setName(" [..]");
|
|
|
|
fslist.emplace_back(parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
// And now add the rest of the entries
|
2014-11-02 23:40:20 +00:00
|
|
|
for (const auto& i: tmp)
|
2019-06-22 23:39:42 +00:00
|
|
|
{
|
|
|
|
#if defined(ZIP_SUPPORT)
|
|
|
|
if (BSPF::endsWithIgnoreCase(i->getPath(), ".zip"))
|
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
// Force ZIP c'tor to be called
|
2019-06-22 23:39:42 +00:00
|
|
|
AbstractFSNodePtr ptr = FilesystemNodeFactory::create(i->getPath(),
|
|
|
|
FilesystemNodeFactory::Type::ZIP);
|
2019-08-17 23:43:15 +00:00
|
|
|
FilesystemNode node(ptr);
|
|
|
|
if (filter(node))
|
|
|
|
fslist.emplace_back(node);
|
2019-06-22 23:39:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
// Make directories stand out
|
|
|
|
if (i->isDirectory())
|
|
|
|
i->setName(" [" + i->getName() + "]");
|
|
|
|
|
2019-08-17 23:43:15 +00:00
|
|
|
FilesystemNode node(i);
|
|
|
|
if (filter(node))
|
|
|
|
fslist.emplace_back(node);
|
2019-06-22 23:39:42 +00:00
|
|
|
}
|
|
|
|
}
|
2009-01-11 19:10:40 +00:00
|
|
|
|
|
|
|
return true;
|
2005-05-09 18:58:19 +00:00
|
|
|
}
|
|
|
|
|
2009-01-11 19:10:40 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2011-12-22 22:11:09 +00:00
|
|
|
const string& FilesystemNode::getName() const
|
2009-01-11 19:10:40 +00:00
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
return _realNode ? _realNode->getName() : EmptyString;
|
2009-01-11 19:10:40 +00:00
|
|
|
}
|
2005-05-09 18:58:19 +00:00
|
|
|
|
2019-06-22 23:39:42 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
void FilesystemNode::setName(const string& name)
|
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
if (_realNode)
|
|
|
|
_realNode->setName(name);
|
2019-06-22 23:39:42 +00:00
|
|
|
}
|
|
|
|
|
2009-01-11 19:10:40 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2013-06-21 12:15:32 +00:00
|
|
|
const string& FilesystemNode::getPath() const
|
2009-01-11 19:10:40 +00:00
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
return _realNode ? _realNode->getPath() : EmptyString;
|
2005-05-09 18:58:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2013-06-21 12:15:32 +00:00
|
|
|
string FilesystemNode::getShortPath() const
|
2005-05-09 18:58:19 +00:00
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
return _realNode ? _realNode->getShortPath() : EmptyString;
|
2013-06-21 12:15:32 +00:00
|
|
|
}
|
2005-05-09 18:58:19 +00:00
|
|
|
|
2013-06-21 12:15:32 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
string FilesystemNode::getNameWithExt(const string& ext) const
|
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
if (!_realNode)
|
|
|
|
return EmptyString;
|
|
|
|
|
2013-06-21 12:15:32 +00:00
|
|
|
size_t pos = _realNode->getName().find_last_of("/\\");
|
|
|
|
string s = pos == string::npos ? _realNode->getName() :
|
|
|
|
_realNode->getName().substr(pos+1);
|
|
|
|
|
2019-12-25 01:41:36 +00:00
|
|
|
pos = s.find_last_of('.');
|
2013-06-21 12:15:32 +00:00
|
|
|
return (pos != string::npos) ? s.replace(pos, string::npos, ext) : s + ext;
|
2005-05-09 18:58:19 +00:00
|
|
|
}
|
2006-03-19 22:06:20 +00:00
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2013-06-21 12:15:32 +00:00
|
|
|
string FilesystemNode::getPathWithExt(const string& ext) const
|
2006-03-19 22:06:20 +00:00
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
if (!_realNode)
|
|
|
|
return EmptyString;
|
|
|
|
|
2013-06-21 12:15:32 +00:00
|
|
|
string s = _realNode->getPath();
|
|
|
|
|
2019-12-25 01:41:36 +00:00
|
|
|
size_t pos = s.find_last_of('.');
|
2013-06-21 12:15:32 +00:00
|
|
|
return (pos != string::npos) ? s.replace(pos, string::npos, ext) : s + ext;
|
2011-12-22 21:29:24 +00:00
|
|
|
}
|
|
|
|
|
2013-06-21 12:15:32 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::hasParent() const
|
|
|
|
{
|
2019-06-22 23:39:42 +00:00
|
|
|
return _realNode ? _realNode->hasParent() : false;
|
2013-06-21 12:15:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
FilesystemNode FilesystemNode::getParent() const
|
|
|
|
{
|
2019-06-23 17:13:09 +00:00
|
|
|
if (!_realNode)
|
2013-06-21 12:15:32 +00:00
|
|
|
return *this;
|
|
|
|
|
2018-08-06 01:02:11 +00:00
|
|
|
AbstractFSNodePtr node = _realNode->getParent();
|
2017-10-11 14:53:54 +00:00
|
|
|
return node ? FilesystemNode(node) : *this;
|
2009-06-18 13:07:51 +00:00
|
|
|
}
|
|
|
|
|
2009-01-11 19:10:40 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::isDirectory() const
|
|
|
|
{
|
2013-02-02 18:34:25 +00:00
|
|
|
return _realNode ? _realNode->isDirectory() : false;
|
2009-01-11 19:10:40 +00:00
|
|
|
}
|
|
|
|
|
2012-05-16 20:52:33 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::isFile() const
|
|
|
|
{
|
2013-02-02 18:34:25 +00:00
|
|
|
return _realNode ? _realNode->isFile() : false;
|
2012-05-16 20:52:33 +00:00
|
|
|
}
|
|
|
|
|
2009-01-11 19:10:40 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::isReadable() const
|
|
|
|
{
|
2013-02-02 18:34:25 +00:00
|
|
|
return _realNode ? _realNode->isReadable() : false;
|
2009-01-11 19:10:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::isWritable() const
|
|
|
|
{
|
2013-02-02 18:34:25 +00:00
|
|
|
return _realNode ? _realNode->isWritable() : false;
|
|
|
|
}
|
2009-01-11 19:10:40 +00:00
|
|
|
|
2013-02-02 18:34:25 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::makeDir()
|
|
|
|
{
|
|
|
|
return (_realNode && !_realNode->exists()) ? _realNode->makeDir() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
bool FilesystemNode::rename(const string& newfile)
|
|
|
|
{
|
|
|
|
return (_realNode && _realNode->exists()) ? _realNode->rename(newfile) : false;
|
2006-03-19 22:06:20 +00:00
|
|
|
}
|
2013-02-03 00:13:08 +00:00
|
|
|
|
2013-02-13 23:09:31 +00:00
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
2019-12-28 00:35:38 +00:00
|
|
|
size_t FilesystemNode::read(ByteBuffer& image) const
|
2013-02-13 23:09:31 +00:00
|
|
|
{
|
2019-12-28 00:35:38 +00:00
|
|
|
size_t size = 0;
|
2013-05-07 14:44:50 +00:00
|
|
|
|
|
|
|
// File must actually exist
|
2019-06-22 23:39:42 +00:00
|
|
|
if (!(exists() && isReadable()))
|
2015-06-12 20:44:09 +00:00
|
|
|
throw runtime_error("File not found/readable");
|
2013-02-13 23:09:31 +00:00
|
|
|
|
2019-04-24 18:06:20 +00:00
|
|
|
// First let the private subclass attempt to open the file
|
2019-06-23 17:13:09 +00:00
|
|
|
if (_realNode && (size = _realNode->read(image)) > 0)
|
2019-04-24 18:06:20 +00:00
|
|
|
return size;
|
|
|
|
|
|
|
|
// Otherwise, the default behaviour is to read from a normal C++ ifstream
|
|
|
|
image = make_unique<uInt8[]>(512 * 1024);
|
|
|
|
ifstream in(getPath(), std::ios::binary);
|
2019-06-22 23:39:42 +00:00
|
|
|
if (in)
|
2013-02-13 23:09:31 +00:00
|
|
|
{
|
2019-04-24 18:06:20 +00:00
|
|
|
in.seekg(0, std::ios::end);
|
|
|
|
std::streampos length = in.tellg();
|
|
|
|
in.seekg(0, std::ios::beg);
|
2013-02-13 23:09:31 +00:00
|
|
|
|
2019-06-22 23:39:42 +00:00
|
|
|
if (length == 0)
|
2015-06-12 20:44:09 +00:00
|
|
|
throw runtime_error("Zero-byte file");
|
2015-06-14 20:02:32 +00:00
|
|
|
|
2019-12-28 00:35:38 +00:00
|
|
|
size = std::min<size_t>(length, 512 * 1024);
|
2019-04-24 18:06:20 +00:00
|
|
|
in.read(reinterpret_cast<char*>(image.get()), size);
|
2013-02-13 23:09:31 +00:00
|
|
|
}
|
2013-05-07 14:44:50 +00:00
|
|
|
else
|
2019-04-24 18:06:20 +00:00
|
|
|
throw runtime_error("File open/read error");
|
|
|
|
|
|
|
|
return size;
|
2013-02-13 23:09:31 +00:00
|
|
|
}
|