Ryujinx/Ryujinx.HLE/HOS/Services/FspSrv/IFileSystem.cs

403 lines
12 KiB
C#
Raw Normal View History

using Ryujinx.HLE.FileSystem;
using Ryujinx.HLE.HOS.Ipc;
using System;
using System.Collections.Generic;
2018-02-04 23:08:20 +00:00
using System.IO;
using static Ryujinx.HLE.HOS.ErrorCode;
using static Ryujinx.HLE.Utilities.StringUtils;
2018-02-04 23:08:20 +00:00
namespace Ryujinx.HLE.HOS.Services.FspSrv
2018-02-04 23:08:20 +00:00
{
class IFileSystem : IpcService
2018-02-04 23:08:20 +00:00
{
2018-12-01 20:01:59 +00:00
private Dictionary<int, ServiceProcessRequest> _commands;
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
public override IReadOnlyDictionary<int, ServiceProcessRequest> Commands => _commands;
2018-12-01 20:01:59 +00:00
private HashSet<string> _openPaths;
2018-12-01 20:01:59 +00:00
private string _path;
2018-12-01 20:01:59 +00:00
private IFileSystemProvider _provider;
2018-12-01 20:01:59 +00:00
public IFileSystem(string path, IFileSystemProvider provider)
2018-02-04 23:08:20 +00:00
{
2018-12-01 20:01:59 +00:00
_commands = new Dictionary<int, ServiceProcessRequest>()
{
2018-04-24 18:57:39 +00:00
{ 0, CreateFile },
{ 1, DeleteFile },
{ 2, CreateDirectory },
{ 3, DeleteDirectory },
{ 4, DeleteDirectoryRecursively },
{ 5, RenameFile },
{ 6, RenameDirectory },
{ 7, GetEntryType },
{ 8, OpenFile },
{ 9, OpenDirectory },
{ 10, Commit },
{ 11, GetFreeSpaceSize },
{ 12, GetTotalSpaceSize },
{ 13, CleanDirectoryRecursively },
//{ 14, GetFileTimeStampRaw }
};
2018-12-01 20:01:59 +00:00
_openPaths = new HashSet<string>();
2018-12-01 20:01:59 +00:00
this._path = path;
this._provider = provider;
2018-02-04 23:08:20 +00:00
}
// CreateFile(u32 mode, u64 size, buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long CreateFile(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
long mode = context.RequestData.ReadInt64();
int size = context.RequestData.ReadInt32();
2018-12-01 20:01:59 +00:00
string fileName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (fileName == null)
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (_provider.FileExists(fileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyExists);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(fileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
return _provider.CreateFile(fileName, size);
}
// DeleteFile(buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long DeleteFile(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
string fileName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (!_provider.FileExists(fileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(fileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
return _provider.DeleteFile(fileName);
}
// CreateDirectory(buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long CreateDirectory(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
string dirName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (dirName == null)
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (_provider.DirectoryExists(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyExists);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
_provider.CreateDirectory(dirName);
return 0;
}
// DeleteDirectory(buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long DeleteDirectory(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
return DeleteDirectory(context, false);
}
// DeleteDirectoryRecursively(buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long DeleteDirectoryRecursively(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
return DeleteDirectory(context, true);
}
2018-12-01 20:01:59 +00:00
private long DeleteDirectory(ServiceCtx context, bool recursive)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
string dirName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (!Directory.Exists(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
_provider.DeleteDirectory(dirName, recursive);
return 0;
}
// RenameFile(buffer<bytes<0x301>, 0x19, 0x301> oldPath, buffer<bytes<0x301>, 0x19, 0x301> newPath)
2018-12-01 20:01:59 +00:00
public long RenameFile(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string oldName = ReadUtf8String(context, 0);
string newName = ReadUtf8String(context, 1);
2018-12-01 20:01:59 +00:00
string oldFileName = _provider.GetFullPath(oldName);
string newFileName = _provider.GetFullPath(newName);
2018-12-01 20:01:59 +00:00
if (_provider.FileExists(oldFileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (_provider.FileExists(newFileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyExists);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(oldFileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
return _provider.RenameFile(oldFileName, newFileName);
}
// RenameDirectory(buffer<bytes<0x301>, 0x19, 0x301> oldPath, buffer<bytes<0x301>, 0x19, 0x301> newPath)
2018-12-01 20:01:59 +00:00
public long RenameDirectory(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string oldName = ReadUtf8String(context, 0);
string newName = ReadUtf8String(context, 1);
2018-12-01 20:01:59 +00:00
string oldDirName = _provider.GetFullPath(oldName);
string newDirName = _provider.GetFullPath(newName);
2018-12-01 20:01:59 +00:00
if (!_provider.DirectoryExists(oldDirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (!_provider.DirectoryExists(newDirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyExists);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(oldDirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
return _provider.RenameDirectory(oldDirName, newDirName);
}
// GetEntryType(buffer<bytes<0x301>, 0x19, 0x301> path) -> nn::fssrv::sf::DirectoryEntryType
2018-12-01 20:01:59 +00:00
public long GetEntryType(ServiceCtx context)
2018-02-04 23:08:20 +00:00
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
string fileName = _provider.GetFullPath(name);
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
if (_provider.FileExists(fileName))
2018-02-04 23:08:20 +00:00
{
2018-12-01 20:01:59 +00:00
context.ResponseData.Write(1);
2018-02-04 23:08:20 +00:00
}
2018-12-01 20:01:59 +00:00
else if (_provider.DirectoryExists(fileName))
{
2018-12-01 20:01:59 +00:00
context.ResponseData.Write(0);
}
else
{
2018-12-01 20:01:59 +00:00
context.ResponseData.Write(0);
2018-02-04 23:08:20 +00:00
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-02-04 23:08:20 +00:00
return 0;
}
// OpenFile(u32 mode, buffer<bytes<0x301>, 0x19, 0x301> path) -> object<nn::fssrv::sf::IFile> file
2018-12-01 20:01:59 +00:00
public long OpenFile(ServiceCtx context)
2018-02-04 23:08:20 +00:00
{
2018-12-01 20:01:59 +00:00
int filterFlags = context.RequestData.ReadInt32();
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
string fileName = _provider.GetFullPath(name);
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
if (!_provider.FileExists(fileName))
2018-02-04 23:08:20 +00:00
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
2018-02-04 23:08:20 +00:00
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(fileName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-02-04 23:08:20 +00:00
2018-12-01 20:01:59 +00:00
long error = _provider.OpenFile(fileName, out IFile fileInterface);
2018-12-01 20:01:59 +00:00
if (error == 0)
{
2018-12-01 20:01:59 +00:00
fileInterface.Disposed += RemoveFileInUse;
2018-12-01 20:01:59 +00:00
lock (_openPaths)
{
2018-12-01 20:01:59 +00:00
_openPaths.Add(fileName);
}
2018-12-01 20:01:59 +00:00
MakeObject(context, fileInterface);
return 0;
}
2018-12-01 20:01:59 +00:00
return error;
}
2018-02-04 23:08:20 +00:00
// OpenDirectory(u32 filter_flags, buffer<bytes<0x301>, 0x19, 0x301> path) -> object<nn::fssrv::sf::IDirectory> directory
2018-12-01 20:01:59 +00:00
public long OpenDirectory(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
int filterFlags = context.RequestData.ReadInt32();
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
string dirName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (!_provider.DirectoryExists(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
long error = _provider.OpenDirectory(dirName, filterFlags, out IDirectory dirInterface);
2018-12-01 20:01:59 +00:00
if (error == 0)
{
2018-12-01 20:01:59 +00:00
dirInterface.Disposed += RemoveDirectoryInUse;
2018-12-01 20:01:59 +00:00
lock (_openPaths)
{
2018-12-01 20:01:59 +00:00
_openPaths.Add(dirName);
}
2018-12-01 20:01:59 +00:00
MakeObject(context, dirInterface);
}
2018-12-01 20:01:59 +00:00
return error;
2018-02-04 23:08:20 +00:00
}
// Commit()
2018-12-01 20:01:59 +00:00
public long Commit(ServiceCtx context)
2018-02-04 23:08:20 +00:00
{
return 0;
}
// GetFreeSpaceSize(buffer<bytes<0x301>, 0x19, 0x301> path) -> u64 totalFreeSpace
2018-12-01 20:01:59 +00:00
public long GetFreeSpaceSize(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
context.ResponseData.Write(_provider.GetFreeSpace(context));
return 0;
}
// GetTotalSpaceSize(buffer<bytes<0x301>, 0x19, 0x301> path) -> u64 totalSize
2018-12-01 20:01:59 +00:00
public long GetTotalSpaceSize(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
context.ResponseData.Write(_provider.GetFreeSpace(context));
return 0;
}
// CleanDirectoryRecursively(buffer<bytes<0x301>, 0x19, 0x301> path)
2018-12-01 20:01:59 +00:00
public long CleanDirectoryRecursively(ServiceCtx context)
{
2018-12-01 20:01:59 +00:00
string name = ReadUtf8String(context);
2018-12-01 20:01:59 +00:00
string dirName = _provider.GetFullPath(name);
2018-12-01 20:01:59 +00:00
if (!_provider.DirectoryExists(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathDoesNotExist);
}
2018-12-01 20:01:59 +00:00
if (IsPathAlreadyInUse(dirName))
{
return MakeError(ErrorModule.Fs, FsErr.PathAlreadyInUse);
}
2018-12-01 20:01:59 +00:00
foreach (DirectoryEntry entry in _provider.GetEntries(dirName))
{
2018-12-01 20:01:59 +00:00
if (_provider.DirectoryExists(entry.Path))
{
2018-12-01 20:01:59 +00:00
_provider.DeleteDirectory(entry.Path, true);
}
2018-12-01 20:01:59 +00:00
else if (_provider.FileExists(entry.Path))
{
2018-12-01 20:01:59 +00:00
_provider.DeleteFile(entry.Path);
}
}
return 0;
}
2018-12-01 20:01:59 +00:00
private bool IsPathAlreadyInUse(string path)
{
2018-12-01 20:01:59 +00:00
lock (_openPaths)
{
2018-12-01 20:01:59 +00:00
return _openPaths.Contains(path);
}
}
private void RemoveFileInUse(object sender, EventArgs e)
{
2018-12-01 20:01:59 +00:00
IFile fileInterface = (IFile)sender;
2018-12-01 20:01:59 +00:00
lock (_openPaths)
{
2018-12-01 20:01:59 +00:00
fileInterface.Disposed -= RemoveFileInUse;
2018-12-01 20:01:59 +00:00
_openPaths.Remove(fileInterface.HostPath);
}
}
private void RemoveDirectoryInUse(object sender, EventArgs e)
{
2018-12-01 20:01:59 +00:00
IDirectory dirInterface = (IDirectory)sender;
2018-12-01 20:01:59 +00:00
lock (_openPaths)
{
2018-12-01 20:01:59 +00:00
dirInterface.Disposed -= RemoveDirectoryInUse;
2018-12-01 20:01:59 +00:00
_openPaths.Remove(dirInterface.DirectoryPath);
}
}
2018-02-04 23:08:20 +00:00
}
}