2016-08-06 06:59:41 +00:00
|
|
|
/*
|
|
|
|
VitaShell
|
2017-01-12 16:45:52 +00:00
|
|
|
Copyright (C) 2015-2017, TheFloW
|
2016-08-06 06:59:41 +00:00
|
|
|
|
|
|
|
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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include "file.h"
|
|
|
|
#include "message_dialog.h"
|
2016-08-28 14:36:27 +00:00
|
|
|
#include "uncommon_dialog.h"
|
|
|
|
#include "theme.h"
|
2016-08-06 06:59:41 +00:00
|
|
|
#include "language.h"
|
|
|
|
#include "utils.h"
|
2016-09-10 11:28:25 +00:00
|
|
|
#include "bm.h"
|
2016-08-06 06:59:41 +00:00
|
|
|
|
|
|
|
SceCtrlData pad;
|
|
|
|
uint32_t old_buttons, current_buttons, pressed_buttons, hold_buttons, hold2_buttons, released_buttons;
|
|
|
|
|
|
|
|
static int netdbg_sock = -1;
|
|
|
|
static void *net_memory = NULL;
|
|
|
|
static int net_init = -1;
|
|
|
|
|
2016-08-23 15:38:31 +00:00
|
|
|
static int lock_power = 0;
|
|
|
|
|
2016-09-03 20:14:07 +00:00
|
|
|
void startDrawing(vita2d_texture *bg) {
|
2016-08-28 14:36:27 +00:00
|
|
|
vita2d_start_drawing();
|
|
|
|
vita2d_set_clear_color(BACKGROUND_COLOR);
|
|
|
|
vita2d_clear_screen();
|
|
|
|
|
2016-11-01 09:01:28 +00:00
|
|
|
if (wallpaper_image) {
|
|
|
|
vita2d_draw_texture(wallpaper_image, 0.0f, 0.0f);
|
|
|
|
} else if (bg) {
|
2016-09-03 20:14:07 +00:00
|
|
|
vita2d_draw_texture(bg, 0.0f, 0.0f);
|
2016-08-28 14:36:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void endDrawing() {
|
|
|
|
drawUncommonDialog();
|
|
|
|
vita2d_end_drawing();
|
|
|
|
vita2d_common_dialog_update();
|
|
|
|
vita2d_swap_buffers();
|
|
|
|
sceDisplayWaitVblankStart();
|
|
|
|
}
|
|
|
|
|
2016-09-12 20:34:03 +00:00
|
|
|
void closeWaitDialog() {
|
|
|
|
sceMsgDialogClose();
|
|
|
|
|
|
|
|
while (updateMessageDialog() != MESSAGE_DIALOG_RESULT_NONE) {
|
|
|
|
sceKernelDelayThread(1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
void errorDialog(int error) {
|
|
|
|
if (error < 0) {
|
|
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_OK, language_container[ERROR], error);
|
|
|
|
dialog_step = DIALOG_STEP_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
void infoDialog(const char *msg, ...) {
|
2016-08-06 06:59:41 +00:00
|
|
|
va_list list;
|
|
|
|
char string[512];
|
|
|
|
|
|
|
|
va_start(list, msg);
|
|
|
|
vsprintf(string, msg, list);
|
|
|
|
va_end(list);
|
|
|
|
|
|
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_OK, string);
|
|
|
|
dialog_step = DIALOG_STEP_INFO;
|
|
|
|
}
|
|
|
|
|
2016-09-12 20:34:03 +00:00
|
|
|
int checkMemoryCardFreeSpace(uint64_t size) {
|
|
|
|
uint64_t free_size = 0, max_size = 0;
|
|
|
|
sceAppMgrGetDevInfo("ux0:", &max_size, &free_size);
|
|
|
|
|
|
|
|
if (size >= (free_size + (40 * 1024 * 1024))) {
|
|
|
|
closeWaitDialog();
|
|
|
|
|
|
|
|
char size_string[16];
|
|
|
|
getSizeString(size_string, size - (free_size + (40 * 1024 * 1024)));
|
|
|
|
infoDialog(language_container[NO_SPACE_ERROR], size_string);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
static int power_tick_thread(SceSize args, void *argp) {
|
2016-08-23 15:38:31 +00:00
|
|
|
while (1) {
|
2016-09-05 18:36:49 +00:00
|
|
|
if (lock_power > 0) {
|
2016-08-23 15:38:31 +00:00
|
|
|
sceKernelPowerTick(SCE_KERNEL_POWER_TICK_DISABLE_AUTO_SUSPEND);
|
|
|
|
}
|
|
|
|
|
|
|
|
sceKernelDelayThread(10 * 1000 * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void initPowerTickThread() {
|
|
|
|
SceUID thid = sceKernelCreateThread("power_tick_thread", power_tick_thread, 0x10000100, 0x40000, 0, 0, NULL);
|
|
|
|
if (thid >= 0)
|
|
|
|
sceKernelStartThread(thid, 0, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void powerLock() {
|
2016-11-03 15:57:08 +00:00
|
|
|
if (!lock_power)
|
|
|
|
sceShellUtilLock(SCE_SHELL_UTIL_LOCK_TYPE_PS_BTN);
|
|
|
|
|
2016-09-05 18:36:49 +00:00
|
|
|
lock_power++;
|
2016-08-23 15:38:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void powerUnlock() {
|
2016-11-03 15:57:08 +00:00
|
|
|
if (lock_power)
|
|
|
|
sceShellUtilUnlock(SCE_SHELL_UTIL_LOCK_TYPE_PS_BTN);
|
|
|
|
|
2016-09-05 18:36:49 +00:00
|
|
|
lock_power--;
|
2016-11-03 15:57:08 +00:00
|
|
|
if (lock_power < 0)
|
|
|
|
lock_power = 0;
|
2016-08-06 06:59:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void readPad() {
|
|
|
|
static int hold_n = 0, hold2_n = 0;
|
|
|
|
|
|
|
|
memset(&pad, 0, sizeof(SceCtrlData));
|
|
|
|
sceCtrlPeekBufferPositive(0, &pad, 1);
|
|
|
|
|
|
|
|
if (pad.ly < ANALOG_CENTER - ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_LEFT_ANALOG_UP;
|
|
|
|
} else if (pad.ly > ANALOG_CENTER + ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_LEFT_ANALOG_DOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pad.lx < ANALOG_CENTER - ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_LEFT_ANALOG_LEFT;
|
|
|
|
} else if (pad.lx > ANALOG_CENTER + ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_LEFT_ANALOG_RIGHT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pad.ry < ANALOG_CENTER - ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_RIGHT_ANALOG_UP;
|
|
|
|
} else if (pad.ry > ANALOG_CENTER + ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_RIGHT_ANALOG_DOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pad.rx < ANALOG_CENTER - ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_RIGHT_ANALOG_LEFT;
|
|
|
|
} else if (pad.rx > ANALOG_CENTER + ANALOG_THRESHOLD) {
|
|
|
|
pad.buttons |= SCE_CTRL_RIGHT_ANALOG_RIGHT;
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
old_buttons = current_buttons;
|
2016-08-06 06:59:41 +00:00
|
|
|
current_buttons = pad.buttons;
|
|
|
|
pressed_buttons = current_buttons & ~old_buttons;
|
2017-02-12 15:27:43 +00:00
|
|
|
released_buttons = ~current_buttons & old_buttons;
|
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
hold_buttons = pressed_buttons;
|
|
|
|
hold2_buttons = pressed_buttons;
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
if (old_buttons & current_buttons) {
|
2016-08-06 06:59:41 +00:00
|
|
|
hold_n++;
|
|
|
|
if (hold_n >= 10) {
|
|
|
|
hold_buttons = current_buttons;
|
|
|
|
hold_n = 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
hold2_n++;
|
|
|
|
if (hold2_n >= 10) {
|
|
|
|
hold2_buttons = current_buttons;
|
|
|
|
hold2_n = 10;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
hold_n = 0;
|
|
|
|
hold2_n = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int holdButtons(SceCtrlData *pad, uint32_t buttons, uint64_t time) {
|
|
|
|
if ((pad->buttons & buttons) == buttons) {
|
|
|
|
uint64_t time_start = sceKernelGetProcessTimeWide();
|
|
|
|
|
|
|
|
while ((pad->buttons & buttons) == buttons) {
|
|
|
|
sceKernelDelayThread(10 * 1000);
|
|
|
|
sceCtrlPeekBufferPositive(0, pad, 1);
|
|
|
|
|
|
|
|
if ((sceKernelGetProcessTimeWide() - time_start) >= time) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
int hasEndSlash(const char *path) {
|
2016-09-06 19:57:04 +00:00
|
|
|
return path[strlen(path) - 1] == '/';
|
|
|
|
}
|
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
int removeEndSlash(char *path) {
|
|
|
|
int len = strlen(path);
|
|
|
|
|
|
|
|
if (path[len - 1] == '/') {
|
|
|
|
path[len - 1] = '\0';
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int addEndSlash(char *path) {
|
|
|
|
int len = strlen(path);
|
|
|
|
if (len < MAX_PATH_LENGTH - 2) {
|
|
|
|
if (path[len - 1] != '/') {
|
|
|
|
strcat(path, "/");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void getSizeString(char *string, uint64_t size) {
|
|
|
|
double double_size = (double)size;
|
|
|
|
|
|
|
|
int i = 0;
|
|
|
|
static char *units[] = { "B", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" };
|
|
|
|
while (double_size >= 1024.0f) {
|
|
|
|
double_size /= 1024.0f;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(string, "%.*f %s", (i == 0) ? 0 : 2, double_size, units[i]);
|
|
|
|
}
|
|
|
|
|
2016-11-01 08:53:35 +00:00
|
|
|
void convertUtcToLocalTime(SceDateTime *time_local, SceDateTime *time_utc) {
|
|
|
|
SceRtcTick tick;
|
|
|
|
sceRtcGetTick(time_utc, &tick);
|
|
|
|
sceRtcConvertUtcToLocalTime(&tick, &tick);
|
|
|
|
sceRtcSetTick(time_local, &tick);
|
|
|
|
}
|
|
|
|
|
|
|
|
void convertLocalTimeToUtc(SceDateTime *time_utc, SceDateTime *time_local) {
|
|
|
|
SceRtcTick tick;
|
|
|
|
sceRtcGetTick(time_local, &tick);
|
|
|
|
sceRtcConvertLocalTimeToUtc(&tick, &tick);
|
|
|
|
sceRtcSetTick(time_utc, &tick);
|
|
|
|
}
|
|
|
|
|
2016-08-23 12:50:39 +00:00
|
|
|
void getDateString(char *string, int date_format, SceDateTime *time) {
|
2016-11-01 08:53:35 +00:00
|
|
|
SceDateTime time_local;
|
|
|
|
convertUtcToLocalTime(&time_local, time);
|
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
switch (date_format) {
|
|
|
|
case SCE_SYSTEM_PARAM_DATE_FORMAT_YYYYMMDD:
|
2016-11-01 08:53:35 +00:00
|
|
|
sprintf(string, "%04d/%02d/%02d", time_local.year, time_local.month, time_local.day);
|
2016-08-06 06:59:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCE_SYSTEM_PARAM_DATE_FORMAT_DDMMYYYY:
|
2016-11-01 08:53:35 +00:00
|
|
|
sprintf(string, "%02d/%02d/%04d", time_local.day, time_local.month, time_local.year);
|
2016-08-06 06:59:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCE_SYSTEM_PARAM_DATE_FORMAT_MMDDYYYY:
|
2016-11-01 08:53:35 +00:00
|
|
|
sprintf(string, "%02d/%02d/%04d", time_local.month, time_local.day, time_local.year);
|
2016-08-06 06:59:41 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-23 12:50:39 +00:00
|
|
|
void getTimeString(char *string, int time_format, SceDateTime *time) {
|
2016-08-29 19:57:53 +00:00
|
|
|
SceDateTime time_local;
|
2016-11-01 08:53:35 +00:00
|
|
|
convertUtcToLocalTime(&time_local, time);
|
2016-08-29 19:57:53 +00:00
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
switch(time_format) {
|
|
|
|
case SCE_SYSTEM_PARAM_TIME_FORMAT_12HR:
|
2016-08-29 19:41:13 +00:00
|
|
|
sprintf(string, "%02d:%02d %s", (time_local.hour > 12) ? (time_local.hour - 12) : ((time_local.hour == 0) ? 12 : time_local.hour), time_local.minute, time_local.hour >= 12 ? "PM" : "AM");
|
2016-08-06 06:59:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCE_SYSTEM_PARAM_TIME_FORMAT_24HR:
|
2016-08-29 19:41:13 +00:00
|
|
|
sprintf(string, "%02d:%02d", time_local.hour, time_local.minute);
|
2016-08-06 06:59:41 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
int debugPrintf(const char *text, ...) {
|
2016-08-06 06:59:41 +00:00
|
|
|
va_list list;
|
|
|
|
char string[512];
|
|
|
|
|
|
|
|
va_start(list, text);
|
|
|
|
vsprintf(string, text, list);
|
|
|
|
va_end(list);
|
|
|
|
|
2016-12-11 21:30:07 +00:00
|
|
|
SceUID fd = sceIoOpen("ux0:data/vitashell_log.txt", SCE_O_WRONLY | SCE_O_CREAT | SCE_O_APPEND, 0777);
|
2016-08-06 06:59:41 +00:00
|
|
|
if (fd >= 0) {
|
|
|
|
sceIoWrite(fd, string, strlen(string));
|
|
|
|
sceIoClose(fd);
|
|
|
|
}
|
|
|
|
|
2016-09-04 10:55:15 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-12 15:27:43 +00:00
|
|
|
int launchAppByUriExit(const char *titleid) {
|
2016-09-04 10:55:15 +00:00
|
|
|
char uri[32];
|
|
|
|
sprintf(uri, "psgm:play?titleid=%s", titleid);
|
|
|
|
|
|
|
|
sceKernelDelayThread(10000);
|
|
|
|
sceAppMgrLaunchAppByUri(0xFFFFF, uri);
|
|
|
|
sceKernelDelayThread(10000);
|
|
|
|
sceAppMgrLaunchAppByUri(0xFFFFF, uri);
|
|
|
|
|
|
|
|
sceKernelExitProcess(0);
|
|
|
|
|
2016-08-06 06:59:41 +00:00
|
|
|
return 0;
|
2016-09-10 11:28:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
char *strcasestr(const char *haystack, const char *needle) {
|
|
|
|
return boyer_moore(haystack, needle);
|
2016-10-18 12:35:35 +00:00
|
|
|
}
|
2017-01-12 16:45:52 +00:00
|
|
|
|
|
|
|
int vshIoUmount(int id, int a2, int a3, int a4);
|
|
|
|
int _vshIoMount(int id, int a2, int permission, void *buf);
|
|
|
|
|
|
|
|
int vshIoMount(int id, int a2, int permission, int a4, int a5, int a6) {
|
|
|
|
uint32_t buf[6];
|
|
|
|
|
|
|
|
buf[0] = a4;
|
|
|
|
buf[1] = a5;
|
|
|
|
buf[2] = a6;
|
|
|
|
buf[3] = 0;
|
|
|
|
buf[4] = 0;
|
|
|
|
buf[5] = 0;
|
|
|
|
|
|
|
|
return _vshIoMount(id, a2, permission, buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
SceUID startUsb(const char *usbDevicePath, const char *imgFilePath, int type) {
|
|
|
|
int res;
|
|
|
|
|
|
|
|
// Load and start usbdevice module
|
|
|
|
SceUID modid = taiLoadStartKernelModule(usbDevicePath, 0, NULL, 0);
|
|
|
|
if (modid < 0)
|
|
|
|
return modid;
|
|
|
|
|
|
|
|
// Stop MTP driver
|
|
|
|
res = sceMtpIfStopDriver(1);
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Set device information
|
|
|
|
res = sceUsbstorVStorSetDeviceInfo("\"PS Vita\" MC", "1.00");
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Set image file path
|
|
|
|
res = sceUsbstorVStorSetImgFilePath(imgFilePath);
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Start USB storage
|
|
|
|
res = sceUsbstorVStorStart(type);
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
return modid;
|
|
|
|
}
|
|
|
|
|
|
|
|
int stopUsb(SceUID modid) {
|
|
|
|
int res;
|
|
|
|
|
|
|
|
// Stop USB storage
|
|
|
|
res = sceUsbstorVStorStop();
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Start MTP driver
|
|
|
|
res = sceMtpIfStartDriver(1);
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Stop and unload usbdevice module
|
|
|
|
res = taiStopUnloadKernelModule(modid, 0, NULL, 0, NULL, NULL);
|
|
|
|
if (res < 0)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
// Remount
|
|
|
|
vshIoUmount(0x800, 0, 0, 0);
|
|
|
|
vshIoMount(0x800, 0, 0, 0, 0, 0);
|
|
|
|
|
|
|
|
// Trick
|
|
|
|
sceAppMgrLoadExec("app0:eboot.bin", NULL, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|