mirror of
https://gitlab.com/Theopse/fbi-i18n-zh.git
synced 2025-04-06 03:58:02 +08:00
* Maintain list position after operations. * Show current target info in batch operations. * Fix freeze when suspending to the home menu during a threaded task. * Other miscellaneous fixes and clean-ups.
155 lines
5.4 KiB
C
155 lines
5.4 KiB
C
#include <malloc.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <3ds.h>
|
|
|
|
#include "task.h"
|
|
#include "../../list.h"
|
|
#include "../../error.h"
|
|
#include "../../../core/linkedlist.h"
|
|
#include "../../../core/screen.h"
|
|
#include "../../../core/util.h"
|
|
|
|
typedef struct {
|
|
linked_list* items;
|
|
|
|
Handle cancelEvent;
|
|
} populate_pending_titles_data;
|
|
|
|
static Result task_populate_pending_titles_from(populate_pending_titles_data* data, FS_MediaType mediaType) {
|
|
Result res = 0;
|
|
|
|
u32 pendingTitleCount = 0;
|
|
if(R_SUCCEEDED(res = AM_GetPendingTitleCount(&pendingTitleCount, mediaType, AM_STATUS_MASK_INSTALLING | AM_STATUS_MASK_AWAITING_FINALIZATION))) {
|
|
u64* pendingTitleIds = (u64*) calloc(pendingTitleCount, sizeof(u64));
|
|
if(pendingTitleIds != NULL) {
|
|
if(R_SUCCEEDED(res = AM_GetPendingTitleList(&pendingTitleCount, pendingTitleCount, mediaType, AM_STATUS_MASK_INSTALLING | AM_STATUS_MASK_AWAITING_FINALIZATION, pendingTitleIds))) {
|
|
qsort(pendingTitleIds, pendingTitleCount, sizeof(u64), util_compare_u64);
|
|
|
|
AM_PendingTitleEntry* pendingTitleInfos = (AM_PendingTitleEntry*) calloc(pendingTitleCount, sizeof(AM_PendingTitleEntry));
|
|
if(pendingTitleInfos != NULL) {
|
|
if(R_SUCCEEDED(res = AM_GetPendingTitleInfo(pendingTitleCount, mediaType, pendingTitleIds, pendingTitleInfos))) {
|
|
for(u32 i = 0; i < pendingTitleCount && R_SUCCEEDED(res); i++) {
|
|
svcWaitSynchronization(task_get_pause_event(), U64_MAX);
|
|
if(task_is_quit_all() || svcWaitSynchronization(data->cancelEvent, 0) == 0) {
|
|
break;
|
|
}
|
|
|
|
list_item* item = (list_item*) calloc(1, sizeof(list_item));
|
|
if(item != NULL) {
|
|
pending_title_info* pendingTitleInfo = (pending_title_info*) calloc(1, sizeof(pending_title_info));
|
|
if(pendingTitleInfo != NULL) {
|
|
pendingTitleInfo->mediaType = mediaType;
|
|
pendingTitleInfo->titleId = pendingTitleIds[i];
|
|
pendingTitleInfo->version = pendingTitleInfos[i].version;
|
|
|
|
snprintf(item->name, LIST_ITEM_NAME_MAX, "%016llX", pendingTitleIds[i]);
|
|
if(mediaType == MEDIATYPE_NAND) {
|
|
item->color = COLOR_NAND;
|
|
} else if(mediaType == MEDIATYPE_SD) {
|
|
item->color = COLOR_SD;
|
|
}
|
|
|
|
item->data = pendingTitleInfo;
|
|
|
|
linked_list_add(data->items, item);
|
|
} else {
|
|
free(item);
|
|
|
|
res = R_FBI_OUT_OF_MEMORY;
|
|
}
|
|
} else {
|
|
res = R_FBI_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
}
|
|
|
|
free(pendingTitleInfos);
|
|
} else {
|
|
res = R_FBI_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
|
|
free(pendingTitleIds);
|
|
} else {
|
|
res = R_FBI_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static void task_populate_pending_titles_thread(void* arg) {
|
|
populate_pending_titles_data* data = (populate_pending_titles_data*) arg;
|
|
|
|
Result res = 0;
|
|
if(R_FAILED(res = task_populate_pending_titles_from(data, MEDIATYPE_SD)) || R_FAILED(res = task_populate_pending_titles_from(data, MEDIATYPE_NAND))) {
|
|
error_display_res(NULL, NULL, NULL, res, "Failed to load pending title listing.");
|
|
}
|
|
|
|
svcCloseHandle(data->cancelEvent);
|
|
free(data);
|
|
}
|
|
|
|
void task_free_pending_title(list_item* item) {
|
|
if(item == NULL) {
|
|
return;
|
|
}
|
|
|
|
if(item->data != NULL) {
|
|
free(item->data);
|
|
}
|
|
|
|
free(item);
|
|
}
|
|
|
|
void task_clear_pending_titles(linked_list* items) {
|
|
if(items == NULL) {
|
|
return;
|
|
}
|
|
|
|
linked_list_iter iter;
|
|
linked_list_iterate(items, &iter);
|
|
|
|
while(linked_list_iter_has_next(&iter)) {
|
|
list_item* item = (list_item*) linked_list_iter_next(&iter);
|
|
task_free_pending_title(item);
|
|
linked_list_iter_remove(&iter);
|
|
}
|
|
}
|
|
|
|
Handle task_populate_pending_titles(linked_list* items) {
|
|
if(items == NULL) {
|
|
return 0;
|
|
}
|
|
|
|
task_clear_pending_titles(items);
|
|
|
|
populate_pending_titles_data* data = (populate_pending_titles_data*) calloc(1, sizeof(populate_pending_titles_data));
|
|
if(data == NULL) {
|
|
error_display(NULL, NULL, NULL, "Failed to allocate pending title list data.");
|
|
|
|
return 0;
|
|
}
|
|
|
|
data->items = items;
|
|
|
|
Result eventRes = svcCreateEvent(&data->cancelEvent, 1);
|
|
if(R_FAILED(eventRes)) {
|
|
error_display_res(NULL, NULL, NULL, eventRes, "Failed to create pending title list cancel event.");
|
|
|
|
free(data);
|
|
return 0;
|
|
}
|
|
|
|
if(threadCreate(task_populate_pending_titles_thread, data, 0x10000, 0x18, 1, true) == NULL) {
|
|
error_display(NULL, NULL, NULL, "Failed to create pending title list thread.");
|
|
|
|
svcCloseHandle(data->cancelEvent);
|
|
free(data);
|
|
return 0;
|
|
}
|
|
|
|
return data->cancelEvent;
|
|
} |