Finish theme scan, add preliminary shuffle install
This commit is contained in:
@@ -9,12 +9,13 @@ typedef struct {
|
|||||||
u16 author[0x40];
|
u16 author[0x40];
|
||||||
char icon_data[0x1200];
|
char icon_data[0x1200];
|
||||||
u16 path[262];
|
u16 path[262];
|
||||||
ssize_t path_len;
|
|
||||||
bool is_zip;
|
bool is_zip;
|
||||||
|
bool selected;
|
||||||
} theme;
|
} theme;
|
||||||
|
|
||||||
Result single_install(theme);
|
|
||||||
int scan_themes(theme **themes);
|
|
||||||
void parse_smdh(theme *entry, u16 *path);
|
void parse_smdh(theme *entry, u16 *path);
|
||||||
|
int scan_themes(theme **themes, int num_themes);
|
||||||
|
Result single_install(theme);
|
||||||
|
Result shuffle_install(theme **themes_list, int num_themes);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
11
source/fs.c
11
source/fs.c
@@ -81,14 +81,11 @@ int get_number_entries(char *path)
|
|||||||
bool done = false;
|
bool done = false;
|
||||||
while (!done)
|
while (!done)
|
||||||
{
|
{
|
||||||
FS_DirectoryEntry *entry = malloc(sizeof(entry));
|
FS_DirectoryEntry *entry = malloc(sizeof(FS_DirectoryEntry));
|
||||||
u32 entries_read;
|
u32 entries_read = 0;
|
||||||
FSDIR_Read(dir_handle, &entries_read, 1, entry);
|
FSDIR_Read(dir_handle, &entries_read, 1, entry);
|
||||||
u32 attributes = entry->attributes;
|
|
||||||
char shortExt[0x4] = {0};
|
|
||||||
strcpy(shortExt, entry->shortExt);
|
|
||||||
free(entry);
|
free(entry);
|
||||||
if (entries_read && (!strcmp(shortExt, "ZIP") || attributes == 0)) count ++;
|
if (entries_read) count++;
|
||||||
else if (!entries_read) break;
|
else if (!entries_read) break;
|
||||||
}
|
}
|
||||||
FSDIR_Close(dir_handle);
|
FSDIR_Close(dir_handle);
|
||||||
@@ -117,7 +114,7 @@ u32 zip_file_to_buf(char *file_name, u16 *zip_path, char **buf)
|
|||||||
u8 *path = calloc(sizeof(u8), len * 4);
|
u8 *path = calloc(sizeof(u8), len * 4);
|
||||||
utf16_to_utf8(path, zip_path, len);
|
utf16_to_utf8(path, zip_path, len);
|
||||||
|
|
||||||
unzFile zip_handle = unzOpen((char*)path);
|
unzFile zip_handle = unzOpen(path); // Can unzOpen really handle utf8?
|
||||||
|
|
||||||
if (zip_handle == NULL) return 0;
|
if (zip_handle == NULL) return 0;
|
||||||
u32 file_size = 0;
|
u32 file_size = 0;
|
||||||
|
|||||||
@@ -37,6 +37,9 @@ int main(void)
|
|||||||
consoleInit(GFX_TOP, NULL);
|
consoleInit(GFX_TOP, NULL);
|
||||||
|
|
||||||
int theme_count = get_number_entries("/Themes");
|
int theme_count = get_number_entries("/Themes");
|
||||||
|
printf("Theme count: %i\n", theme_count);
|
||||||
|
theme **themes_list = calloc(theme_count, sizeof(theme));
|
||||||
|
scan_themes(themes_list, theme_count);
|
||||||
|
|
||||||
while(aptMainLoop())
|
while(aptMainLoop())
|
||||||
{
|
{
|
||||||
@@ -44,11 +47,18 @@ int main(void)
|
|||||||
u32 kDown = hidKeysDown();
|
u32 kDown = hidKeysDown();
|
||||||
if (kDown & KEY_A)
|
if (kDown & KEY_A)
|
||||||
{
|
{
|
||||||
theme *theme_data = malloc(sizeof(theme));
|
for (int i = 0; i < theme_count; i++)
|
||||||
u16 path[262] = {0};
|
{
|
||||||
struacat(path, "/Themes/Saber Lily");
|
printu(themes_list[i]->name);
|
||||||
parse_smdh(theme_data, path);
|
printu(themes_list[i]->path);
|
||||||
printu(theme_data->name);
|
}
|
||||||
|
}
|
||||||
|
if (kDown & KEY_B)
|
||||||
|
{
|
||||||
|
shuffle_install(themes_list, theme_count);
|
||||||
|
close_archives();
|
||||||
|
PTMSYSM_ShutdownAsync(0);
|
||||||
|
ptmSysmExit();
|
||||||
}
|
}
|
||||||
if (kDown & KEY_START)
|
if (kDown & KEY_START)
|
||||||
{
|
{
|
||||||
|
|||||||
207
source/themes.c
207
source/themes.c
@@ -7,11 +7,62 @@
|
|||||||
#include "unicode.h"
|
#include "unicode.h"
|
||||||
#include "fs.h"
|
#include "fs.h"
|
||||||
|
|
||||||
int scan_themes(theme **themes)
|
void parse_smdh(theme *entry, u16 *path)
|
||||||
{
|
{
|
||||||
|
char *info_buffer;
|
||||||
|
u64 size = 0;
|
||||||
|
if (!(entry->is_zip))
|
||||||
|
{
|
||||||
|
u16 path_to_smdh[0x106] = {0};
|
||||||
|
strucat(path_to_smdh, path);
|
||||||
|
struacat(path_to_smdh, "/info.smdh");
|
||||||
|
size = file_to_buf(fsMakePath(PATH_UTF16, path_to_smdh), ArchiveSD, &info_buffer);
|
||||||
|
} else {
|
||||||
|
size = zip_file_to_buf("info.smdh", path, &info_buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!size)
|
||||||
|
{
|
||||||
|
printf("Failed on path: ");
|
||||||
|
printu(path);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(entry->name, info_buffer + 0x08, 0x80);
|
||||||
|
memcpy(entry->desc, info_buffer + 0x88, 0x100);
|
||||||
|
memcpy(entry->author, info_buffer + 0x188, 0x80);
|
||||||
|
memcpy(entry->icon_data, info_buffer + 0x2040, 0x1200);
|
||||||
|
free(info_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
int scan_themes(theme **themes, int num_themes)
|
||||||
|
{
|
||||||
|
Handle dir_handle;
|
||||||
|
FSUSER_OpenDirectory(&dir_handle, ArchiveSD, fsMakePath(PATH_ASCII, "/Themes"));
|
||||||
|
for (int i = 0; i < num_themes; i++)
|
||||||
|
{
|
||||||
|
FS_DirectoryEntry *entry = malloc(sizeof(FS_DirectoryEntry));
|
||||||
|
FSDIR_Read(dir_handle, NULL, 1, entry);
|
||||||
|
theme *theme_info = malloc(sizeof(theme));
|
||||||
|
u16 theme_path[0x106] = {0};
|
||||||
|
struacat(theme_path, "/Themes/");
|
||||||
|
strucat(theme_path, entry->name);
|
||||||
|
if (!strcmp(entry->shortExt, "ZIP"))
|
||||||
|
{
|
||||||
|
theme_info->is_zip = true;
|
||||||
|
} else {
|
||||||
|
theme_info->is_zip = false;
|
||||||
|
}
|
||||||
|
parse_smdh(theme_info, theme_path);
|
||||||
|
memcpy(theme_info->path, theme_path, 0x106 * sizeof(u16));
|
||||||
|
theme_info->selected = true;
|
||||||
|
themes[i] = theme_info;
|
||||||
|
free(entry);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Install a single theme
|
||||||
Result single_install(theme theme_to_install)
|
Result single_install(theme theme_to_install)
|
||||||
{
|
{
|
||||||
char *body;
|
char *body;
|
||||||
@@ -115,20 +166,142 @@ Result single_install(theme theme_to_install)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void parse_smdh(theme *entry, u16 *path)
|
Result shuffle_install(theme **themes_list, int num_themes)
|
||||||
{
|
{
|
||||||
u16 path_to_smdh[0x106] = {0};
|
u8 count = 0;
|
||||||
printu(path);
|
theme *shuffle_themes[10] = {0};
|
||||||
strucat(path_to_smdh, path);
|
u32 body_sizes[10] = {0};
|
||||||
struacat(path_to_smdh, "/info.smdh");
|
u32 bgm_sizes[10] = {0};
|
||||||
printu(path_to_smdh);
|
for (int i = 0; i < num_themes; i++)
|
||||||
char *info_buffer;
|
{
|
||||||
|
if (count > 9) return MAKERESULT(RL_USAGE, RS_INVALIDARG, RM_COMMON, RD_INVALID_SELECTION);
|
||||||
u64 size = file_to_buf(fsMakePath(PATH_UTF16, path_to_smdh), ArchiveSD, &info_buffer);
|
if (themes_list[i]->selected)
|
||||||
printf("Size: %llu\n", size);
|
{
|
||||||
memcpy(entry->name, info_buffer + 0x08, 0x80);
|
shuffle_themes[count++] = themes_list[i];
|
||||||
memcpy(entry->desc, info_buffer + 0x88, 0x100);
|
}
|
||||||
memcpy(entry->author, info_buffer + 0x188, 0x80);
|
}
|
||||||
memcpy(entry->icon_data, info_buffer + 0x2040, 0x1200);
|
|
||||||
free(info_buffer);
|
char *savedata_buf;
|
||||||
|
u32 size = file_to_buf(fsMakePath(PATH_ASCII, "/SaveData.dat"), ArchiveHomeExt, &savedata_buf);
|
||||||
|
if (size == 0)
|
||||||
|
{
|
||||||
|
return MAKERESULT(RL_USAGE, RS_CANCELED, RM_APPLICATION, RD_NOT_FOUND);
|
||||||
|
}
|
||||||
|
|
||||||
|
savedata_buf[0x141b] = 1;
|
||||||
|
memset(&savedata_buf[0x13b8], 0, 8);
|
||||||
|
savedata_buf[0x13bd] = 3;
|
||||||
|
savedata_buf[0x13b8] = 0xff;
|
||||||
|
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
memset(&savedata_buf[0x13c0 + 0x8 * i], 0, 8); // clear any existing theme structure. 8 is the length of the theme structure, so 8 * i is the pos of the current one
|
||||||
|
if (count > i) // if we are still installing themes...
|
||||||
|
{
|
||||||
|
savedata_buf[0x13c0 + (8 * i)] = i; // index
|
||||||
|
savedata_buf[0x13c0 + (8 * i) + 5] = 3; // persistence (?)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
buf_to_file(size, "/SaveData.dat", ArchiveHomeExt, savedata_buf);
|
||||||
|
free(savedata_buf);
|
||||||
|
|
||||||
|
Handle body_cache_handle;
|
||||||
|
FSUSER_OpenFile(&body_cache_handle, ArchiveThemeExt, fsMakePath(PATH_ASCII, "/BodyCache_rd.bin"), FS_OPEN_WRITE, 0);
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
if (count > i)
|
||||||
|
{
|
||||||
|
if (shuffle_themes[i]->is_zip)
|
||||||
|
{
|
||||||
|
char *body_buf;
|
||||||
|
u32 body_size = zip_file_to_buf("body_LZ.bin", shuffle_themes[i]->path, &body_buf);
|
||||||
|
body_sizes[i] = body_size;
|
||||||
|
FSFILE_Write(body_cache_handle, NULL, 0x150000 * i, body_buf, body_size, FS_WRITE_FLUSH);
|
||||||
|
free(body_buf);
|
||||||
|
} else {
|
||||||
|
u16 path[0x106] = {0};
|
||||||
|
strucat(path, shuffle_themes[i]->path);
|
||||||
|
struacat(path, "/body_LZ.bin");
|
||||||
|
char *body_buf;
|
||||||
|
u32 body_size = file_to_buf(fsMakePath(PATH_UTF16, path), ArchiveSD, &body_buf);
|
||||||
|
body_sizes[i] = body_size;
|
||||||
|
FSFILE_Write(body_cache_handle, NULL, 0x150000 * i, body_buf, body_size, FS_WRITE_FLUSH);
|
||||||
|
free(body_buf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FSFILE_Close(body_cache_handle);
|
||||||
|
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
char bgm_cache_path[17] = {0};
|
||||||
|
sprintf(bgm_cache_path, "/BgmCache0%i.bin", i);
|
||||||
|
if (count > i)
|
||||||
|
{
|
||||||
|
char *music_buf;
|
||||||
|
u32 music_size;
|
||||||
|
|
||||||
|
if (shuffle_themes[i]->is_zip)
|
||||||
|
{
|
||||||
|
music_size = zip_file_to_buf("bgm.bcstm", shuffle_themes[i]->path, &music_buf);
|
||||||
|
} else {
|
||||||
|
u16 path[0x106] = {0};
|
||||||
|
strucat(path, shuffle_themes[i]->path);
|
||||||
|
struacat(path, "/bgm.bcstm");
|
||||||
|
music_size = file_to_buf(fsMakePath(PATH_UTF16, path), ArchiveSD, &music_buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!music_size)
|
||||||
|
{
|
||||||
|
char *empty = calloc(1, 3371008);
|
||||||
|
buf_to_file(3371008, bgm_cache_path, ArchiveThemeExt, empty);
|
||||||
|
bgm_sizes[i] = 0;
|
||||||
|
free(empty);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
bgm_sizes[i] = music_size;
|
||||||
|
buf_to_file(music_size, bgm_cache_path, ArchiveThemeExt, music_buf);
|
||||||
|
free(music_buf);
|
||||||
|
} else {
|
||||||
|
char *empty = calloc(1, 3371008);
|
||||||
|
buf_to_file(3371008, bgm_cache_path, ArchiveThemeExt, empty);
|
||||||
|
bgm_sizes[i] = 0;
|
||||||
|
free(empty);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char *thememanage_buf;
|
||||||
|
file_to_buf(fsMakePath(PATH_ASCII, "/ThemeManage.bin"), ArchiveThemeExt, &thememanage_buf);
|
||||||
|
thememanage_buf[0x00] = 1; // Unknown, normally 0x1 with size of 4
|
||||||
|
thememanage_buf[0x01] = 0;
|
||||||
|
thememanage_buf[0x02] = 0;
|
||||||
|
thememanage_buf[0x03] = 0;
|
||||||
|
thememanage_buf[0x04] = 0; // Unknown, normally 0 with size of 4
|
||||||
|
thememanage_buf[0x05] = 0;
|
||||||
|
thememanage_buf[0x06] = 0;
|
||||||
|
thememanage_buf[0x07] = 0;
|
||||||
|
thememanage_buf[0x10] = 0xFF; // Unknown
|
||||||
|
thememanage_buf[0x14] = 0x01; // Unkown
|
||||||
|
thememanage_buf[0x18] = 0xFF; // DLC theme index - 0xFF indicates no DLC theme
|
||||||
|
thememanage_buf[0x1D] = 0x02; // Unknown, usually 0x200 to indicate theme-cache is being used
|
||||||
|
|
||||||
|
u32 *bodysizeloc = (u32*) (&thememanage_buf[0x08]); // Set non-shuffle theme sizes to 0
|
||||||
|
u32 *bgmsizeloc = (u32*) (&thememanage_buf[0x0C]);
|
||||||
|
*bodysizeloc = (u32) 0;
|
||||||
|
*bgmsizeloc = (u32) 0;
|
||||||
|
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
bodysizeloc = (u32*) (&thememanage_buf[0x338 + (4 * i)]); // body size info for shuffle themes starts at 0x338 and is 4 bytes for each theme
|
||||||
|
bgmsizeloc = (u32*) (&thememanage_buf[0x360 + (4 * i)]); // same thing for bgm but starting at 0x360
|
||||||
|
*bodysizeloc = body_sizes[i]; // We don't need to check if we've already installed all the themes because all sizes initialized to 0
|
||||||
|
*bgmsizeloc = bgm_sizes[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
buf_to_file(0x800, "/ThemeManage.bin", ArchiveThemeExt, thememanage_buf);
|
||||||
|
free(thememanage_buf);
|
||||||
|
|
||||||
|
return MAKERESULT(RL_SUCCESS, RS_SUCCESS, RM_COMMON, RD_SUCCESS);
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user