mirror of
https://github.com/joel16/VitaShell.git
synced 2024-11-27 05:40:21 +00:00
259 lines
6.7 KiB
C
259 lines
6.7 KiB
C
/*
|
|
VitaShell
|
|
Copyright (C) 2015-2017, TheFloW
|
|
|
|
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 "qr.h"
|
|
|
|
#include <psp2/kernel/processmgr.h>
|
|
#include <psp2/kernel/threadmgr.h>
|
|
#include <psp2/camera.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
#include <quirc.h>
|
|
#include <vita2d.h>
|
|
|
|
#include "main.h"
|
|
#include "io_process.h"
|
|
#include "network_download.h"
|
|
#include "package_installer.h"
|
|
#include "archive.h"
|
|
#include "file.h"
|
|
#include "message_dialog.h"
|
|
#include "language.h"
|
|
#include "utils.h"
|
|
|
|
static int qr_enabled;
|
|
|
|
static struct quirc *qr;
|
|
static uint32_t* qr_data;
|
|
static int qr_next;
|
|
|
|
static vita2d_texture *camera_tex;
|
|
|
|
static SceCameraInfo cam_info;
|
|
static SceCameraRead cam_info_read;
|
|
|
|
static char last_qr[MAX_QR_LENGTH];
|
|
static char last_download[MAX_QR_LENGTH];
|
|
static int last_qr_len;
|
|
static int qr_scanned = 0;
|
|
|
|
static SceUID thid;
|
|
|
|
int qr_thread() {
|
|
qr = quirc_new();
|
|
quirc_resize(qr, CAM_WIDTH, CAM_HEIGHT);
|
|
qr_next = 1;
|
|
while (1) {
|
|
sceKernelDelayThread(10);
|
|
if (qr_next == 0 && qr_scanned == 0) {
|
|
uint8_t *image;
|
|
int w, h;
|
|
image = quirc_begin(qr, &w, &h);
|
|
uint32_t colourRGBA;
|
|
int i;
|
|
for (i = 0; i < w*h; i++) {
|
|
colourRGBA = qr_data[i];
|
|
image[i] = ((colourRGBA & 0x000000FF) + ((colourRGBA & 0x0000FF00) >> 8) + ((colourRGBA & 0x00FF0000) >> 16)) / 3;
|
|
}
|
|
quirc_end(qr);
|
|
int num_codes = quirc_count(qr);
|
|
if (num_codes > 0) {
|
|
struct quirc_code code;
|
|
struct quirc_data data;
|
|
quirc_decode_error_t err;
|
|
|
|
quirc_extract(qr, 0, &code);
|
|
err = quirc_decode(&code, &data);
|
|
if (err) {
|
|
} else {
|
|
memcpy(last_qr, data.payload, data.payload_len);
|
|
last_qr_len = data.payload_len;
|
|
qr_scanned = 1;
|
|
}
|
|
} else {
|
|
memset(last_qr, 0, MAX_QR_LENGTH);
|
|
}
|
|
qr_next = 1;
|
|
sceKernelDelayThread(250000);
|
|
}
|
|
}
|
|
}
|
|
|
|
int qr_scan_thread(SceSize args, void *argp) {
|
|
if (last_qr_len > 4) {
|
|
if (!(last_qr[last_qr_len-4] == '.' && last_qr[last_qr_len-3] == 'v' && last_qr[last_qr_len-2] == 'p' && last_qr[last_qr_len-1] == 'k')) {
|
|
if (last_qr[0] == 'h' && last_qr[1] == 't' && last_qr[2] == 't' && last_qr[3] == 'p') {
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_YESNO, language_container[QR_OPEN_WEBSITE], last_qr);
|
|
setDialogStep(DIALOG_STEP_QR_OPEN_WEBSITE);
|
|
} else {
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_OK, language_container[QR_SHOW_CONTENTS], last_qr);
|
|
setDialogStep(DIALOG_STEP_QR_SHOW_CONTENTS);
|
|
}
|
|
return sceKernelExitDeleteThread(0);
|
|
}
|
|
} else {
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_OK, language_container[QR_SHOW_CONTENTS], last_qr);
|
|
setDialogStep(DIALOG_STEP_QR_SHOW_CONTENTS);
|
|
return sceKernelExitDeleteThread(0);
|
|
}
|
|
|
|
initMessageDialog(SCE_MSG_DIALOG_BUTTON_TYPE_YESNO, language_container[QR_CONFIRM_INSTALL], last_qr);
|
|
setDialogStep(DIALOG_STEP_QR_CONFIRM);
|
|
|
|
// Wait for response
|
|
while (getDialogStep() == DIALOG_STEP_QR_CONFIRM) {
|
|
sceKernelDelayThread(10 * 1000);
|
|
}
|
|
|
|
// No
|
|
if (getDialogStep() == DIALOG_STEP_NONE) {
|
|
goto EXIT;
|
|
}
|
|
|
|
// Yes
|
|
char download_path[MAX_URL_LENGTH];
|
|
char short_name[MAX_URL_LENGTH];
|
|
int count = 0;
|
|
|
|
char *next;
|
|
char *file_name = strdup(last_qr);
|
|
while ((next = strpbrk(file_name + 1, "\\/"))) file_name = next;
|
|
if (file_name != last_qr) file_name++;
|
|
|
|
char *ext = strrchr(file_name, '.');
|
|
if (ext) {
|
|
int len = ext-file_name;
|
|
if (len > sizeof(short_name)-1)
|
|
len = sizeof(short_name)-1;
|
|
strncpy(short_name, file_name, len);
|
|
short_name[len] = '\0';
|
|
} else {
|
|
strncpy(short_name, file_name, sizeof(short_name)-1);
|
|
ext = "";
|
|
}
|
|
while (1) {
|
|
if (count == 0)
|
|
snprintf(download_path, sizeof(download_path)-1, "ux0:download/qr/%s", file_name);
|
|
else
|
|
snprintf(download_path, sizeof(download_path)-1, "ux0:download/qr/%s (%d)%s", short_name, count, ext);
|
|
|
|
SceIoStat stat;
|
|
memset(&stat, 0, sizeof(SceIoStat));
|
|
if (sceIoGetstat(download_path, &stat) < 0)
|
|
break;
|
|
count++;
|
|
}
|
|
|
|
sceIoMkdir("ux0:download", 0006);
|
|
sceIoMkdir("ux0:download/qr", 0006);
|
|
|
|
strcpy(last_download, download_path);
|
|
return downloadFileProcess(last_qr, download_path, DIALOG_STEP_QR_DOWNLOADED);
|
|
|
|
EXIT:
|
|
return sceKernelExitDeleteThread(0);
|
|
}
|
|
|
|
int initQR() {
|
|
SceKernelMemBlockType orig = vita2d_texture_get_alloc_memblock_type();
|
|
vita2d_texture_set_alloc_memblock_type(SCE_KERNEL_MEMBLOCK_TYPE_USER_CDRAM_RW);
|
|
camera_tex = vita2d_create_empty_texture(CAM_WIDTH, CAM_HEIGHT);
|
|
vita2d_texture_set_alloc_memblock_type(orig);
|
|
|
|
cam_info.size = sizeof(SceCameraInfo);
|
|
cam_info.format = SCE_CAMERA_FORMAT_ABGR;
|
|
cam_info.resolution = SCE_CAMERA_RESOLUTION_640_360;
|
|
cam_info.pitch = vita2d_texture_get_stride(camera_tex) - (CAM_WIDTH << 2);
|
|
cam_info.sizeIBase = (CAM_WIDTH * CAM_HEIGHT) << 2;
|
|
cam_info.pIBase = vita2d_texture_get_datap(camera_tex);
|
|
cam_info.framerate = 30;
|
|
|
|
cam_info_read.size = sizeof(SceCameraRead);
|
|
cam_info_read.mode = 0;
|
|
if (sceCameraOpen(1, &cam_info) < 0) {
|
|
qr_enabled = 0;
|
|
vita2d_free_texture(camera_tex);
|
|
return -1;
|
|
}
|
|
|
|
thid = sceKernelCreateThread("qr_decode_thread", qr_thread, 0x40, 0x100000, 0, 0, NULL);
|
|
if (thid >= 0) sceKernelStartThread(thid, 0, NULL);
|
|
qr_enabled = 1;
|
|
return 0;
|
|
}
|
|
|
|
int finishQR() {
|
|
sceKernelDeleteThread(thid);
|
|
vita2d_free_texture(camera_tex);
|
|
sceCameraClose(1);
|
|
quirc_destroy(qr);
|
|
return 0;
|
|
}
|
|
|
|
int startQR() {
|
|
return sceCameraStart(1);
|
|
}
|
|
|
|
int stopQR() {
|
|
int res = sceCameraStop(1);
|
|
|
|
int y;
|
|
for (y = 0; y < CAM_HEIGHT; y++) {
|
|
int x;
|
|
for (x = 0; x < CAM_WIDTH; x++) {
|
|
((uint32_t *)qr_data)[x + 960 * y] = 0;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
int renderCameraQR(int x, int y) {
|
|
sceCameraRead(1, &cam_info_read);
|
|
vita2d_draw_texture(camera_tex, x, y);
|
|
if (qr_next) {
|
|
qr_data = (uint32_t *)vita2d_texture_get_datap(camera_tex);
|
|
qr_next = 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
char *getLastQR() {
|
|
return last_qr;
|
|
}
|
|
|
|
char *getLastDownloadQR() {
|
|
return last_download;
|
|
}
|
|
|
|
int scannedQR() {
|
|
return qr_scanned;
|
|
}
|
|
|
|
void setScannedQR(int s) {
|
|
qr_scanned = s;
|
|
}
|
|
|
|
int enabledQR() {
|
|
return qr_enabled;
|
|
} |