File: android-fs-factory.cpp

package info (click to toggle)
scummvm 2.9.1%2Bdfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 450,580 kB
  • sloc: cpp: 4,299,825; asm: 28,322; python: 12,901; sh: 11,302; java: 9,289; xml: 7,895; perl: 2,639; ansic: 2,465; yacc: 1,670; javascript: 1,020; makefile: 933; lex: 578; awk: 275; objc: 82; sed: 11; php: 1
file content (131 lines) | stat: -rw-r--r-- 3,912 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
/* ScummVM - Graphic Adventure Engine
 *
 * ScummVM is the legal property of its developers, whose names
 * are too numerous to list here. Please refer to the COPYRIGHT
 * file distributed with this source distribution.
 *
 * 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 3 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, see <http://www.gnu.org/licenses/>.
 *
 */

#include "backends/platform/android/jni-android.h"

#include "backends/fs/android/android-fs-factory.h"
#include "backends/fs/android/android-posix-fs.h"
#include "backends/fs/android/android-saf-fs.h"

namespace Common {
DECLARE_SINGLETON(AndroidFilesystemFactory);
}

AndroidFilesystemFactory::AndroidFilesystemFactory() : _withSAF(false), _config(this) {
}

void AndroidFilesystemFactory::initSAF() {
	_withSAF = true;
	AndroidSAFFilesystemNode::initJNI();
}

AbstractFSNode *AndroidFilesystemFactory::makeRootFileNode() const {
	return new AndroidPOSIXFilesystemNode(_config);
}

AbstractFSNode *AndroidFilesystemFactory::makeCurrentDirectoryFileNode() const {
	// As current working directory can point outside of our data don't take any risk
	return makeRootFileNode();
}

AbstractFSNode *AndroidFilesystemFactory::makeFileNodePath(const Common::String &path) const {
	if (path.empty() || path.equals("/")) {
		return makeRootFileNode();
	}

	// If SAF works, it was a SAF URL
	if (_withSAF) {
		// Accept /saf as it can be used to create the tree in DumpFile
		if (path == AddSAFFakeNode::SAF_ADD_FAKE_PATH) {
			// Not a SAF mount point
			return new AddSAFFakeNode(true);
		}

		AbstractFSNode *node = AndroidSAFFilesystemNode::makeFromPath(path);
		if (node) {
			return node;
		}
	}

	return new AndroidPOSIXFilesystemNode(path, _config);
}

AndroidFilesystemFactory::Config::Config(const AndroidFilesystemFactory *factory) : _factory(factory),
	_storages(JNI::getAllStorageLocations()) {
}

void AndroidFilesystemFactory::getSAFTrees(AbstractFSList &list, bool allowSAFadd) const {
	if (!_withSAF) {
		// Nothing if no SAF
		return;
	}

	Common::Array<jobject> trees = JNI::getSAFTrees();

	list.reserve(trees.size() + (allowSAFadd ? 1 : 0));

	for (Common::Array<jobject>::iterator it = trees.begin(); it != trees.end(); it++) {
		AbstractFSNode *node = AndroidSAFFilesystemNode::makeFromTree(*it);
		if (!node) {
			continue;
		}

		list.push_back(node);
	}

	if (allowSAFadd) {
		list.push_back(new AddSAFFakeNode(false));
	}

}

bool AndroidFilesystemFactory::Config::getDrives(AbstractFSList &list, bool hidden) const {
	// For SAF
	_factory->getSAFTrees(list, true);

	list.reserve(list.size() + _storages.size() / 2);

	// For old POSIX way
	for (Common::Array<Common::String>::const_iterator it = _storages.begin(); it != _storages.end(); ++it) {
		const Common::String &driveName = *it;
		++it;
		const Common::String &drivePath = *it;

		AndroidPOSIXFilesystemNode *node = new AndroidPOSIXFilesystemNode(drivePath, *this);
		node->_displayName = driveName;

		list.push_back(node);
	}
	return true;
}

bool AndroidFilesystemFactory::Config::isDrive(const Common::String &path) const {
	// This function is called from DrivePOSIXFilesystemNode::isDrive
	// DrivePOSIXFilesystemNode is only used for POSIX code so no need to look for SAF

	for (Common::Array<Common::String>::const_iterator it = _storages.begin(); it != _storages.end(); it++) {
		++it;
		if (*it == path) {
			return true;
		}
	}
	return false;
}