fat: add support for mounting fat bis paritions (prodinfof, safe, user, system), inital work needed for fetching personalised es tickets.
This commit is contained in:
@@ -88,6 +88,7 @@ add_executable(sphaira
|
|||||||
source/threaded_file_transfer.cpp
|
source/threaded_file_transfer.cpp
|
||||||
source/title_info.cpp
|
source/title_info.cpp
|
||||||
source/minizip_helper.cpp
|
source/minizip_helper.cpp
|
||||||
|
source/fatfs.cpp
|
||||||
|
|
||||||
source/usb/base.cpp
|
source/usb/base.cpp
|
||||||
source/usb/usbds.cpp
|
source/usb/usbds.cpp
|
||||||
@@ -345,6 +346,14 @@ if (NOT USE_NEW_ZSTD)
|
|||||||
find_library(zstd_lib zstd REQUIRED)
|
find_library(zstd_lib zstd REQUIRED)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
add_library(fatfs
|
||||||
|
source/ff16/diskio.c
|
||||||
|
source/ff16/ff.c
|
||||||
|
source/ff16/ffunicode.c
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(fatfs PUBLIC source/ff16)
|
||||||
|
|
||||||
set_target_properties(sphaira PROPERTIES
|
set_target_properties(sphaira PROPERTIES
|
||||||
C_STANDARD 23
|
C_STANDARD 23
|
||||||
C_EXTENSIONS ON
|
C_EXTENSIONS ON
|
||||||
@@ -362,6 +371,7 @@ target_link_libraries(sphaira PRIVATE
|
|||||||
yyjson
|
yyjson
|
||||||
# libusbhsfs
|
# libusbhsfs
|
||||||
libnxtc
|
libnxtc
|
||||||
|
fatfs
|
||||||
|
|
||||||
${minizip_lib}
|
${minizip_lib}
|
||||||
ZLIB::ZLIB
|
ZLIB::ZLIB
|
||||||
|
|||||||
12
sphaira/include/fatfs.hpp
Normal file
12
sphaira/include/fatfs.hpp
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <switch.h>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace sphaira::fatfs {
|
||||||
|
|
||||||
|
Result MountAll();
|
||||||
|
void UnmountAll();
|
||||||
|
|
||||||
|
} // namespace sphaira::fatfs
|
||||||
@@ -37,5 +37,6 @@ using StdioEntries = std::vector<StdioEntry>;
|
|||||||
|
|
||||||
// set write=true to filter out write protected devices.
|
// set write=true to filter out write protected devices.
|
||||||
auto GetStdio(bool write) -> StdioEntries;
|
auto GetStdio(bool write) -> StdioEntries;
|
||||||
|
auto GetFat() -> StdioEntries;
|
||||||
|
|
||||||
} // namespace sphaira::location
|
} // namespace sphaira::location
|
||||||
|
|||||||
@@ -215,4 +215,9 @@ Result IsRightsIdPersonalised(const FsRightsId& id, bool* out);
|
|||||||
bool IsRightsIdValid(const FsRightsId& id);
|
bool IsRightsIdValid(const FsRightsId& id);
|
||||||
bool IsRightsIdFound(const FsRightsId& id, std::span<const FsRightsId> ids);
|
bool IsRightsIdFound(const FsRightsId& id, std::span<const FsRightsId> ids);
|
||||||
|
|
||||||
|
// wrapper around ipc.
|
||||||
|
Result GetCommonTicketAndCertificate(const FsRightsId& rights_id, std::vector<u8>& tik_out, std::vector<u8>& cert_out);
|
||||||
|
// fetches data from system es save.
|
||||||
|
Result GetPersonalisedTicketAndCertificate(const FsRightsId& rights_id, std::vector<u8>& tik_out, std::vector<u8>& cert_out);
|
||||||
|
|
||||||
} // namespace sphaira::es
|
} // namespace sphaira::es
|
||||||
|
|||||||
@@ -22,6 +22,7 @@
|
|||||||
#include "haze_helper.hpp"
|
#include "haze_helper.hpp"
|
||||||
#include "web.hpp"
|
#include "web.hpp"
|
||||||
#include "swkbd.hpp"
|
#include "swkbd.hpp"
|
||||||
|
#include "fatfs.hpp"
|
||||||
|
|
||||||
#include <nanovg_dk.h>
|
#include <nanovg_dk.h>
|
||||||
#include <minIni.h>
|
#include <minIni.h>
|
||||||
@@ -1432,6 +1433,10 @@ App::App(const char* argv0) {
|
|||||||
usbHsFsInitialize(1);
|
usbHsFsInitialize(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (R_FAILED(fatfs::MountAll())) {
|
||||||
|
log_write("[FAT] failed to mount bis\n");
|
||||||
|
}
|
||||||
|
|
||||||
curl::Init();
|
curl::Init();
|
||||||
|
|
||||||
#ifdef USE_NVJPG
|
#ifdef USE_NVJPG
|
||||||
@@ -2069,6 +2074,8 @@ App::~App() {
|
|||||||
usbHsFsExit();
|
usbHsFsExit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fatfs::UnmountAll();
|
||||||
|
|
||||||
log_write("\t[EXIT] time taken: %.2fs %zums\n", ts.GetSecondsD(), ts.GetMs());
|
log_write("\t[EXIT] time taken: %.2fs %zums\n", ts.GetSecondsD(), ts.GetMs());
|
||||||
|
|
||||||
if (App::GetLogEnable()) {
|
if (App::GetLogEnable()) {
|
||||||
|
|||||||
325
sphaira/source/fatfs.cpp
Normal file
325
sphaira/source/fatfs.cpp
Normal file
@@ -0,0 +1,325 @@
|
|||||||
|
#include "fatfs.hpp"
|
||||||
|
#include "defines.hpp"
|
||||||
|
#include "log.hpp"
|
||||||
|
#include "ff.h"
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include <cstring>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <sys/iosupport.h>
|
||||||
|
|
||||||
|
namespace sphaira::fatfs {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
// 256-512 are the best values, anything more has serious slow down
|
||||||
|
// due to non-seq reads.
|
||||||
|
struct BufferedFileData {
|
||||||
|
u8 data[1024 * 256];
|
||||||
|
s64 off;
|
||||||
|
s64 size;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum BisMountType {
|
||||||
|
BisMountType_PRODINFOF,
|
||||||
|
BisMountType_SAFE,
|
||||||
|
BisMountType_USER,
|
||||||
|
BisMountType_SYSTEM,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct FatStorageEntry {
|
||||||
|
FsStorage storage;
|
||||||
|
BufferedFileData buffered;
|
||||||
|
FATFS fs;
|
||||||
|
devoptab_t devoptab;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct BisMountEntry {
|
||||||
|
const FsBisPartitionId id;
|
||||||
|
const char* volume_name;
|
||||||
|
const char* mount_name;
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr BisMountEntry BIS_MOUNT_ENTRIES[] {
|
||||||
|
[BisMountType_PRODINFOF] = { FsBisPartitionId_CalibrationFile, "PRODINFOF", "PRODINFOF:/" },
|
||||||
|
[BisMountType_SAFE] = { FsBisPartitionId_SafeMode, "SAFE", "SAFE:/" },
|
||||||
|
[BisMountType_USER] = { FsBisPartitionId_User, "USER", "USER:/" },
|
||||||
|
[BisMountType_SYSTEM] = { FsBisPartitionId_System, "SYSTEM", "SYSTEM:/" },
|
||||||
|
};
|
||||||
|
static_assert(std::size(BIS_MOUNT_ENTRIES) == FF_VOLUMES);
|
||||||
|
|
||||||
|
FatStorageEntry g_fat_storage[FF_VOLUMES];
|
||||||
|
|
||||||
|
// crappy generic buffered io i wrote a while ago.
|
||||||
|
// this allows for 3-4x speed increase reading from storage.
|
||||||
|
// as it avoids reading very small chunks at a time.
|
||||||
|
// note: this works best when the file is not fragmented.
|
||||||
|
Result ReadFile(FsStorage* storage, BufferedFileData& m_buffered, void *_buffer, size_t file_off, size_t read_size) {
|
||||||
|
auto dst = static_cast<u8*>(_buffer);
|
||||||
|
size_t amount = 0;
|
||||||
|
|
||||||
|
// check if we already have this data buffered.
|
||||||
|
if (m_buffered.size) {
|
||||||
|
// check if we can read this data into the beginning of dst.
|
||||||
|
if (file_off < m_buffered.off + m_buffered.size && file_off >= m_buffered.off) {
|
||||||
|
const auto off = file_off - m_buffered.off;
|
||||||
|
const auto size = std::min<s64>(read_size, m_buffered.size - off);
|
||||||
|
std::memcpy(dst, m_buffered.data + off, size);
|
||||||
|
|
||||||
|
read_size -= size;
|
||||||
|
file_off += size;
|
||||||
|
amount += size;
|
||||||
|
dst += size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (read_size) {
|
||||||
|
m_buffered.off = 0;
|
||||||
|
m_buffered.size = 0;
|
||||||
|
|
||||||
|
// if the dst dst is big enough, read data in place.
|
||||||
|
if (read_size >= sizeof(m_buffered.data)) {
|
||||||
|
if (R_SUCCEEDED(fsStorageRead(storage, file_off, dst, read_size))) {
|
||||||
|
const auto bytes_read = read_size;
|
||||||
|
read_size -= bytes_read;
|
||||||
|
file_off += bytes_read;
|
||||||
|
amount += bytes_read;
|
||||||
|
dst += bytes_read;
|
||||||
|
|
||||||
|
// save the last chunk of data to the m_buffered io.
|
||||||
|
const auto max_advance = std::min(amount, sizeof(m_buffered.data));
|
||||||
|
m_buffered.off = file_off - max_advance;
|
||||||
|
m_buffered.size = max_advance;
|
||||||
|
std::memcpy(m_buffered.data, dst - max_advance, max_advance);
|
||||||
|
}
|
||||||
|
} else if (R_SUCCEEDED(fsStorageRead(storage, file_off, m_buffered.data, sizeof(m_buffered.data)))) {
|
||||||
|
const auto bytes_read = sizeof(m_buffered.data);
|
||||||
|
const auto max_advance = std::min(read_size, bytes_read);
|
||||||
|
std::memcpy(dst, m_buffered.data, max_advance);
|
||||||
|
|
||||||
|
m_buffered.off = file_off;
|
||||||
|
m_buffered.size = bytes_read;
|
||||||
|
|
||||||
|
read_size -= max_advance;
|
||||||
|
file_off += max_advance;
|
||||||
|
amount += max_advance;
|
||||||
|
dst += max_advance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void fill_stat(const FILINFO* fno, struct stat *st) {
|
||||||
|
memset(st, 0, sizeof(*st));
|
||||||
|
|
||||||
|
st->st_nlink = 1;
|
||||||
|
|
||||||
|
if (fno->fattrib & AM_DIR) {
|
||||||
|
st->st_mode = S_IFDIR | S_IRUSR | S_IRGRP | S_IROTH;
|
||||||
|
} else {
|
||||||
|
st->st_size = fno->fsize;
|
||||||
|
st->st_mode = S_IFREG | S_IRUSR | S_IRGRP | S_IROTH;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_errno(struct _reent *r, int err) {
|
||||||
|
r->_errno = err;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_open(struct _reent *r, void *fileStruct, const char *path, int flags, int mode) {
|
||||||
|
memset(fileStruct, 0, sizeof(FIL));
|
||||||
|
|
||||||
|
if (FR_OK != f_open((FIL*)fileStruct, path, FA_READ)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_close(struct _reent *r, void *fd) {
|
||||||
|
if (FR_OK != f_close((FIL*)fd)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t fat_read(struct _reent *r, void *fd, char *ptr, size_t len) {
|
||||||
|
UINT bytes_read;
|
||||||
|
if (FR_OK != f_read((FIL*)fd, ptr, len, &bytes_read)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
return bytes_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
off_t fat_seek(struct _reent *r, void *fd, off_t pos, int dir) {
|
||||||
|
if (dir == SEEK_CUR) {
|
||||||
|
pos += f_tell((FIL*)fd);
|
||||||
|
} else if (dir == SEEK_END) {
|
||||||
|
pos = f_size((FIL*)fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (FR_OK != f_lseek((FIL*)fd, pos)) {
|
||||||
|
set_errno(r, ENOENT);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
r->_errno = 0;
|
||||||
|
return f_tell((FIL*)fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_fstat(struct _reent *r, void *fd, struct stat *st) {
|
||||||
|
const FIL* file = (FIL*)fd;
|
||||||
|
|
||||||
|
/* Only fill the attr and size field, leaving the timestamp blank. */
|
||||||
|
FILINFO info = {0};
|
||||||
|
info.fattrib = file->obj.attr;
|
||||||
|
info.fsize = file->obj.objsize;
|
||||||
|
|
||||||
|
/* Fill stat info. */
|
||||||
|
fill_stat(&info, st);
|
||||||
|
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
DIR_ITER* fat_diropen(struct _reent *r, DIR_ITER *dirState, const char *path) {
|
||||||
|
memset(dirState->dirStruct, 0, sizeof(FDIR));
|
||||||
|
|
||||||
|
if (FR_OK != f_opendir((FDIR*)dirState->dirStruct, path)) {
|
||||||
|
set_errno(r, ENOENT);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
r->_errno = 0;
|
||||||
|
return dirState;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_dirreset(struct _reent *r, DIR_ITER *dirState) {
|
||||||
|
if (FR_OK != f_rewinddir((FDIR*)dirState->dirStruct)) {
|
||||||
|
log_write("[FAT] fat_dirreset failed\n");
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_dirnext(struct _reent *r, DIR_ITER *dirState, char *filename, struct stat *filestat) {
|
||||||
|
FILINFO fno{};
|
||||||
|
if (FR_OK != f_readdir((FDIR*)dirState->dirStruct, &fno)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fno.fname[0]) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(filename, fno.fname);
|
||||||
|
fill_stat(&fno, filestat);
|
||||||
|
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_dirclose(struct _reent *r, DIR_ITER *dirState) {
|
||||||
|
if (FR_OK != f_closedir((FDIR*)dirState->dirStruct)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_statvfs(struct _reent *r, const char *path, struct statvfs *buf) {
|
||||||
|
memset(buf, 0, sizeof(*buf));
|
||||||
|
|
||||||
|
// todo: find out how to calculate free size in read only.
|
||||||
|
const auto fat = (FatStorageEntry*)r->deviceData;
|
||||||
|
buf->f_bsize = FF_MAX_SS;
|
||||||
|
buf->f_frsize = FF_MAX_SS;
|
||||||
|
buf->f_blocks = ((fat->fs.n_fatent - 2) * (DWORD)fat->fs.csize);
|
||||||
|
buf->f_namemax = FF_LFN_BUF;
|
||||||
|
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fat_lstat(struct _reent *r, const char *file, struct stat *st) {
|
||||||
|
FILINFO fno;
|
||||||
|
if (FR_OK != f_stat(file, &fno)) {
|
||||||
|
return set_errno(r, ENOENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
fill_stat(&fno, st);
|
||||||
|
return r->_errno = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr devoptab_t DEVOPTAB = {
|
||||||
|
.structSize = sizeof(FIL),
|
||||||
|
.open_r = fat_open,
|
||||||
|
.close_r = fat_close,
|
||||||
|
.read_r = fat_read,
|
||||||
|
.seek_r = fat_seek,
|
||||||
|
.fstat_r = fat_fstat,
|
||||||
|
.stat_r = fat_lstat,
|
||||||
|
.dirStateSize = sizeof(FDIR),
|
||||||
|
.diropen_r = fat_diropen,
|
||||||
|
.dirreset_r = fat_dirreset,
|
||||||
|
.dirnext_r = fat_dirnext,
|
||||||
|
.dirclose_r = fat_dirclose,
|
||||||
|
.statvfs_r = fat_statvfs,
|
||||||
|
.lstat_r = fat_lstat,
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
Result MountAll() {
|
||||||
|
for (u32 i = 0; i < FF_VOLUMES; i++) {
|
||||||
|
auto& fat = g_fat_storage[i];
|
||||||
|
const auto& bis = BIS_MOUNT_ENTRIES[i];
|
||||||
|
|
||||||
|
log_write("[FAT] %s\n", bis.volume_name);
|
||||||
|
|
||||||
|
fat.devoptab = DEVOPTAB;
|
||||||
|
fat.devoptab.name = bis.volume_name;
|
||||||
|
fat.devoptab.deviceData = &fat;
|
||||||
|
|
||||||
|
R_TRY(fsOpenBisStorage(&fat.storage, bis.id));
|
||||||
|
log_write("[FAT] BIS SUCCESS %s\n", bis.volume_name);
|
||||||
|
|
||||||
|
R_UNLESS(FR_OK == f_mount(&fat.fs, bis.mount_name, 1), 0x1);
|
||||||
|
log_write("[FAT] MOUNT SUCCESS %s\n", bis.volume_name);
|
||||||
|
|
||||||
|
R_UNLESS(AddDevice(&fat.devoptab) >= 0, 0x1);
|
||||||
|
log_write("[FAT] DEVICE SUCCESS %s\n", bis.volume_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnmountAll() {
|
||||||
|
for (u32 i = 0; i < FF_VOLUMES; i++) {
|
||||||
|
auto& fat = g_fat_storage[i];
|
||||||
|
const auto& bis = BIS_MOUNT_ENTRIES[i];
|
||||||
|
|
||||||
|
RemoveDevice(bis.mount_name);
|
||||||
|
f_unmount(bis.mount_name);
|
||||||
|
fsStorageClose(&fat.storage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace sphaira::fatfs
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
|
||||||
|
const char* VolumeStr[] {
|
||||||
|
sphaira::fatfs::BIS_MOUNT_ENTRIES[0].volume_name,
|
||||||
|
sphaira::fatfs::BIS_MOUNT_ENTRIES[1].volume_name,
|
||||||
|
sphaira::fatfs::BIS_MOUNT_ENTRIES[2].volume_name,
|
||||||
|
sphaira::fatfs::BIS_MOUNT_ENTRIES[3].volume_name,
|
||||||
|
};
|
||||||
|
|
||||||
|
Result fatfs_read(u8 num, void* dst, u64 offset, u64 size) {
|
||||||
|
// log_write("[FAT] num: %u\n", num);
|
||||||
|
auto& fat = sphaira::fatfs::g_fat_storage[num];
|
||||||
|
return sphaira::fatfs::ReadFile(&fat.storage, fat.buffered, dst, offset, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // extern "C"
|
||||||
24
sphaira/source/ff16/LICENSE.txt
Normal file
24
sphaira/source/ff16/LICENSE.txt
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
FatFs License
|
||||||
|
|
||||||
|
FatFs has being developped as a personal project of the author, ChaN. It is free from the code anyone else wrote at current release. Following code block shows a copy of the FatFs license document that heading the source files.
|
||||||
|
|
||||||
|
/*----------------------------------------------------------------------------/
|
||||||
|
/ FatFs - Generic FAT Filesystem Module Rx.xx /
|
||||||
|
/-----------------------------------------------------------------------------/
|
||||||
|
/
|
||||||
|
/ Copyright (C) 20xx, ChaN, all right reserved.
|
||||||
|
/
|
||||||
|
/ FatFs module is an open source software. Redistribution and use of FatFs in
|
||||||
|
/ source and binary forms, with or without modification, are permitted provided
|
||||||
|
/ that the following condition is met:
|
||||||
|
/
|
||||||
|
/ 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
/ this condition and the following disclaimer.
|
||||||
|
/
|
||||||
|
/ This software is provided by the copyright holder and contributors "AS IS"
|
||||||
|
/ and any warranties related to this software are DISCLAIMED.
|
||||||
|
/ The copyright owner or contributors be NOT LIABLE for any damages caused
|
||||||
|
/ by use of this software.
|
||||||
|
/----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
Therefore FatFs license is one of the BSD-style licenses, but there is a significant feature. FatFs is mainly intended for embedded systems. In order to extend the usability for commercial products, the redistributions of FatFs in binary form, such as embedded code, binary library and any forms without source code, do not need to include about FatFs in the documentations. This is equivalent to the 1-clause BSD license. Of course FatFs is compatible with the most of open source software licenses include GNU GPL. When you redistribute the FatFs source code with changes or create a fork, the license can also be changed to GNU GPL, BSD-style license or any open source software license that not conflict with FatFs license.
|
||||||
74
sphaira/source/ff16/diskio.c
Normal file
74
sphaira/source/ff16/diskio.c
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* Low level disk I/O module SKELETON for FatFs (C)ChaN, 2019 */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* If a working storage control module is available, it should be */
|
||||||
|
/* attached to the FatFs via a glue function rather than modifying it. */
|
||||||
|
/* This is an example of glue functions to attach various exsisting */
|
||||||
|
/* storage control modules to the FatFs module with a defined API. */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#include "ff.h" /* Obtains integer types */
|
||||||
|
#include "diskio.h" /* Declarations of disk functions */
|
||||||
|
#include <switch.h>
|
||||||
|
|
||||||
|
extern Result fatfs_read(u8 num, void* dst, u64 offset, u64 size);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* Get Drive Status */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
DSTATUS disk_status (
|
||||||
|
BYTE pdrv /* Physical drive nmuber to identify the drive */
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return RES_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* Inidialize a Drive */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
DSTATUS disk_initialize (
|
||||||
|
BYTE pdrv /* Physical drive nmuber to identify the drive */
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return RES_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* Read Sector(s) */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
DRESULT disk_read (
|
||||||
|
BYTE pdrv, /* Physical drive nmuber to identify the drive */
|
||||||
|
BYTE *buff, /* Data buffer to store read data */
|
||||||
|
LBA_t sector, /* Start sector in LBA */
|
||||||
|
UINT count /* Number of sectors to read */
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (R_FAILED(fatfs_read(pdrv, buff, sector * FF_MAX_SS, count * FF_MAX_SS))) {
|
||||||
|
return RES_ERROR;
|
||||||
|
}
|
||||||
|
return RES_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
/* Miscellaneous Functions */
|
||||||
|
/*-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
DRESULT disk_ioctl (
|
||||||
|
BYTE pdrv, /* Physical drive nmuber (0..) */
|
||||||
|
BYTE cmd, /* Control code */
|
||||||
|
void *buff /* Buffer to send/receive control data */
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return RES_PARERR;
|
||||||
|
}
|
||||||
77
sphaira/source/ff16/diskio.h
Normal file
77
sphaira/source/ff16/diskio.h
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
/*-----------------------------------------------------------------------/
|
||||||
|
/ Low level disk interface modlue include file (C)ChaN, 2025 /
|
||||||
|
/-----------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#ifndef _DISKIO_DEFINED
|
||||||
|
#define _DISKIO_DEFINED
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Status of Disk Functions */
|
||||||
|
typedef BYTE DSTATUS;
|
||||||
|
|
||||||
|
/* Results of Disk Functions */
|
||||||
|
typedef enum {
|
||||||
|
RES_OK = 0, /* 0: Successful */
|
||||||
|
RES_ERROR, /* 1: R/W Error */
|
||||||
|
RES_WRPRT, /* 2: Write Protected */
|
||||||
|
RES_NOTRDY, /* 3: Not Ready */
|
||||||
|
RES_PARERR /* 4: Invalid Parameter */
|
||||||
|
} DRESULT;
|
||||||
|
|
||||||
|
|
||||||
|
/*---------------------------------------*/
|
||||||
|
/* Prototypes for disk control functions */
|
||||||
|
|
||||||
|
|
||||||
|
DSTATUS disk_initialize (BYTE pdrv);
|
||||||
|
DSTATUS disk_status (BYTE pdrv);
|
||||||
|
DRESULT disk_read (BYTE pdrv, BYTE* buff, LBA_t sector, UINT count);
|
||||||
|
DRESULT disk_write (BYTE pdrv, const BYTE* buff, LBA_t sector, UINT count);
|
||||||
|
DRESULT disk_ioctl (BYTE pdrv, BYTE cmd, void* buff);
|
||||||
|
|
||||||
|
|
||||||
|
/* Disk Status Bits (DSTATUS) */
|
||||||
|
|
||||||
|
#define STA_NOINIT 0x01 /* Drive not initialized */
|
||||||
|
#define STA_NODISK 0x02 /* No medium in the drive */
|
||||||
|
#define STA_PROTECT 0x04 /* Write protected */
|
||||||
|
|
||||||
|
|
||||||
|
/* Command code for disk_ioctrl fucntion */
|
||||||
|
|
||||||
|
/* Generic command (Used by FatFs) */
|
||||||
|
#define CTRL_SYNC 0 /* Complete pending write process (needed at FF_FS_READONLY == 0) */
|
||||||
|
#define GET_SECTOR_COUNT 1 /* Get media size (needed at FF_USE_MKFS == 1) */
|
||||||
|
#define GET_SECTOR_SIZE 2 /* Get sector size (needed at FF_MAX_SS != FF_MIN_SS) */
|
||||||
|
#define GET_BLOCK_SIZE 3 /* Get erase block size (needed at FF_USE_MKFS == 1) */
|
||||||
|
#define CTRL_TRIM 4 /* Inform device that the data on the block of sectors is no longer used (needed at FF_USE_TRIM == 1) */
|
||||||
|
|
||||||
|
/* Generic command (Not used by FatFs) */
|
||||||
|
#define CTRL_POWER 5 /* Get/Set power status */
|
||||||
|
#define CTRL_LOCK 6 /* Lock/Unlock media removal */
|
||||||
|
#define CTRL_EJECT 7 /* Eject media */
|
||||||
|
#define CTRL_FORMAT 8 /* Create physical format on the media */
|
||||||
|
|
||||||
|
/* MMC/SDC specific ioctl command (Not used by FatFs) */
|
||||||
|
#define MMC_GET_TYPE 10 /* Get card type */
|
||||||
|
#define MMC_GET_CSD 11 /* Get CSD */
|
||||||
|
#define MMC_GET_CID 12 /* Get CID */
|
||||||
|
#define MMC_GET_OCR 13 /* Get OCR */
|
||||||
|
#define MMC_GET_SDSTAT 14 /* Get SD status */
|
||||||
|
#define ISDIO_READ 55 /* Read data form SD iSDIO register */
|
||||||
|
#define ISDIO_WRITE 56 /* Write data to SD iSDIO register */
|
||||||
|
#define ISDIO_MRITE 57 /* Masked write data to SD iSDIO register */
|
||||||
|
|
||||||
|
/* ATA/CF specific ioctl command (Not used by FatFs) */
|
||||||
|
#define ATA_GET_REV 20 /* Get F/W revision */
|
||||||
|
#define ATA_GET_MODEL 21 /* Get model name */
|
||||||
|
#define ATA_GET_SN 22 /* Get serial number */
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
7248
sphaira/source/ff16/ff.c
Normal file
7248
sphaira/source/ff16/ff.c
Normal file
File diff suppressed because it is too large
Load Diff
448
sphaira/source/ff16/ff.h
Normal file
448
sphaira/source/ff16/ff.h
Normal file
@@ -0,0 +1,448 @@
|
|||||||
|
/*----------------------------------------------------------------------------/
|
||||||
|
/ FatFs - Generic FAT Filesystem module R0.16 /
|
||||||
|
/-----------------------------------------------------------------------------/
|
||||||
|
/
|
||||||
|
/ Copyright (C) 2025, ChaN, all right reserved.
|
||||||
|
/
|
||||||
|
/ FatFs module is an open source software. Redistribution and use of FatFs in
|
||||||
|
/ source and binary forms, with or without modification, are permitted provided
|
||||||
|
/ that the following condition is met:
|
||||||
|
|
||||||
|
/ 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
/ this condition and the following disclaimer.
|
||||||
|
/
|
||||||
|
/ This software is provided by the copyright holder and contributors "AS IS"
|
||||||
|
/ and any warranties related to this software are DISCLAIMED.
|
||||||
|
/ The copyright owner or contributors be NOT LIABLE for any damages caused
|
||||||
|
/ by use of this software.
|
||||||
|
/
|
||||||
|
/----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef FF_DEFINED
|
||||||
|
#define FF_DEFINED 80386 /* Revision ID */
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(FFCONF_DEF)
|
||||||
|
#include "ffconf.h" /* FatFs configuration options */
|
||||||
|
#endif
|
||||||
|
#if FF_DEFINED != FFCONF_DEF
|
||||||
|
#error Wrong configuration file (ffconf.h).
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* Integer types used for FatFs API */
|
||||||
|
|
||||||
|
#if defined(_WIN32) /* Windows VC++ (for development only) */
|
||||||
|
#define FF_INTDEF 2
|
||||||
|
#include <windows.h>
|
||||||
|
typedef unsigned __int64 QWORD;
|
||||||
|
#include <float.h>
|
||||||
|
#define isnan(v) _isnan(v)
|
||||||
|
#define isinf(v) (!_finite(v))
|
||||||
|
|
||||||
|
#elif (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__cplusplus) /* C99 or later */
|
||||||
|
#define FF_INTDEF 2
|
||||||
|
#include <stdint.h>
|
||||||
|
typedef unsigned int UINT; /* int must be 16-bit or 32-bit */
|
||||||
|
typedef unsigned char BYTE; /* char must be 8-bit */
|
||||||
|
typedef uint16_t WORD; /* 16-bit unsigned */
|
||||||
|
typedef uint32_t DWORD; /* 32-bit unsigned */
|
||||||
|
typedef uint64_t QWORD; /* 64-bit unsigned */
|
||||||
|
typedef WORD WCHAR; /* UTF-16 code unit */
|
||||||
|
|
||||||
|
#else /* Earlier than C99 */
|
||||||
|
#define FF_INTDEF 1
|
||||||
|
typedef unsigned int UINT; /* int must be 16-bit or 32-bit */
|
||||||
|
typedef unsigned char BYTE; /* char must be 8-bit */
|
||||||
|
typedef unsigned short WORD; /* short must be 16-bit */
|
||||||
|
typedef unsigned long DWORD; /* long must be 32-bit */
|
||||||
|
typedef WORD WCHAR; /* UTF-16 code unit */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* Type of file size and LBA variables */
|
||||||
|
|
||||||
|
#if FF_FS_EXFAT
|
||||||
|
#if FF_INTDEF != 2
|
||||||
|
#error exFAT feature wants C99 or later
|
||||||
|
#endif
|
||||||
|
typedef QWORD FSIZE_t;
|
||||||
|
#if FF_LBA64
|
||||||
|
typedef QWORD LBA_t;
|
||||||
|
#else
|
||||||
|
typedef DWORD LBA_t;
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
#if FF_LBA64
|
||||||
|
#error exFAT needs to be enabled when enable 64-bit LBA
|
||||||
|
#endif
|
||||||
|
typedef DWORD FSIZE_t;
|
||||||
|
typedef DWORD LBA_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Type of path name strings on FatFs API (TCHAR) */
|
||||||
|
|
||||||
|
#if FF_USE_LFN && FF_LFN_UNICODE == 1 /* Unicode in UTF-16 encoding */
|
||||||
|
typedef WCHAR TCHAR;
|
||||||
|
#define _T(x) L ## x
|
||||||
|
#define _TEXT(x) L ## x
|
||||||
|
#elif FF_USE_LFN && FF_LFN_UNICODE == 2 /* Unicode in UTF-8 encoding */
|
||||||
|
typedef char TCHAR;
|
||||||
|
#define _T(x) u8 ## x
|
||||||
|
#define _TEXT(x) u8 ## x
|
||||||
|
#elif FF_USE_LFN && FF_LFN_UNICODE == 3 /* Unicode in UTF-32 encoding */
|
||||||
|
typedef DWORD TCHAR;
|
||||||
|
#define _T(x) U ## x
|
||||||
|
#define _TEXT(x) U ## x
|
||||||
|
#elif FF_USE_LFN && (FF_LFN_UNICODE < 0 || FF_LFN_UNICODE > 3)
|
||||||
|
#error Wrong FF_LFN_UNICODE setting
|
||||||
|
#else /* ANSI/OEM code in SBCS/DBCS */
|
||||||
|
typedef char TCHAR;
|
||||||
|
#define _T(x) x
|
||||||
|
#define _TEXT(x) x
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Definitions of volume management */
|
||||||
|
|
||||||
|
#if FF_MULTI_PARTITION /* Multiple partition configuration */
|
||||||
|
typedef struct {
|
||||||
|
BYTE pd; /* Associated physical drive */
|
||||||
|
BYTE pt; /* Associated partition (0:Auto detect, 1-4:Forced partition) */
|
||||||
|
} PARTITION;
|
||||||
|
extern PARTITION VolToPart[]; /* Volume to partition mapping table */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if FF_STR_VOLUME_ID
|
||||||
|
#ifndef FF_VOLUME_STRS
|
||||||
|
extern const char* VolumeStr[FF_VOLUMES]; /* User defined volume ID table */
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* Current working directory structure (FFXCWDS) */
|
||||||
|
|
||||||
|
#if FF_FS_EXFAT && FF_FS_RPATH
|
||||||
|
#if FF_PATH_DEPTH < 1
|
||||||
|
#error FF_PATH_DEPTH must not be zero
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
DWORD d_scl; /* Directory start cluster (0:root dir) */
|
||||||
|
DWORD d_size; /* Size of directory (b7-b0: cluster chain status) (invalid if d_scl == 0) */
|
||||||
|
DWORD nxt_ofs; /* Offset of entry of next dir in this directory (invalid if last link) */
|
||||||
|
} FFXCWDL;
|
||||||
|
typedef struct {
|
||||||
|
UINT depth; /* Current directory depth (0:root dir) */
|
||||||
|
FFXCWDL tbl[FF_PATH_DEPTH + 1]; /* Directory chain of current working directory path */
|
||||||
|
} FFXCWDS;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* Filesystem object structure (FATFS) */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
BYTE fs_type; /* Filesystem type (0:not mounted) */
|
||||||
|
BYTE pdrv; /* Physical drive that holds this volume */
|
||||||
|
BYTE ldrv; /* Logical drive number (used only when FF_FS_REENTRANT) */
|
||||||
|
BYTE n_fats; /* Number of FATs (1 or 2) */
|
||||||
|
BYTE wflag; /* win[] status (b0:dirty) */
|
||||||
|
BYTE fsi_flag; /* Allocation information control (b7:disabled, b0:dirty) */
|
||||||
|
WORD id; /* Volume mount ID */
|
||||||
|
WORD n_rootdir; /* Number of root directory entries (FAT12/16) */
|
||||||
|
WORD csize; /* Cluster size [sectors] */
|
||||||
|
#if FF_MAX_SS != FF_MIN_SS
|
||||||
|
WORD ssize; /* Sector size (512, 1024, 2048 or 4096) */
|
||||||
|
#endif
|
||||||
|
#if FF_USE_LFN
|
||||||
|
WCHAR* lfnbuf; /* Pointer to LFN working buffer */
|
||||||
|
#endif
|
||||||
|
#if !FF_FS_READONLY
|
||||||
|
DWORD last_clst; /* Last allocated cluster (invalid if >=n_fatent) */
|
||||||
|
DWORD free_clst; /* Number of free clusters (invalid if >=fs->n_fatent-2) */
|
||||||
|
#endif
|
||||||
|
#if FF_FS_RPATH
|
||||||
|
DWORD cdir; /* Current directory start cluster (0:root) */
|
||||||
|
#endif
|
||||||
|
DWORD n_fatent; /* Number of FAT entries (number of clusters + 2) */
|
||||||
|
DWORD fsize; /* Number of sectors per FAT */
|
||||||
|
LBA_t winsect; /* Current sector appearing in the win[] */
|
||||||
|
LBA_t volbase; /* Volume base sector */
|
||||||
|
LBA_t fatbase; /* FAT base sector */
|
||||||
|
LBA_t dirbase; /* Root directory base sector (FAT12/16) or cluster (FAT32/exFAT) */
|
||||||
|
LBA_t database; /* Data base sector */
|
||||||
|
#if FF_FS_EXFAT
|
||||||
|
LBA_t bitbase; /* Allocation bitmap base sector */
|
||||||
|
BYTE* dirbuf; /* Pointer to directory entry block buffer */
|
||||||
|
#if FF_FS_RPATH
|
||||||
|
FFXCWDS xcwds; /* Crrent working directory structure */
|
||||||
|
FFXCWDS xcwds2; /* Working buffer to follow the path */
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
BYTE win[FF_MAX_SS]; /* Disk access window for directory, FAT (and file data in tiny cfg) */
|
||||||
|
} FATFS;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Object ID and allocation information (FFOBJID) */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
FATFS* fs; /* Pointer to the volume holding this object */
|
||||||
|
WORD id; /* Volume mount ID when this object was opened */
|
||||||
|
BYTE attr; /* Object attribute */
|
||||||
|
BYTE stat; /* Object chain status (exFAT: b1-0: =0:not contiguous, =2:contiguous, =3:fragmented in this session, b2:sub-directory stretched) */
|
||||||
|
DWORD sclust; /* Object data cluster (0:no data or root directory) */
|
||||||
|
FSIZE_t objsize; /* Object size (valid when sclust != 0) */
|
||||||
|
#if FF_FS_EXFAT
|
||||||
|
DWORD n_cont; /* Size of first fragment - 1 (valid when stat == 3) */
|
||||||
|
DWORD n_frag; /* Size of last fragment needs to be written to FAT (valid when not zero) */
|
||||||
|
DWORD c_scl; /* Cluster of directory holding this object (valid when sclust != 0) */
|
||||||
|
DWORD c_size; /* Size of directory holding this object (b7-b0: allocation status, valid when c_scl != 0) */
|
||||||
|
DWORD c_ofs; /* Offset of entry in the holding directory */
|
||||||
|
#endif
|
||||||
|
#if FF_FS_LOCK
|
||||||
|
UINT lockid; /* File lock ID origin from 1 (index of file semaphore table Files[]) */
|
||||||
|
#endif
|
||||||
|
} FFOBJID;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* File object structure (FIL) */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
FFOBJID obj; /* Object identifier (must be the 1st member to detect invalid object pointer) */
|
||||||
|
BYTE flag; /* File status flags */
|
||||||
|
BYTE err; /* Abort flag (error code) */
|
||||||
|
FSIZE_t fptr; /* File read/write pointer (0 on open) */
|
||||||
|
DWORD clust; /* Current cluster of fptr (invalid when fptr is 0) */
|
||||||
|
LBA_t sect; /* Sector number appearing in buf[] (0:invalid) */
|
||||||
|
#if !FF_FS_READONLY
|
||||||
|
LBA_t dir_sect; /* Sector number containing the directory entry (not used in exFAT) */
|
||||||
|
BYTE* dir_ptr; /* Pointer to the directory entry in the win[] (not used in exFAT) */
|
||||||
|
#endif
|
||||||
|
#if FF_USE_FASTSEEK
|
||||||
|
DWORD* cltbl; /* Pointer to the cluster link map table (nulled on open; set by application) */
|
||||||
|
#endif
|
||||||
|
#if !FF_FS_TINY
|
||||||
|
BYTE buf[FF_MAX_SS]; /* File private data read/write window */
|
||||||
|
#endif
|
||||||
|
} FIL;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Directory object structure (FDIR) */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
FFOBJID obj; /* Object identifier (must be the 1st member to detect invalid object pointer) */
|
||||||
|
DWORD dptr; /* Current read/write offset */
|
||||||
|
DWORD clust; /* Current cluster */
|
||||||
|
LBA_t sect; /* Current sector (0:no more item to read) */
|
||||||
|
BYTE* dir; /* Pointer to the directory item in the win[] in filesystem object */
|
||||||
|
BYTE fn[12]; /* SFN (in/out) {body[0-7],ext[8-10],status[11]} */
|
||||||
|
#if FF_USE_LFN
|
||||||
|
DWORD blk_ofs; /* Offset of current entry block being processed (0xFFFFFFFF:invalid) */
|
||||||
|
#endif
|
||||||
|
#if FF_USE_FIND
|
||||||
|
const TCHAR *pat; /* Pointer to the name matching pattern */
|
||||||
|
#endif
|
||||||
|
} FDIR;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* File/directory information structure (FILINFO) */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
FSIZE_t fsize; /* File size (invalid for directory) */
|
||||||
|
WORD fdate; /* Date of file modification or directory creation */
|
||||||
|
WORD ftime; /* Time of file modification or directory creation */
|
||||||
|
#if FF_FS_CRTIME
|
||||||
|
WORD crdate; /* Date of object createion */
|
||||||
|
WORD crtime; /* Time of object createion */
|
||||||
|
#endif
|
||||||
|
BYTE fattrib; /* Object attribute */
|
||||||
|
#if FF_USE_LFN
|
||||||
|
TCHAR altname[FF_SFN_BUF + 1];/* Alternative object name */
|
||||||
|
TCHAR fname[FF_LFN_BUF + 1]; /* Primary object name */
|
||||||
|
#else
|
||||||
|
TCHAR fname[12 + 1]; /* Object name */
|
||||||
|
#endif
|
||||||
|
} FILINFO;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Format parameter structure (MKFS_PARM) used for f_mkfs() */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
BYTE fmt; /* Format option (FM_FAT, FM_FAT32, FM_EXFAT and FM_SFD) */
|
||||||
|
BYTE n_fat; /* Number of FATs */
|
||||||
|
UINT align; /* Data area alignment (sector) */
|
||||||
|
UINT n_root; /* Number of root directory entries */
|
||||||
|
DWORD au_size; /* Cluster size (byte) */
|
||||||
|
} MKFS_PARM;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* File function return code (FRESULT) */
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
FR_OK = 0, /* (0) Function succeeded */
|
||||||
|
FR_DISK_ERR, /* (1) A hard error occurred in the low level disk I/O layer */
|
||||||
|
FR_INT_ERR, /* (2) Assertion failed */
|
||||||
|
FR_NOT_READY, /* (3) The physical drive does not work */
|
||||||
|
FR_NO_FILE, /* (4) Could not find the file */
|
||||||
|
FR_NO_PATH, /* (5) Could not find the path */
|
||||||
|
FR_INVALID_NAME, /* (6) The path name format is invalid */
|
||||||
|
FR_DENIED, /* (7) Access denied due to a prohibited access or directory full */
|
||||||
|
FR_EXIST, /* (8) Access denied due to a prohibited access */
|
||||||
|
FR_INVALID_OBJECT, /* (9) The file/directory object is invalid */
|
||||||
|
FR_WRITE_PROTECTED, /* (10) The physical drive is write protected */
|
||||||
|
FR_INVALID_DRIVE, /* (11) The logical drive number is invalid */
|
||||||
|
FR_NOT_ENABLED, /* (12) The volume has no work area */
|
||||||
|
FR_NO_FILESYSTEM, /* (13) Could not find a valid FAT volume */
|
||||||
|
FR_MKFS_ABORTED, /* (14) The f_mkfs function aborted due to some problem */
|
||||||
|
FR_TIMEOUT, /* (15) Could not take control of the volume within defined period */
|
||||||
|
FR_LOCKED, /* (16) The operation is rejected according to the file sharing policy */
|
||||||
|
FR_NOT_ENOUGH_CORE, /* (17) LFN working buffer could not be allocated, given buffer size is insufficient or too deep path */
|
||||||
|
FR_TOO_MANY_OPEN_FILES, /* (18) Number of open files > FF_FS_LOCK */
|
||||||
|
FR_INVALID_PARAMETER /* (19) Given parameter is invalid */
|
||||||
|
} FRESULT;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
/* FatFs Module Application Interface */
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
|
||||||
|
FRESULT f_open (FIL* fp, const TCHAR* path, BYTE mode); /* Open or create a file */
|
||||||
|
FRESULT f_close (FIL* fp); /* Close an open file object */
|
||||||
|
FRESULT f_read (FIL* fp, void* buff, UINT btr, UINT* br); /* Read data from the file */
|
||||||
|
FRESULT f_write (FIL* fp, const void* buff, UINT btw, UINT* bw); /* Write data to the file */
|
||||||
|
FRESULT f_lseek (FIL* fp, FSIZE_t ofs); /* Move file pointer of the file object */
|
||||||
|
FRESULT f_truncate (FIL* fp); /* Truncate the file */
|
||||||
|
FRESULT f_sync (FIL* fp); /* Flush cached data of the writing file */
|
||||||
|
FRESULT f_opendir (FDIR* dp, const TCHAR* path); /* Open a directory */
|
||||||
|
FRESULT f_closedir (FDIR* dp); /* Close an open directory */
|
||||||
|
FRESULT f_readdir (FDIR* dp, FILINFO* fno); /* Read a directory item */
|
||||||
|
FRESULT f_findfirst (FDIR* dp, FILINFO* fno, const TCHAR* path, const TCHAR* pattern); /* Find first file */
|
||||||
|
FRESULT f_findnext (FDIR* dp, FILINFO* fno); /* Find next file */
|
||||||
|
FRESULT f_mkdir (const TCHAR* path); /* Create a sub directory */
|
||||||
|
FRESULT f_unlink (const TCHAR* path); /* Delete an existing file or directory */
|
||||||
|
FRESULT f_rename (const TCHAR* path_old, const TCHAR* path_new); /* Rename/Move a file or directory */
|
||||||
|
FRESULT f_stat (const TCHAR* path, FILINFO* fno); /* Get file status */
|
||||||
|
FRESULT f_chmod (const TCHAR* path, BYTE attr, BYTE mask); /* Change attribute of a file/dir */
|
||||||
|
FRESULT f_utime (const TCHAR* path, const FILINFO* fno); /* Change timestamp of a file/dir */
|
||||||
|
FRESULT f_chdir (const TCHAR* path); /* Change current directory */
|
||||||
|
FRESULT f_chdrive (const TCHAR* path); /* Change current drive */
|
||||||
|
FRESULT f_getcwd (TCHAR* buff, UINT len); /* Get current directory */
|
||||||
|
FRESULT f_getfree (const TCHAR* path, DWORD* nclst, FATFS** fatfs); /* Get number of free clusters on the drive */
|
||||||
|
FRESULT f_getlabel (const TCHAR* path, TCHAR* label, DWORD* vsn); /* Get volume label */
|
||||||
|
FRESULT f_setlabel (const TCHAR* label); /* Set volume label */
|
||||||
|
FRESULT f_forward (FIL* fp, UINT(*func)(const BYTE*,UINT), UINT btf, UINT* bf); /* Forward data to the stream */
|
||||||
|
FRESULT f_expand (FIL* fp, FSIZE_t fsz, BYTE opt); /* Allocate a contiguous block to the file */
|
||||||
|
FRESULT f_mount (FATFS* fs, const TCHAR* path, BYTE opt); /* Mount/Unmount a logical drive */
|
||||||
|
FRESULT f_mkfs (const TCHAR* path, const MKFS_PARM* opt, void* work, UINT len); /* Create a FAT volume */
|
||||||
|
FRESULT f_fdisk (BYTE pdrv, const LBA_t ptbl[], void* work); /* Divide a physical drive into some partitions */
|
||||||
|
FRESULT f_setcp (WORD cp); /* Set current code page */
|
||||||
|
int f_putc (TCHAR c, FIL* fp); /* Put a character to the file */
|
||||||
|
int f_puts (const TCHAR* str, FIL* cp); /* Put a string to the file */
|
||||||
|
int f_printf (FIL* fp, const TCHAR* str, ...); /* Put a formatted string to the file */
|
||||||
|
TCHAR* f_gets (TCHAR* buff, int len, FIL* fp); /* Get a string from the file */
|
||||||
|
|
||||||
|
/* Some API fucntions are implemented as macro */
|
||||||
|
|
||||||
|
#define f_eof(fp) ((int)((fp)->fptr == (fp)->obj.objsize))
|
||||||
|
#define f_error(fp) ((fp)->err)
|
||||||
|
#define f_tell(fp) ((fp)->fptr)
|
||||||
|
#define f_size(fp) ((fp)->obj.objsize)
|
||||||
|
#define f_rewind(fp) f_lseek((fp), 0)
|
||||||
|
#define f_rewinddir(dp) f_readdir((dp), 0)
|
||||||
|
#define f_rmdir(path) f_unlink(path)
|
||||||
|
#define f_unmount(path) f_mount(0, path, 0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
/* Additional Functions */
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* RTC function (provided by user) */
|
||||||
|
#if !FF_FS_READONLY && !FF_FS_NORTC
|
||||||
|
DWORD get_fattime (void); /* Get current time */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* LFN support functions (defined in ffunicode.c) */
|
||||||
|
|
||||||
|
#if FF_USE_LFN >= 1
|
||||||
|
WCHAR ff_oem2uni (WCHAR oem, WORD cp); /* OEM code to Unicode conversion */
|
||||||
|
WCHAR ff_uni2oem (DWORD uni, WORD cp); /* Unicode to OEM code conversion */
|
||||||
|
DWORD ff_wtoupper (DWORD uni); /* Unicode upper-case conversion */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* O/S dependent functions (samples available in ffsystem.c) */
|
||||||
|
|
||||||
|
#if FF_USE_LFN == 3 /* Dynamic memory allocation */
|
||||||
|
void* ff_memalloc (UINT msize); /* Allocate memory block */
|
||||||
|
void ff_memfree (void* mblock); /* Free memory block */
|
||||||
|
#endif
|
||||||
|
#if FF_FS_REENTRANT /* Sync functions */
|
||||||
|
int ff_mutex_create (int vol); /* Create a sync object */
|
||||||
|
void ff_mutex_delete (int vol); /* Delete a sync object */
|
||||||
|
int ff_mutex_take (int vol); /* Lock sync object */
|
||||||
|
void ff_mutex_give (int vol); /* Unlock sync object */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
/* Flags and Offset Address */
|
||||||
|
/*--------------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* File access mode and open method flags (3rd argument of f_open function) */
|
||||||
|
#define FA_READ 0x01
|
||||||
|
#define FA_WRITE 0x02
|
||||||
|
#define FA_OPEN_EXISTING 0x00
|
||||||
|
#define FA_CREATE_NEW 0x04
|
||||||
|
#define FA_CREATE_ALWAYS 0x08
|
||||||
|
#define FA_OPEN_ALWAYS 0x10
|
||||||
|
#define FA_OPEN_APPEND 0x30
|
||||||
|
|
||||||
|
/* Fast seek controls (2nd argument of f_lseek function) */
|
||||||
|
#define CREATE_LINKMAP ((FSIZE_t)0 - 1)
|
||||||
|
|
||||||
|
/* Format options (2nd argument of f_mkfs function) */
|
||||||
|
#define FM_FAT 0x01
|
||||||
|
#define FM_FAT32 0x02
|
||||||
|
#define FM_EXFAT 0x04
|
||||||
|
#define FM_ANY 0x07
|
||||||
|
#define FM_SFD 0x08
|
||||||
|
|
||||||
|
/* Filesystem type (FATFS.fs_type) */
|
||||||
|
#define FS_FAT12 1
|
||||||
|
#define FS_FAT16 2
|
||||||
|
#define FS_FAT32 3
|
||||||
|
#define FS_EXFAT 4
|
||||||
|
|
||||||
|
/* File attribute bits for directory entry (FILINFO.fattrib) */
|
||||||
|
#define AM_RDO 0x01 /* Read only */
|
||||||
|
#define AM_HID 0x02 /* Hidden */
|
||||||
|
#define AM_SYS 0x04 /* System */
|
||||||
|
#define AM_DIR 0x10 /* Directory */
|
||||||
|
#define AM_ARC 0x20 /* Archive */
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* FF_DEFINED */
|
||||||
313
sphaira/source/ff16/ffconf.h
Normal file
313
sphaira/source/ff16/ffconf.h
Normal file
@@ -0,0 +1,313 @@
|
|||||||
|
/*---------------------------------------------------------------------------/
|
||||||
|
/ Configurations of FatFs Module
|
||||||
|
/---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define FFCONF_DEF 80386 /* Revision ID */
|
||||||
|
|
||||||
|
/*---------------------------------------------------------------------------/
|
||||||
|
/ Function Configurations
|
||||||
|
/---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define FF_FS_READONLY 1
|
||||||
|
/* This option switches read-only configuration. (0:Read/Write or 1:Read-only)
|
||||||
|
/ Read-only configuration removes writing API functions, f_write(), f_sync(),
|
||||||
|
/ f_unlink(), f_mkdir(), f_chmod(), f_rename(), f_truncate(), f_getfree()
|
||||||
|
/ and optional writing functions as well. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_MINIMIZE 0
|
||||||
|
/* This option defines minimization level to remove some basic API functions.
|
||||||
|
/
|
||||||
|
/ 0: Basic functions are fully enabled.
|
||||||
|
/ 1: f_stat(), f_getfree(), f_unlink(), f_mkdir(), f_truncate() and f_rename()
|
||||||
|
/ are removed.
|
||||||
|
/ 2: f_opendir(), f_readdir() and f_closedir() are removed in addition to 1.
|
||||||
|
/ 3: f_lseek() function is removed in addition to 2. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_FIND 0
|
||||||
|
/* This option switches filtered directory read functions, f_findfirst() and
|
||||||
|
/ f_findnext(). (0:Disable, 1:Enable 2:Enable with matching altname[] too) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_MKFS 0
|
||||||
|
/* This option switches f_mkfs(). (0:Disable or 1:Enable) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_FASTSEEK 0
|
||||||
|
/* This option switches fast seek feature. (0:Disable or 1:Enable) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_EXPAND 0
|
||||||
|
/* This option switches f_expand(). (0:Disable or 1:Enable) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_CHMOD 0
|
||||||
|
/* This option switches attribute control API functions, f_chmod() and f_utime().
|
||||||
|
/ (0:Disable or 1:Enable) Also FF_FS_READONLY needs to be 0 to enable this option. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_LABEL 0
|
||||||
|
/* This option switches volume label API functions, f_getlabel() and f_setlabel().
|
||||||
|
/ (0:Disable or 1:Enable) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_FORWARD 0
|
||||||
|
/* This option switches f_forward(). (0:Disable or 1:Enable) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_STRFUNC 0
|
||||||
|
#define FF_PRINT_LLI 0
|
||||||
|
#define FF_PRINT_FLOAT 0
|
||||||
|
#define FF_STRF_ENCODE 0
|
||||||
|
/* FF_USE_STRFUNC switches string API functions, f_gets(), f_putc(), f_puts() and
|
||||||
|
/ f_printf().
|
||||||
|
/
|
||||||
|
/ 0: Disable. FF_PRINT_LLI, FF_PRINT_FLOAT and FF_STRF_ENCODE have no effect.
|
||||||
|
/ 1: Enable without LF-CRLF conversion.
|
||||||
|
/ 2: Enable with LF-CRLF conversion.
|
||||||
|
/
|
||||||
|
/ FF_PRINT_LLI = 1 makes f_printf() support long long argument and FF_PRINT_FLOAT = 1/2
|
||||||
|
/ makes f_printf() support floating point argument. These features want C99 or later.
|
||||||
|
/ When FF_LFN_UNICODE >= 1 with LFN enabled, string API functions convert the character
|
||||||
|
/ encoding in it. FF_STRF_ENCODE selects assumption of character encoding ON THE FILE
|
||||||
|
/ to be read/written via those functions.
|
||||||
|
/
|
||||||
|
/ 0: ANSI/OEM in current CP
|
||||||
|
/ 1: Unicode in UTF-16LE
|
||||||
|
/ 2: Unicode in UTF-16BE
|
||||||
|
/ 3: Unicode in UTF-8
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*---------------------------------------------------------------------------/
|
||||||
|
/ Locale and Namespace Configurations
|
||||||
|
/---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define FF_CODE_PAGE 437
|
||||||
|
/* This option specifies the OEM code page to be used on the target system.
|
||||||
|
/ Incorrect code page setting can cause a file open failure.
|
||||||
|
/
|
||||||
|
/ 437 - U.S.
|
||||||
|
/ 720 - Arabic
|
||||||
|
/ 737 - Greek
|
||||||
|
/ 771 - KBL
|
||||||
|
/ 775 - Baltic
|
||||||
|
/ 850 - Latin 1
|
||||||
|
/ 852 - Latin 2
|
||||||
|
/ 855 - Cyrillic
|
||||||
|
/ 857 - Turkish
|
||||||
|
/ 860 - Portuguese
|
||||||
|
/ 861 - Icelandic
|
||||||
|
/ 862 - Hebrew
|
||||||
|
/ 863 - Canadian French
|
||||||
|
/ 864 - Arabic
|
||||||
|
/ 865 - Nordic
|
||||||
|
/ 866 - Russian
|
||||||
|
/ 869 - Greek 2
|
||||||
|
/ 932 - Japanese (DBCS)
|
||||||
|
/ 936 - Simplified Chinese (DBCS)
|
||||||
|
/ 949 - Korean (DBCS)
|
||||||
|
/ 950 - Traditional Chinese (DBCS)
|
||||||
|
/ 0 - Include all code pages above and configured by f_setcp()
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_LFN 3
|
||||||
|
#define FF_MAX_LFN 255
|
||||||
|
/* The FF_USE_LFN switches the support for LFN (long file name).
|
||||||
|
/
|
||||||
|
/ 0: Disable LFN. FF_MAX_LFN has no effect.
|
||||||
|
/ 1: Enable LFN with static working buffer on the BSS. Always NOT thread-safe.
|
||||||
|
/ 2: Enable LFN with dynamic working buffer on the STACK.
|
||||||
|
/ 3: Enable LFN with dynamic working buffer on the HEAP.
|
||||||
|
/
|
||||||
|
/ To enable the LFN, ffunicode.c needs to be added to the project. The LFN feature
|
||||||
|
/ requiers certain internal working buffer occupies (FF_MAX_LFN + 1) * 2 bytes and
|
||||||
|
/ additional (FF_MAX_LFN + 44) / 15 * 32 bytes when exFAT is enabled.
|
||||||
|
/ The FF_MAX_LFN defines size of the working buffer in UTF-16 code unit and it can
|
||||||
|
/ be in range of 12 to 255. It is recommended to be set 255 to fully support the LFN
|
||||||
|
/ specification.
|
||||||
|
/ When use stack for the working buffer, take care on stack overflow. When use heap
|
||||||
|
/ memory for the working buffer, memory management functions, ff_memalloc() and
|
||||||
|
/ ff_memfree() exemplified in ffsystem.c, need to be added to the project. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_LFN_UNICODE 2
|
||||||
|
/* This option switches the character encoding on the API when LFN is enabled.
|
||||||
|
/
|
||||||
|
/ 0: ANSI/OEM in current CP (TCHAR = char)
|
||||||
|
/ 1: Unicode in UTF-16 (TCHAR = WCHAR)
|
||||||
|
/ 2: Unicode in UTF-8 (TCHAR = char)
|
||||||
|
/ 3: Unicode in UTF-32 (TCHAR = DWORD)
|
||||||
|
/
|
||||||
|
/ Also behavior of string I/O functions will be affected by this option.
|
||||||
|
/ When LFN is not enabled, this option has no effect. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_LFN_BUF 255
|
||||||
|
#define FF_SFN_BUF 12
|
||||||
|
/* This set of options defines size of file name members in the FILINFO structure
|
||||||
|
/ which is used to read out directory items. These values should be suffcient for
|
||||||
|
/ the file names to read. The maximum possible length of the read file name depends
|
||||||
|
/ on character encoding. When LFN is not enabled, these options have no effect. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_RPATH 0
|
||||||
|
/* This option configures support for relative path feature.
|
||||||
|
/
|
||||||
|
/ 0: Disable relative path and remove related API functions.
|
||||||
|
/ 1: Enable relative path and dot names. f_chdir() and f_chdrive() are available.
|
||||||
|
/ 2: f_getcwd() is available in addition to 1.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_PATH_DEPTH 10
|
||||||
|
/* This option defines maximum depth of directory in the exFAT volume. It is NOT
|
||||||
|
/ relevant to FAT/FAT32 volume.
|
||||||
|
/ For example, FF_PATH_DEPTH = 3 will able to follow a path "/dir1/dir2/dir3/file"
|
||||||
|
/ but a sub-directory in the dir3 will not able to be followed and set current
|
||||||
|
/ directory.
|
||||||
|
/ The size of filesystem object (FATFS) increases FF_PATH_DEPTH * 24 bytes.
|
||||||
|
/ When FF_FS_EXFAT == 0 or FF_FS_RPATH == 0, this option has no effect.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*---------------------------------------------------------------------------/
|
||||||
|
/ Drive/Volume Configurations
|
||||||
|
/---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define FF_VOLUMES 4
|
||||||
|
/* Number of volumes (logical drives) to be used. (1-10) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_STR_VOLUME_ID 1
|
||||||
|
// #define FF_VOLUME_STRS "SAFE","USER","SYSTEM"
|
||||||
|
/* FF_STR_VOLUME_ID switches support for volume ID in arbitrary strings.
|
||||||
|
/ When FF_STR_VOLUME_ID is set to 1 or 2, arbitrary strings can be used as drive
|
||||||
|
/ number in the path name. FF_VOLUME_STRS defines the volume ID strings for each
|
||||||
|
/ logical drive. Number of items must not be less than FF_VOLUMES. Valid
|
||||||
|
/ characters for the volume ID strings are A-Z, a-z and 0-9, however, they are
|
||||||
|
/ compared in case-insensitive. If FF_STR_VOLUME_ID >= 1 and FF_VOLUME_STRS is
|
||||||
|
/ not defined, a user defined volume string table is needed as:
|
||||||
|
/
|
||||||
|
/ const char* VolumeStr[FF_VOLUMES] = {"ram","flash","sd","usb",...
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_MULTI_PARTITION 0
|
||||||
|
/* This option switches support for multiple volumes on the physical drive.
|
||||||
|
/ By default (0), each logical drive number is bound to the same physical drive
|
||||||
|
/ number and only an FAT volume found on the physical drive will be mounted.
|
||||||
|
/ When this feature is enabled (1), each logical drive number can be bound to
|
||||||
|
/ arbitrary physical drive and partition listed in the VolToPart[]. Also f_fdisk()
|
||||||
|
/ will be available. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_MIN_SS 512
|
||||||
|
#define FF_MAX_SS 512
|
||||||
|
/* This set of options configures the range of sector size to be supported. (512,
|
||||||
|
/ 1024, 2048 or 4096) Always set both 512 for most systems, generic memory card and
|
||||||
|
/ harddisk, but a larger value may be required for on-board flash memory and some
|
||||||
|
/ type of optical media. When FF_MAX_SS is larger than FF_MIN_SS, FatFs is
|
||||||
|
/ configured for variable sector size mode and disk_ioctl() needs to implement
|
||||||
|
/ GET_SECTOR_SIZE command. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_LBA64 1
|
||||||
|
/* This option switches support for 64-bit LBA. (0:Disable or 1:Enable)
|
||||||
|
/ To enable the 64-bit LBA, also exFAT needs to be enabled. (FF_FS_EXFAT == 1) */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_MIN_GPT 0x10000000
|
||||||
|
/* Minimum number of sectors to switch GPT as partitioning format in f_mkfs() and
|
||||||
|
/ f_fdisk(). 2^32 sectors maximum. This option has no effect when FF_LBA64 == 0. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_USE_TRIM 0
|
||||||
|
/* This option switches support for ATA-TRIM. (0:Disable or 1:Enable)
|
||||||
|
/ To enable this feature, also CTRL_TRIM command should be implemented to
|
||||||
|
/ the disk_ioctl(). */
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*---------------------------------------------------------------------------/
|
||||||
|
/ System Configurations
|
||||||
|
/---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define FF_FS_TINY 0
|
||||||
|
/* This option switches tiny buffer configuration. (0:Normal or 1:Tiny)
|
||||||
|
/ At the tiny configuration, size of file object (FIL) is reduced FF_MAX_SS bytes.
|
||||||
|
/ Instead of private sector buffer eliminated from the file object, common sector
|
||||||
|
/ buffer in the filesystem object (FATFS) is used for the file data transfer. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_EXFAT 1
|
||||||
|
/* This option switches support for exFAT filesystem. (0:Disable or 1:Enable)
|
||||||
|
/ To enable exFAT, also LFN needs to be enabled. (FF_USE_LFN >= 1)
|
||||||
|
/ Note that enabling exFAT discards ANSI C (C89) compatibility. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_NORTC 1
|
||||||
|
#define FF_NORTC_MON 1
|
||||||
|
#define FF_NORTC_MDAY 1
|
||||||
|
#define FF_NORTC_YEAR 2025
|
||||||
|
/* The option FF_FS_NORTC switches timestamp feature. If the system does not have
|
||||||
|
/ an RTC or valid timestamp is not needed, set FF_FS_NORTC = 1 to disable the
|
||||||
|
/ timestamp feature. Every object modified by FatFs will have a fixed timestamp
|
||||||
|
/ defined by FF_NORTC_MON, FF_NORTC_MDAY and FF_NORTC_YEAR in local time.
|
||||||
|
/ To enable timestamp function (FF_FS_NORTC = 0), get_fattime() need to be added
|
||||||
|
/ to the project to read current time form real-time clock. FF_NORTC_MON,
|
||||||
|
/ FF_NORTC_MDAY and FF_NORTC_YEAR have no effect.
|
||||||
|
/ These options have no effect in read-only configuration (FF_FS_READONLY = 1). */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_CRTIME 0
|
||||||
|
/* This option enables(1)/disables(0) the timestamp of the file created. When
|
||||||
|
/ set 1, the file created time is available in FILINFO structure. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_NOFSINFO 0
|
||||||
|
/* If you need to know the correct free space on the FAT32 volume, set bit 0 of
|
||||||
|
/ this option, and f_getfree() on the first time after volume mount will force
|
||||||
|
/ a full FAT scan. Bit 1 controls the use of last allocated cluster number.
|
||||||
|
/
|
||||||
|
/ bit0=0: Use free cluster count in the FSINFO if available.
|
||||||
|
/ bit0=1: Do not trust free cluster count in the FSINFO.
|
||||||
|
/ bit1=0: Use last allocated cluster number in the FSINFO if available.
|
||||||
|
/ bit1=1: Do not trust last allocated cluster number in the FSINFO.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_LOCK 0
|
||||||
|
/* The option FF_FS_LOCK switches file lock function to control duplicated file open
|
||||||
|
/ and illegal operation to open objects. This option must be 0 when FF_FS_READONLY
|
||||||
|
/ is 1.
|
||||||
|
/
|
||||||
|
/ 0: Disable file lock function. To avoid volume corruption, application program
|
||||||
|
/ should avoid illegal open, remove and rename to the open objects.
|
||||||
|
/ >0: Enable file lock function. The value defines how many files/sub-directories
|
||||||
|
/ can be opened simultaneously under file lock control. Note that the file
|
||||||
|
/ lock control is independent of re-entrancy. */
|
||||||
|
|
||||||
|
|
||||||
|
#define FF_FS_REENTRANT 0
|
||||||
|
#define FF_FS_TIMEOUT 1000
|
||||||
|
/* The option FF_FS_REENTRANT switches the re-entrancy (thread safe) of the FatFs
|
||||||
|
/ module itself. Note that regardless of this option, file access to different
|
||||||
|
/ volume is always re-entrant and volume control functions, f_mount(), f_mkfs()
|
||||||
|
/ and f_fdisk(), are always not re-entrant. Only file/directory access to
|
||||||
|
/ the same volume is under control of this featuer.
|
||||||
|
/
|
||||||
|
/ 0: Disable re-entrancy. FF_FS_TIMEOUT have no effect.
|
||||||
|
/ 1: Enable re-entrancy. Also user provided synchronization handlers,
|
||||||
|
/ ff_mutex_create(), ff_mutex_delete(), ff_mutex_take() and ff_mutex_give(),
|
||||||
|
/ must be added to the project. Samples are available in ffsystem.c.
|
||||||
|
/
|
||||||
|
/ The FF_FS_TIMEOUT defines timeout period in unit of O/S time tick.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*--- End of configuration options ---*/
|
||||||
10908
sphaira/source/ff16/ffunicode.c
Normal file
10908
sphaira/source/ff16/ffunicode.c
Normal file
File diff suppressed because it is too large
Load Diff
@@ -2,6 +2,7 @@
|
|||||||
#include "fs.hpp"
|
#include "fs.hpp"
|
||||||
#include "app.hpp"
|
#include "app.hpp"
|
||||||
|
|
||||||
|
#include <ff.h>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <minIni.h>
|
#include <minIni.h>
|
||||||
#include <usbhsfs.h>
|
#include <usbhsfs.h>
|
||||||
@@ -105,4 +106,20 @@ auto GetStdio(bool write) -> StdioEntries {
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto GetFat() -> StdioEntries {
|
||||||
|
StdioEntries out{};
|
||||||
|
|
||||||
|
for (auto& e : VolumeStr) {
|
||||||
|
char path[64];
|
||||||
|
std::snprintf(path, sizeof(path), "%s:/", e);
|
||||||
|
|
||||||
|
char display_name[0x100];
|
||||||
|
std::snprintf(display_name, sizeof(display_name), "%s (Read Only)", path);
|
||||||
|
|
||||||
|
out.emplace_back(path, display_name, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace sphaira::location
|
} // namespace sphaira::location
|
||||||
|
|||||||
@@ -107,8 +107,10 @@ auto Menu::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
|
|
||||||
m_path = new_path;
|
m_path = new_path;
|
||||||
m_entries.clear();
|
m_entries.clear();
|
||||||
m_index = 0;
|
m_entries_index.clear();
|
||||||
m_list->SetYoff(0);
|
m_entries_index_hidden.clear();
|
||||||
|
m_entries_current = {};
|
||||||
|
SetIndex(0);
|
||||||
SetTitleSubHeading(m_path);
|
SetTitleSubHeading(m_path);
|
||||||
|
|
||||||
fs::Dir d;
|
fs::Dir d;
|
||||||
@@ -120,10 +122,6 @@ auto Menu::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
|
|
||||||
const auto count = dir_entries.size();
|
const auto count = dir_entries.size();
|
||||||
m_entries.reserve(count);
|
m_entries.reserve(count);
|
||||||
|
|
||||||
m_entries_index.clear();
|
|
||||||
m_entries_index_hidden.clear();
|
|
||||||
|
|
||||||
m_entries_index.reserve(count);
|
m_entries_index.reserve(count);
|
||||||
m_entries_index_hidden.reserve(count);
|
m_entries_index_hidden.reserve(count);
|
||||||
|
|
||||||
@@ -154,7 +152,6 @@ auto Menu::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Sort();
|
Sort();
|
||||||
SetIndex(0);
|
|
||||||
|
|
||||||
// find previous entry
|
// find previous entry
|
||||||
if (is_walk_up && !m_previous_highlighted_file.empty()) {
|
if (is_walk_up && !m_previous_highlighted_file.empty()) {
|
||||||
|
|||||||
@@ -1040,10 +1040,13 @@ auto FsView::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
|
|
||||||
m_path = new_path;
|
m_path = new_path;
|
||||||
m_entries.clear();
|
m_entries.clear();
|
||||||
m_index = 0;
|
m_entries_index.clear();
|
||||||
m_list->SetYoff(0);
|
m_entries_index_hidden.clear();
|
||||||
m_menu->SetTitleSubHeading(m_path);
|
m_entries_index_search.clear();
|
||||||
|
m_entries_current = {};
|
||||||
m_selected_count = 0;
|
m_selected_count = 0;
|
||||||
|
SetIndex(0);
|
||||||
|
m_menu->SetTitleSubHeading(m_path);
|
||||||
|
|
||||||
fs::Dir d;
|
fs::Dir d;
|
||||||
R_TRY(m_fs->OpenDirectory(new_path, FsDirOpenMode_ReadDirs | FsDirOpenMode_ReadFiles, &d));
|
R_TRY(m_fs->OpenDirectory(new_path, FsDirOpenMode_ReadDirs | FsDirOpenMode_ReadFiles, &d));
|
||||||
@@ -1054,11 +1057,6 @@ auto FsView::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
|
|
||||||
const auto count = dir_entries.size();
|
const auto count = dir_entries.size();
|
||||||
m_entries.reserve(count);
|
m_entries.reserve(count);
|
||||||
|
|
||||||
m_entries_index.clear();
|
|
||||||
m_entries_index_hidden.clear();
|
|
||||||
m_entries_index_search.clear();
|
|
||||||
|
|
||||||
m_entries_index.reserve(count);
|
m_entries_index.reserve(count);
|
||||||
m_entries_index_hidden.reserve(count);
|
m_entries_index_hidden.reserve(count);
|
||||||
|
|
||||||
@@ -1078,8 +1076,6 @@ auto FsView::Scan(const fs::FsPath& new_path, bool is_walk_up) -> Result {
|
|||||||
// quick check to see if this is an update folder
|
// quick check to see if this is an update folder
|
||||||
m_is_update_folder = R_SUCCEEDED(CheckIfUpdateFolder());
|
m_is_update_folder = R_SUCCEEDED(CheckIfUpdateFolder());
|
||||||
|
|
||||||
SetIndex(0);
|
|
||||||
|
|
||||||
// find previous entry
|
// find previous entry
|
||||||
if (is_walk_up && !m_previous_highlighted_file.empty()) {
|
if (is_walk_up && !m_previous_highlighted_file.empty()) {
|
||||||
ON_SCOPE_EXIT(m_previous_highlighted_file.pop_back());
|
ON_SCOPE_EXIT(m_previous_highlighted_file.pop_back());
|
||||||
@@ -1645,16 +1641,18 @@ void FsView::DisplayOptions() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (m_entries_current.size()) {
|
if (m_entries_current.size()) {
|
||||||
options->Add<SidebarEntryCallback>("Cut"_i18n, [this](){
|
if (!m_fs_entry.IsReadOnly()) {
|
||||||
m_menu->AddSelectedEntries(SelectedType::Cut);
|
options->Add<SidebarEntryCallback>("Cut"_i18n, [this](){
|
||||||
}, true);
|
m_menu->AddSelectedEntries(SelectedType::Cut);
|
||||||
|
}, true);
|
||||||
|
}
|
||||||
|
|
||||||
options->Add<SidebarEntryCallback>("Copy"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Copy"_i18n, [this](){
|
||||||
m_menu->AddSelectedEntries(SelectedType::Copy);
|
m_menu->AddSelectedEntries(SelectedType::Copy);
|
||||||
}, true);
|
}, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!m_menu->m_selected.Empty() && (m_menu->m_selected.Type() == SelectedType::Cut || m_menu->m_selected.Type() == SelectedType::Copy)) {
|
if (!m_menu->m_selected.Empty() && !m_fs_entry.IsReadOnly() && (m_menu->m_selected.Type() == SelectedType::Cut || m_menu->m_selected.Type() == SelectedType::Copy)) {
|
||||||
options->Add<SidebarEntryCallback>("Paste"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Paste"_i18n, [this](){
|
||||||
const std::string buf = "Paste file(s)?"_i18n;
|
const std::string buf = "Paste file(s)?"_i18n;
|
||||||
App::Push<OptionBox>(
|
App::Push<OptionBox>(
|
||||||
@@ -1668,7 +1666,7 @@ void FsView::DisplayOptions() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// can't rename more than 1 file
|
// can't rename more than 1 file
|
||||||
if (m_entries_current.size() && !m_selected_count) {
|
if (m_entries_current.size() && !m_selected_count && !m_fs_entry.IsReadOnly()) {
|
||||||
options->Add<SidebarEntryCallback>("Rename"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Rename"_i18n, [this](){
|
||||||
std::string out;
|
std::string out;
|
||||||
const auto& entry = GetEntry();
|
const auto& entry = GetEntry();
|
||||||
@@ -1696,7 +1694,7 @@ void FsView::DisplayOptions() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_entries_current.size()) {
|
if (m_entries_current.size() && !m_fs_entry.IsReadOnly()) {
|
||||||
options->Add<SidebarEntryCallback>("Delete"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Delete"_i18n, [this](){
|
||||||
m_menu->AddSelectedEntries(SelectedType::Delete);
|
m_menu->AddSelectedEntries(SelectedType::Delete);
|
||||||
|
|
||||||
@@ -1744,7 +1742,7 @@ void FsView::DisplayOptions() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (m_entries_current.size()) {
|
if (m_entries_current.size()) {
|
||||||
if (check_all_ext(ZIP_EXTENSIONS)) {
|
if (check_all_ext(ZIP_EXTENSIONS) && !m_fs_entry.IsReadOnly()) {
|
||||||
options->Add<SidebarEntryCallback>("Extract zip"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Extract zip"_i18n, [this](){
|
||||||
auto options = std::make_unique<Sidebar>("Extract Options"_i18n, Sidebar::Side::RIGHT);
|
auto options = std::make_unique<Sidebar>("Extract Options"_i18n, Sidebar::Side::RIGHT);
|
||||||
ON_SCOPE_EXIT(App::Push(std::move(options)));
|
ON_SCOPE_EXIT(App::Push(std::move(options)));
|
||||||
@@ -1771,7 +1769,7 @@ void FsView::DisplayOptions() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!check_all_ext(ZIP_EXTENSIONS) || m_selected_count) {
|
if ((!check_all_ext(ZIP_EXTENSIONS) || m_selected_count) && !m_fs_entry.IsReadOnly()) {
|
||||||
options->Add<SidebarEntryCallback>("Compress to zip"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("Compress to zip"_i18n, [this](){
|
||||||
auto options = std::make_unique<Sidebar>("Compress Options"_i18n, Sidebar::Side::RIGHT);
|
auto options = std::make_unique<Sidebar>("Compress Options"_i18n, Sidebar::Side::RIGHT);
|
||||||
ON_SCOPE_EXIT(App::Push(std::move(options)));
|
ON_SCOPE_EXIT(App::Push(std::move(options)));
|
||||||
@@ -1818,53 +1816,66 @@ void FsView::DisplayAdvancedOptions() {
|
|||||||
mount_items.push_back(i18n::get(e.name));
|
mount_items.push_back(i18n::get(e.name));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const auto fat_entries = location::GetFat();
|
||||||
|
for (const auto& e: fat_entries) {
|
||||||
|
u32 flags{};
|
||||||
|
if (e.write_protect) {
|
||||||
|
flags |= FsEntryFlag_ReadOnly;
|
||||||
|
}
|
||||||
|
|
||||||
|
fs_entries.emplace_back(e.name, e.mount, FsType::Stdio, flags);
|
||||||
|
mount_items.push_back(e.name);
|
||||||
|
}
|
||||||
|
|
||||||
options->Add<SidebarEntryArray>("Mount"_i18n, mount_items, [this, fs_entries](s64& index_out){
|
options->Add<SidebarEntryArray>("Mount"_i18n, mount_items, [this, fs_entries](s64& index_out){
|
||||||
App::PopToMenu();
|
App::PopToMenu();
|
||||||
SetFs(fs_entries[index_out].root, fs_entries[index_out]);
|
SetFs(fs_entries[index_out].root, fs_entries[index_out]);
|
||||||
}, i18n::get(m_fs_entry.name));
|
}, i18n::get(m_fs_entry.name));
|
||||||
|
|
||||||
options->Add<SidebarEntryCallback>("Create File"_i18n, [this](){
|
if (!m_fs_entry.IsReadOnly()) {
|
||||||
std::string out;
|
options->Add<SidebarEntryCallback>("Create File"_i18n, [this](){
|
||||||
if (R_SUCCEEDED(swkbd::ShowText(out, "Set File Name"_i18n.c_str(), fs::AppendPath(m_path, ""))) && !out.empty()) {
|
std::string out;
|
||||||
App::PopToMenu();
|
if (R_SUCCEEDED(swkbd::ShowText(out, "Set File Name"_i18n.c_str(), fs::AppendPath(m_path, ""))) && !out.empty()) {
|
||||||
|
App::PopToMenu();
|
||||||
|
|
||||||
fs::FsPath full_path;
|
fs::FsPath full_path;
|
||||||
if (out.starts_with(m_fs_entry.root.s)) {
|
if (out.starts_with(m_fs_entry.root.s)) {
|
||||||
full_path = out;
|
full_path = out;
|
||||||
} else {
|
} else {
|
||||||
full_path = fs::AppendPath(m_path, out);
|
full_path = fs::AppendPath(m_path, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_fs->CreateDirectoryRecursivelyWithPath(full_path);
|
||||||
|
if (R_SUCCEEDED(m_fs->CreateFile(full_path, 0, 0))) {
|
||||||
|
log_write("created file: %s\n", full_path.s);
|
||||||
|
Scan(m_path);
|
||||||
|
} else {
|
||||||
|
log_write("failed to create file: %s\n", full_path.s);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
|
||||||
m_fs->CreateDirectoryRecursivelyWithPath(full_path);
|
options->Add<SidebarEntryCallback>("Create Folder"_i18n, [this](){
|
||||||
if (R_SUCCEEDED(m_fs->CreateFile(full_path, 0, 0))) {
|
std::string out;
|
||||||
log_write("created file: %s\n", full_path.s);
|
if (R_SUCCEEDED(swkbd::ShowText(out, "Set Folder Name"_i18n.c_str(), fs::AppendPath(m_path, ""))) && !out.empty()) {
|
||||||
Scan(m_path);
|
App::PopToMenu();
|
||||||
} else {
|
|
||||||
log_write("failed to create file: %s\n", full_path.s);
|
fs::FsPath full_path;
|
||||||
|
if (out.starts_with(m_fs_entry.root.s)) {
|
||||||
|
full_path = out;
|
||||||
|
} else {
|
||||||
|
full_path = fs::AppendPath(m_path, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(m_fs->CreateDirectoryRecursively(full_path))) {
|
||||||
|
log_write("created dir: %s\n", full_path.s);
|
||||||
|
Scan(m_path);
|
||||||
|
} else {
|
||||||
|
log_write("failed to create dir: %s\n", full_path.s);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
});
|
}
|
||||||
|
|
||||||
options->Add<SidebarEntryCallback>("Create Folder"_i18n, [this](){
|
|
||||||
std::string out;
|
|
||||||
if (R_SUCCEEDED(swkbd::ShowText(out, "Set Folder Name"_i18n.c_str(), fs::AppendPath(m_path, ""))) && !out.empty()) {
|
|
||||||
App::PopToMenu();
|
|
||||||
|
|
||||||
fs::FsPath full_path;
|
|
||||||
if (out.starts_with(m_fs_entry.root.s)) {
|
|
||||||
full_path = out;
|
|
||||||
} else {
|
|
||||||
full_path = fs::AppendPath(m_path, out);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (R_SUCCEEDED(m_fs->CreateDirectoryRecursively(full_path))) {
|
|
||||||
log_write("created dir: %s\n", full_path.s);
|
|
||||||
Scan(m_path);
|
|
||||||
} else {
|
|
||||||
log_write("failed to create dir: %s\n", full_path.s);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (IsSd() && m_entries_current.size() && !m_selected_count && GetEntry().IsFile() && GetEntry().file_size < 1024*64) {
|
if (IsSd() && m_entries_current.size() && !m_selected_count && GetEntry().IsFile() && GetEntry().file_size < 1024*64) {
|
||||||
options->Add<SidebarEntryCallback>("View as text (unfinished)"_i18n, [this](){
|
options->Add<SidebarEntryCallback>("View as text (unfinished)"_i18n, [this](){
|
||||||
|
|||||||
@@ -847,23 +847,18 @@ Result BuildNspEntry(const Entry& e, const ContentInfoEntry& info, const keys::K
|
|||||||
TikEntry entry{rights_id, key_gen};
|
TikEntry entry{rights_id, key_gen};
|
||||||
log_write("rights id is valid, fetching common ticket and cert\n");
|
log_write("rights id is valid, fetching common ticket and cert\n");
|
||||||
|
|
||||||
u64 tik_size;
|
// todo: fetch array of tickets to know where the ticket is stored.
|
||||||
u64 cert_size;
|
if (R_FAILED(es::GetCommonTicketAndCertificate(rights_id, entry.tik_data, entry.cert_data))) {
|
||||||
R_TRY(es::GetCommonTicketAndCertificateSize(&tik_size, &cert_size, &rights_id));
|
R_TRY(es::GetPersonalisedTicketAndCertificate(rights_id, entry.tik_data, entry.cert_data));
|
||||||
log_write("got tik_size: %zu cert_size: %zu\n", tik_size, cert_size);
|
}
|
||||||
|
|
||||||
entry.tik_data.resize(tik_size);
|
|
||||||
entry.cert_data.resize(cert_size);
|
|
||||||
R_TRY(es::GetCommonTicketAndCertificateData(&tik_size, &cert_size, entry.tik_data.data(), entry.tik_data.size(), entry.cert_data.data(), entry.cert_data.size(), &rights_id));
|
|
||||||
log_write("got tik_data: %zu cert_data: %zu\n", tik_size, cert_size);
|
|
||||||
|
|
||||||
// patch fake ticket / convert personalised to common if needed.
|
// patch fake ticket / convert personalised to common if needed.
|
||||||
R_TRY(es::PatchTicket(entry.tik_data, entry.cert_data, key_gen, keys, App::GetApp()->m_dump_convert_to_common_ticket.Get()));
|
R_TRY(es::PatchTicket(entry.tik_data, entry.cert_data, key_gen, keys, App::GetApp()->m_dump_convert_to_common_ticket.Get()));
|
||||||
|
|
||||||
char tik_name[0x200];
|
char tik_name[64];
|
||||||
std::snprintf(tik_name, sizeof(tik_name), "%s%s", hexIdToStr(rights_id).str, ".tik");
|
std::snprintf(tik_name, sizeof(tik_name), "%s%s", hexIdToStr(rights_id).str, ".tik");
|
||||||
|
|
||||||
char cert_name[0x200];
|
char cert_name[64];
|
||||||
std::snprintf(cert_name, sizeof(cert_name), "%s%s", hexIdToStr(rights_id).str, ".cert");
|
std::snprintf(cert_name, sizeof(cert_name), "%s%s", hexIdToStr(rights_id).str, ".cert");
|
||||||
|
|
||||||
out.collections.emplace_back(tik_name, offset, entry.tik_data.size());
|
out.collections.emplace_back(tik_name, offset, entry.tik_data.size());
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
#include "yati/nx/nxdumptool_rsa.h"
|
#include "yati/nx/nxdumptool_rsa.h"
|
||||||
#include "yati/nx/service_guard.h"
|
#include "yati/nx/service_guard.h"
|
||||||
#include "defines.hpp"
|
#include "defines.hpp"
|
||||||
|
#include "ui/types.hpp"
|
||||||
#include "log.hpp"
|
#include "log.hpp"
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
@@ -383,4 +384,45 @@ bool IsRightsIdFound(const FsRightsId& id, std::span<const FsRightsId> ids) {
|
|||||||
return it != ids.end();
|
return it != ids.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result GetCommonTicketAndCertificate(const FsRightsId& rights_id, std::vector<u8>& tik_out, std::vector<u8>& cert_out) {
|
||||||
|
u64 tik_size, cert_size;
|
||||||
|
R_TRY(es::GetCommonTicketAndCertificateSize(&tik_size, &cert_size, &rights_id));
|
||||||
|
|
||||||
|
tik_out.resize(tik_size);
|
||||||
|
cert_out.resize(cert_size);
|
||||||
|
|
||||||
|
return GetCommonTicketAndCertificateData(&tik_size, &cert_size, tik_out.data(), tik_out.size(), cert_out.data(), cert_out.size(), &rights_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
Result GetPersonalisedTicketAndCertificate(const FsRightsId& rights_id, std::vector<u8>& tik_out, std::vector<u8>& cert_out) {
|
||||||
|
R_THROW(0x1);
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
fs::FsStdio fs;
|
||||||
|
|
||||||
|
TimeStamp ts;
|
||||||
|
std::vector<u8> tik_buf;
|
||||||
|
// R_TRY(fs.read_entire_file("system:/save/80000000000000e1", tik_buf));
|
||||||
|
R_TRY(fs.read_entire_file("SYSTEM:/save/80000000000000e2", tik_buf));
|
||||||
|
log_write("[ES] size: %zu\n", tik_buf.size());
|
||||||
|
log_write("\t\t[ticket read] time taken: %.2fs %zums\n", ts.GetSecondsD(), ts.GetMs());
|
||||||
|
ts.Update();
|
||||||
|
|
||||||
|
for (u32 i = 0; i < tik_buf.size() - 0x400; i += 0x400) {
|
||||||
|
const auto tikRsa2048 = (const TicketRsa2048*)(tik_buf.data() + i);
|
||||||
|
if (tikRsa2048->signature_block.sig_type != SigType_Rsa2048Sha256) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!std::memcmp(&rights_id, &tikRsa2048->data.rights_id, sizeof(rights_id))) {
|
||||||
|
log_write("\t[ES] tikRsa2048, found at: %u\n", i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log_write("[ES] finished es search\n");
|
||||||
|
log_write("\t\t[ticket search] time taken: %.2fs %zums\n", ts.GetSecondsD(), ts.GetMs());
|
||||||
|
R_THROW(0x1);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace sphaira::es
|
} // namespace sphaira::es
|
||||||
|
|||||||
Reference in New Issue
Block a user