mirror of
https://gitlab.com/Nanolx/homebrewfilter.git
synced 2025-01-14 18:19:14 +01:00
253 lines
4.8 KiB
C
253 lines
4.8 KiB
C
/*******************************************************************************
|
|
* tools.c
|
|
*
|
|
* Copyright (c) 2009 The Lemon Man
|
|
* Copyright (c) 2009 Nicksasa
|
|
* Copyright (c) 2009 WiiPower
|
|
*
|
|
* Distributed under the terms of the GNU General Public License (v2)
|
|
* See http://www.gnu.org/licenses/gpl-2.0.txt for more info.
|
|
*
|
|
* Description:
|
|
* -----------
|
|
*
|
|
******************************************************************************/
|
|
|
|
#include <gccore.h>
|
|
#include <malloc.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <wiiuse/wpad.h>
|
|
|
|
#include "tools.h"
|
|
|
|
void printheadline()
|
|
{
|
|
int rows, cols;
|
|
CON_GetMetrics(&cols, &rows);
|
|
|
|
printf("TriiForce beta 7");
|
|
|
|
char buf[64];
|
|
sprintf(buf, "IOS%u (Rev %u)\n", IOS_GetVersion(), IOS_GetRevision());
|
|
printf("\x1B[%d;%dH", 0, cols-strlen(buf)-1);
|
|
printf(buf);
|
|
}
|
|
|
|
void set_highlight(bool highlight)
|
|
{
|
|
if (highlight)
|
|
{
|
|
printf("\x1b[%u;%um", 47, false);
|
|
printf("\x1b[%u;%um", 30, false);
|
|
} else
|
|
{
|
|
printf("\x1b[%u;%um", 37, false);
|
|
printf("\x1b[%u;%um", 40, false);
|
|
}
|
|
}
|
|
|
|
void *allocate_memory(u32 size)
|
|
{
|
|
return memalign(32, (size+31)&(~31) );
|
|
}
|
|
|
|
void Verify_Flags()
|
|
{
|
|
if (Power_Flag)
|
|
{
|
|
WPAD_Shutdown();
|
|
STM_ShutdownToStandby();
|
|
}
|
|
if (Reset_Flag)
|
|
{
|
|
WPAD_Shutdown();
|
|
STM_RebootSystem();
|
|
}
|
|
}
|
|
|
|
|
|
void waitforbuttonpress(u32 *out, u32 *outGC)
|
|
{
|
|
u32 pressed = 0;
|
|
u32 pressedGC = 0;
|
|
|
|
while (true)
|
|
{
|
|
Verify_Flags();
|
|
|
|
WPAD_ScanPads();
|
|
pressed = WPAD_ButtonsDown(0);
|
|
|
|
PAD_ScanPads();
|
|
pressedGC = PAD_ButtonsDown(0);
|
|
|
|
if(pressed || pressedGC)
|
|
{
|
|
if (pressedGC)
|
|
{
|
|
// Without waiting you can't select anything
|
|
usleep (20000);
|
|
}
|
|
if (out) *out = pressed;
|
|
if (outGC) *outGC = pressedGC;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
s32 read_file(char *filepath, u8 **buffer, u32 *filesize)
|
|
{
|
|
s32 Fd;
|
|
int ret;
|
|
|
|
if (buffer == NULL)
|
|
{
|
|
printf("NULL Pointer\n");
|
|
return -1;
|
|
}
|
|
|
|
Fd = ISFS_Open(filepath, ISFS_OPEN_READ);
|
|
if (Fd < 0)
|
|
{
|
|
printf("ISFS_Open %s failed %d\n", filepath, Fd);
|
|
return Fd;
|
|
}
|
|
|
|
fstats *status;
|
|
status = allocate_memory(sizeof(fstats));
|
|
if (status == NULL)
|
|
{
|
|
printf("Out of memory for status\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = ISFS_GetFileStats(Fd, status);
|
|
if (ret < 0)
|
|
{
|
|
printf("ISFS_GetFileStats failed %d\n", ret);
|
|
ISFS_Close(Fd);
|
|
free(status);
|
|
return -1;
|
|
}
|
|
|
|
*buffer = allocate_memory(status->file_length);
|
|
if (*buffer == NULL)
|
|
{
|
|
printf("Out of memory for buffer\n");
|
|
ISFS_Close(Fd);
|
|
free(status);
|
|
return -1;
|
|
}
|
|
|
|
ret = ISFS_Read(Fd, *buffer, status->file_length);
|
|
if (ret < 0)
|
|
{
|
|
printf("ISFS_Read failed %d\n", ret);
|
|
ISFS_Close(Fd);
|
|
free(status);
|
|
free(*buffer);
|
|
return ret;
|
|
}
|
|
ISFS_Close(Fd);
|
|
|
|
*filesize = status->file_length;
|
|
free(status);
|
|
|
|
return 0;
|
|
}
|
|
|
|
s32 identify(u64 titleid, u32 *ios)
|
|
{
|
|
char filepath[ISFS_MAXPATH] ATTRIBUTE_ALIGN(0x20);
|
|
u8 *tmdBuffer = NULL;
|
|
u32 tmdSize;
|
|
u8 *tikBuffer = NULL;
|
|
u32 tikSize;
|
|
u8 *certBuffer = NULL;
|
|
u32 certSize;
|
|
|
|
int ret;
|
|
|
|
printf("Reading TMD...");
|
|
fflush(stdout);
|
|
|
|
sprintf(filepath, "/title/%08x/%08x/content/title.tmd", TITLE_UPPER(titleid), TITLE_LOWER(titleid));
|
|
ret = read_file(filepath, &tmdBuffer, &tmdSize);
|
|
if (ret < 0)
|
|
{
|
|
printf("Reading TMD failed\n");
|
|
return ret;
|
|
}
|
|
printf("done\n");
|
|
|
|
*ios = (u32)(tmdBuffer[0x18b]);
|
|
|
|
printf("Reading ticket...");
|
|
fflush(stdout);
|
|
|
|
sprintf(filepath, "/ticket/%08x/%08x.tik", TITLE_UPPER(titleid), TITLE_LOWER(titleid));
|
|
ret = read_file(filepath, &tikBuffer, &tikSize);
|
|
if (ret < 0)
|
|
{
|
|
printf("Reading ticket failed\n");
|
|
free(tmdBuffer);
|
|
return ret;
|
|
}
|
|
printf("done\n");
|
|
|
|
printf("Reading certs...");
|
|
fflush(stdout);
|
|
|
|
sprintf(filepath, "/sys/cert.sys");
|
|
ret = read_file(filepath, &certBuffer, &certSize);
|
|
if (ret < 0)
|
|
{
|
|
printf("Reading certs failed\n");
|
|
free(tmdBuffer);
|
|
free(tikBuffer);
|
|
return ret;
|
|
}
|
|
printf("done\n");
|
|
|
|
printf("ES_Identify...");
|
|
fflush(stdout);
|
|
|
|
ret = ES_Identify((signed_blob*)certBuffer, certSize, (signed_blob*)tmdBuffer, tmdSize, (signed_blob*)tikBuffer, tikSize, NULL);
|
|
if (ret < 0)
|
|
{
|
|
switch(ret)
|
|
{
|
|
case ES_EINVAL:
|
|
printf("Error! ES_Identify (ret = %d;) Data invalid!\n", ret);
|
|
break;
|
|
case ES_EALIGN:
|
|
printf("Error! ES_Identify (ret = %d;) Data not aligned!\n", ret);
|
|
break;
|
|
case ES_ENOTINIT:
|
|
printf("Error! ES_Identify (ret = %d;) ES not initialized!\n", ret);
|
|
break;
|
|
case ES_ENOMEM:
|
|
printf("Error! ES_Identify (ret = %d;) No memory!\n", ret);
|
|
break;
|
|
default:
|
|
printf("Error! ES_Identify (ret = %d)\n", ret);
|
|
break;
|
|
}
|
|
free(tmdBuffer);
|
|
free(tikBuffer);
|
|
free(certBuffer);
|
|
return ret;
|
|
}
|
|
printf("done\n");
|
|
|
|
free(tmdBuffer);
|
|
free(tikBuffer);
|
|
free(certBuffer);
|
|
return 0;
|
|
}
|
|
|
|
|