* No Nightlies for the Full-Rewrite. * Initial push, i guess. * Forgot to push the Test UniStore + T3X... * Use C2D flags for wrapping and centering * gitignore t3x correctly * Remove Test Store and hardcode to `sdmc:/3ds/Universal-Updater/stores/Universal-DB.unistore` for now. * Is functional now. * *More special checks and work.* * const <typename T> &. * Universal-DB, not Universal DB. * Derp. * Make 3DSX, NDS & Archive path configurable. * Last fixes + Fade out screen on exit. * See Desc. for more. - Add QR Code scan for downloading UniStores. - Add new Graphics. - Some fixes + improvements. * Fix search filtering, re-sort after search * Fix update check * Clear search items with X, not just reset results * The next progress. * PLEASE tell me, this is the only error.. Co-authored-by: Pk11 <epicpkmn11@outlook.com>
165 lines
No EOL
4.5 KiB
C++
165 lines
No EOL
4.5 KiB
C++
/*
|
|
* This file is part of Universal-Updater
|
|
* Copyright (C) 2019-2020 Universal-Team
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Additional Terms 7.b and 7.c of GPLv3 apply to this file:
|
|
* * Requiring preservation of specified reasonable legal notices or
|
|
* author attributions in that material or in the Appropriate Legal
|
|
* Notices displayed by works containing it.
|
|
* * Prohibiting misrepresentation of the origin of that material,
|
|
* or requiring that modified versions of such material be marked in
|
|
* reasonable ways as different from the original version.
|
|
*/
|
|
|
|
#include "cia.hpp"
|
|
#include "files.hpp"
|
|
|
|
Result CIA_LaunchTitle(const u64 &titleId, const FS_MediaType &mediaType) {
|
|
Result ret = 0;
|
|
u8 param[0x300];
|
|
u8 hmac[0x20];
|
|
|
|
if (R_FAILED(ret = APT_PrepareToDoApplicationJump(0, titleId, mediaType))) {
|
|
printf("Error In:\nAPT_PrepareToDoApplicationJump");
|
|
return ret;
|
|
}
|
|
|
|
if (R_FAILED(ret = APT_DoApplicationJump(param, sizeof(param), hmac))) {
|
|
printf("Error In:\nAPT_DoApplicationJump");
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
Result deletePrevious(const u64 &titleid, const FS_MediaType &media) {
|
|
Result ret = 0;
|
|
u32 titles_amount = 0;
|
|
|
|
ret = AM_GetTitleCount(media, &titles_amount);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nAM_GetTitleCount\n");
|
|
return ret;
|
|
}
|
|
|
|
u32 read_titles = 0;
|
|
u64 *titleIDs = (u64 *)malloc(titles_amount * sizeof(u64));
|
|
|
|
ret = AM_GetTitleList(&read_titles, media, titles_amount, titleIDs);
|
|
if (R_FAILED(ret)) {
|
|
free(titleIDs);
|
|
printf("Error in:\nAM_GetTitleList\n");
|
|
return ret;
|
|
}
|
|
|
|
for (u32 i = 0; i < read_titles; i++) {
|
|
if (titleIDs[i] == titleid) {
|
|
ret = AM_DeleteAppTitle(media, titleid);
|
|
break;
|
|
}
|
|
}
|
|
|
|
free(titleIDs);
|
|
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nAM_DeleteAppTitle\n");
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
FS_MediaType getTitleDestination(const u64 &titleId) {
|
|
u16 platform = (u16) ((titleId >> 48) & 0xFFFF);
|
|
u16 category = (u16) ((titleId >> 32) & 0xFFFF);
|
|
u8 variation = (u8) (titleId & 0xFF);
|
|
|
|
// DSiWare 3DS DSiWare, System, DLP Application System Title
|
|
return platform == 0x0003 || (platform == 0x0004 && ((category & 0x8011) != 0 || (category == 0x0000 && variation == 0x02))) ? MEDIATYPE_NAND : MEDIATYPE_SD;
|
|
}
|
|
|
|
u64 installSize = 0, installOffset = 0;
|
|
|
|
Result installCia(const char *ciaPath, const bool &updatingSelf) {
|
|
u32 bytes_read = 0, bytes_written;
|
|
installSize = 0, installOffset = 0; u64 size = 0;
|
|
Handle ciaHandle, fileHandle;
|
|
AM_TitleEntry info;
|
|
Result ret = 0;
|
|
FS_MediaType media = MEDIATYPE_SD;
|
|
|
|
ret = openFile(&fileHandle, ciaPath, false);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nopenFile\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = AM_GetCiaFileInfo(media, &info, fileHandle);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nAM_GetCiaFileInfo\n");
|
|
return ret;
|
|
}
|
|
|
|
media = getTitleDestination(info.titleID);
|
|
|
|
if (!updatingSelf) {
|
|
ret = deletePrevious(info.titleID, media);
|
|
if (R_FAILED(ret)) return ret;
|
|
}
|
|
|
|
ret = FSFILE_GetSize(fileHandle, &size);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nFSFILE_GetSize\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = AM_StartCiaInstall(media, &ciaHandle);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nAM_StartCiaInstall\n");
|
|
return ret;
|
|
}
|
|
|
|
u32 toRead = 0x200000;
|
|
u8 *buf = new u8[toRead];
|
|
|
|
if (!buf) return -1;
|
|
|
|
installSize = size;
|
|
do {
|
|
FSFILE_Read(fileHandle, &bytes_read, installOffset, buf, toRead);
|
|
FSFILE_Write(ciaHandle, &bytes_written, installOffset, buf, toRead, FS_WRITE_FLUSH);
|
|
installOffset += bytes_read;
|
|
} while(installOffset < installSize);
|
|
delete[] buf;
|
|
|
|
ret = AM_FinishCiaInstall(ciaHandle);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nAM_FinishCiaInstall\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = FSFILE_Close(fileHandle);
|
|
if (R_FAILED(ret)) {
|
|
printf("Error in:\nFSFILE_Close\n");
|
|
return ret;
|
|
}
|
|
|
|
if (updatingSelf) {
|
|
if (R_FAILED(ret = CIA_LaunchTitle(info.titleID, MEDIATYPE_SD))) return ret;
|
|
}
|
|
|
|
return 0;
|
|
} |