Browse Source
Core: Add a Directory object, with both a stub and a passthrough implementations.
pull/15/merge
Core: Add a Directory object, with both a stub and a passthrough implementations.
pull/15/merge
6 changed files with 264 additions and 0 deletions
-
5src/core/CMakeLists.txt
-
53src/core/file_sys/directory.h
-
38src/core/file_sys/directory_romfs.cpp
-
37src/core/file_sys/directory_romfs.h
-
86src/core/file_sys/directory_sdmc.cpp
-
45src/core/file_sys/directory_sdmc.h
@ -0,0 +1,53 @@ |
|||
// Copyright 2014 Citra Emulator Project |
|||
// Licensed under GPLv2 |
|||
// Refer to the license.txt file included. |
|||
|
|||
#pragma once |
|||
|
|||
#include "common/common_types.h" |
|||
|
|||
#include "core/hle/kernel/kernel.h" |
|||
|
|||
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|||
// FileSys namespace |
|||
|
|||
namespace FileSys { |
|||
|
|||
// Structure of a directory entry, from http://3dbrew.org/wiki/FSDir:Read#Entry_format |
|||
const size_t FILENAME_LENGTH = 0x20C / 2; |
|||
struct Entry { |
|||
char16_t filename[FILENAME_LENGTH]; // Entry name (UTF-16, null-terminated) |
|||
char short_name[8]; // 8.3 file name ('longfilename' -> 'LONGFI~1') |
|||
char unknown1; // unknown (observed values: 0x0A, 0x70, 0xFD) |
|||
char extension[3]; // 8.3 file extension (set to spaces for directories) |
|||
char unknown2; // unknown (always 0x01) |
|||
char unknown3; // unknown (0x00 or 0x08) |
|||
char is_directory; // directory flag |
|||
char is_hidden; // hidden flag |
|||
char is_archive; // archive flag |
|||
char is_read_only; // read-only flag |
|||
u64 file_size; // file size (for files only) |
|||
}; |
|||
static_assert(sizeof(Entry) == 0x228, "Directory Entry struct isn't exactly 0x228 bytes long!"); |
|||
|
|||
class Directory : NonCopyable { |
|||
public: |
|||
Directory() { } |
|||
virtual ~Directory() { } |
|||
|
|||
/** |
|||
* List files contained in the directory |
|||
* @param count Number of entries to return at once in entries |
|||
* @param entries Buffer to read data into |
|||
* @return Number of entries listed |
|||
*/ |
|||
virtual u32 Read(const u32 count, Entry* entries) = 0; |
|||
|
|||
/** |
|||
* Close the directory |
|||
* @return true if the directory closed correctly |
|||
*/ |
|||
virtual bool Close() const = 0; |
|||
}; |
|||
|
|||
} // namespace FileSys |
|||
@ -0,0 +1,38 @@ |
|||
// Copyright 2014 Citra Emulator Project
|
|||
// Licensed under GPLv2
|
|||
// Refer to the license.txt file included.
|
|||
|
|||
#include "common/common_types.h"
|
|||
|
|||
#include "core/file_sys/directory_romfs.h"
|
|||
|
|||
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|||
// FileSys namespace
|
|||
|
|||
namespace FileSys { |
|||
|
|||
Directory_RomFS::Directory_RomFS() { |
|||
} |
|||
|
|||
Directory_RomFS::~Directory_RomFS() { |
|||
} |
|||
|
|||
/**
|
|||
* List files contained in the directory |
|||
* @param count Number of entries to return at once in entries |
|||
* @param entries Buffer to read data into |
|||
* @return Number of entries listed |
|||
*/ |
|||
u32 Directory_RomFS::Read(const u32 count, Entry* entries) { |
|||
return 0; |
|||
} |
|||
|
|||
/**
|
|||
* Close the directory |
|||
* @return true if the directory closed correctly |
|||
*/ |
|||
bool Directory_RomFS::Close() const { |
|||
return false; |
|||
} |
|||
|
|||
} // namespace FileSys
|
|||
@ -0,0 +1,37 @@ |
|||
// Copyright 2014 Citra Emulator Project |
|||
// Licensed under GPLv2 |
|||
// Refer to the license.txt file included. |
|||
|
|||
#pragma once |
|||
|
|||
#include "common/common_types.h" |
|||
|
|||
#include "core/file_sys/directory.h" |
|||
#include "core/loader/loader.h" |
|||
|
|||
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|||
// FileSys namespace |
|||
|
|||
namespace FileSys { |
|||
|
|||
class Directory_RomFS final : public Directory { |
|||
public: |
|||
Directory_RomFS(); |
|||
~Directory_RomFS() override; |
|||
|
|||
/** |
|||
* List files contained in the directory |
|||
* @param count Number of entries to return at once in entries |
|||
* @param entries Buffer to read data into |
|||
* @return Number of entries listed |
|||
*/ |
|||
u32 Read(const u32 count, Entry* entries) override; |
|||
|
|||
/** |
|||
* Close the directory |
|||
* @return true if the directory closed correctly |
|||
*/ |
|||
bool Close() const override; |
|||
}; |
|||
|
|||
} // namespace FileSys |
|||
@ -0,0 +1,86 @@ |
|||
// Copyright 2014 Citra Emulator Project
|
|||
// Licensed under GPLv2
|
|||
// Refer to the license.txt file included.
|
|||
|
|||
#include <sys/stat.h>
|
|||
|
|||
#include "common/common_types.h"
|
|||
#include "common/file_util.h"
|
|||
|
|||
#include "core/file_sys/directory_sdmc.h"
|
|||
#include "core/file_sys/archive_sdmc.h"
|
|||
|
|||
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|||
// FileSys namespace
|
|||
|
|||
namespace FileSys { |
|||
|
|||
Directory_SDMC::Directory_SDMC(const Archive_SDMC* archive, const std::string& path) { |
|||
// TODO(Link Mauve): normalize path into an absolute path without "..", it can currently bypass
|
|||
// the root directory we set while opening the archive.
|
|||
// For example, opening /../../usr/bin can give the emulated program your installed programs.
|
|||
std::string absolute_path = archive->GetMountPoint() + path; |
|||
entry_count = FileUtil::ScanDirectoryTree(absolute_path, entry); |
|||
current_entry = 0; |
|||
} |
|||
|
|||
Directory_SDMC::~Directory_SDMC() { |
|||
Close(); |
|||
} |
|||
|
|||
/**
|
|||
* List files contained in the directory |
|||
* @param count Number of entries to return at once in entries |
|||
* @param entries Buffer to read data into |
|||
* @return Number of entries listed |
|||
*/ |
|||
u32 Directory_SDMC::Read(const u32 count, Entry* entries) { |
|||
u32 i; |
|||
for (i = 0; i < count && current_entry < entry_count; ++i) { |
|||
FileUtil::FSTEntry file = entry.children[current_entry]; |
|||
std::string filename = file.virtualName; |
|||
WARN_LOG(FILESYS, "File %s: size=%d dir=%d", filename.c_str(), file.size, file.isDirectory); |
|||
|
|||
Entry* entry = &entries[i]; |
|||
|
|||
// TODO(Link Mauve): use a proper conversion to UTF-16.
|
|||
for (int j = 0; j < FILENAME_LENGTH; ++j) { |
|||
entry->filename[j] = filename[j]; |
|||
if (!filename[j]) |
|||
break; |
|||
} |
|||
|
|||
// Split the filename into 8.3 format.
|
|||
// TODO(Link Mauve): move that to common, I guess, and make it more robust to long filenames.
|
|||
std::string::size_type n = filename.rfind('.'); |
|||
if (n == std::string::npos) { |
|||
strncpy(entry->short_name, filename.c_str(), 8); |
|||
memset(entry->extension, '\0', 3); |
|||
} else { |
|||
strncpy(entry->short_name, filename.substr(0, n).c_str(), 8); |
|||
strncpy(entry->extension, filename.substr(n + 1).c_str(), 8); |
|||
} |
|||
|
|||
entry->is_directory = file.isDirectory; |
|||
entry->file_size = file.size; |
|||
|
|||
// We emulate a SD card where the archive bit has never been cleared, as it would be on
|
|||
// most user SD cards.
|
|||
// Some homebrews (blargSNES for instance) are known to mistakenly use the archive bit as a
|
|||
// file bit.
|
|||
entry->is_archive = !file.isDirectory; |
|||
|
|||
++current_entry; |
|||
} |
|||
return i; |
|||
} |
|||
|
|||
/**
|
|||
* Close the directory |
|||
* @return true if the directory closed correctly |
|||
*/ |
|||
bool Directory_SDMC::Close() const { |
|||
return true; |
|||
} |
|||
|
|||
} // namespace FileSys
|
|||
@ -0,0 +1,45 @@ |
|||
// Copyright 2014 Citra Emulator Project |
|||
// Licensed under GPLv2 |
|||
// Refer to the license.txt file included. |
|||
|
|||
#pragma once |
|||
|
|||
#include "common/common_types.h" |
|||
#include "common/file_util.h" |
|||
|
|||
#include "core/file_sys/directory.h" |
|||
#include "core/file_sys/archive_sdmc.h" |
|||
#include "core/loader/loader.h" |
|||
|
|||
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|||
// FileSys namespace |
|||
|
|||
namespace FileSys { |
|||
|
|||
class Directory_SDMC final : public Directory { |
|||
public: |
|||
Directory_SDMC(); |
|||
Directory_SDMC(const Archive_SDMC* archive, const std::string& path); |
|||
~Directory_SDMC() override; |
|||
|
|||
/** |
|||
* List files contained in the directory |
|||
* @param count Number of entries to return at once in entries |
|||
* @param entries Buffer to read data into |
|||
* @return Number of entries listed |
|||
*/ |
|||
u32 Read(const u32 count, Entry* entries) override; |
|||
|
|||
/** |
|||
* Close the directory |
|||
* @return true if the directory closed correctly |
|||
*/ |
|||
bool Close() const override; |
|||
|
|||
private: |
|||
u32 entry_count; |
|||
u32 current_entry; |
|||
FileUtil::FSTEntry entry; |
|||
}; |
|||
|
|||
} // namespace FileSys |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue