/* RetroArch - A frontend for libretro. * Copyright (C) 2011-2016 - Daniel De Matteis * * RetroArch 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 Found- * ation, either version 3 of the License, or (at your option) any later version. * * RetroArch 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 RetroArch. * If not, see . */ #include #include #include #include #include #include #include #include "tasks.h" #include "../input/input_config.h" #include "../input/input_overlay.h" #include "../configuration.h" #include "../verbosity.h" typedef struct { enum overlay_status state; enum overlay_image_transfer_status loading_status; config_file_t *conf; char *overlay_path; unsigned size; unsigned pos; unsigned pos_increment; struct overlay *overlays; struct overlay *active; size_t resolve_pos; } overlay_loader_t; static void rarch_task_overlay_resolve_iterate(overlay_loader_t *loader); static void rarch_task_overlay_image_done(struct overlay *overlay) { overlay->pos = 0; /* Divide iteration steps by half of total descs if size is even, * otherwise default to 8 (arbitrary value for now to speed things up). */ overlay->pos_increment = (overlay->size / 2) ? (overlay->size / 2) : 8; } static bool rarch_task_overlay_load_texture_image(struct overlay *overlay, struct texture_image *image, const char *path) { if (!image) return false; if (!video_texture_image_load(image, path)) return false; overlay->load_images[overlay->load_images_size++] = *image; return true; } static void rarch_task_overlay_load_desc_image( overlay_loader_t *loader, struct overlay_desc *desc, struct overlay *input_overlay, unsigned ol_idx, unsigned desc_idx) { char overlay_desc_image_key[64] = {0}; char image_path[PATH_MAX_LENGTH] = {0}; config_file_t *conf = loader->conf; snprintf(overlay_desc_image_key, sizeof(overlay_desc_image_key), "overlay%u_desc%u_overlay", ol_idx, desc_idx); if (config_get_path(conf, overlay_desc_image_key, image_path, sizeof(image_path))) { char path[PATH_MAX_LENGTH] = {0}; fill_pathname_resolve_relative(path, loader->overlay_path, image_path, sizeof(path)); if (rarch_task_overlay_load_texture_image(input_overlay, &desc->image, path)) desc->image_index = input_overlay->load_images_size - 1; } input_overlay->pos ++; } static bool rarch_task_overlay_load_desc( overlay_loader_t *loader, struct overlay_desc *desc, struct overlay *input_overlay, unsigned ol_idx, unsigned desc_idx, unsigned width, unsigned height, bool normalized, float alpha_mod, float range_mod) { float width_mod, height_mod; uint32_t box_hash, key_hash; bool ret = true; bool by_pixel = false; char overlay_desc_key[64] = {0}; char conf_key[64] = {0}; char overlay_desc_normalized_key[64] = {0}; char overlay[256] = {0}; char *save = NULL; char *key = NULL; struct string_list *list = NULL; const char *x = NULL; const char *y = NULL; const char *box = NULL; config_file_t *conf = loader->conf; snprintf(overlay_desc_key, sizeof(overlay_desc_key), "overlay%u_desc%u", ol_idx, desc_idx); snprintf(overlay_desc_normalized_key, sizeof(overlay_desc_normalized_key), "overlay%u_desc%u_normalized", ol_idx, desc_idx); config_get_bool(conf, overlay_desc_normalized_key, &normalized); by_pixel = !normalized; if (by_pixel && (width == 0 || height == 0)) { RARCH_ERR("[Overlay]: Base overlay is not set and not using normalized coordinates.\n"); goto error; } if (!config_get_array(conf, overlay_desc_key, overlay, sizeof(overlay))) { RARCH_ERR("[Overlay]: Didn't find key: %s.\n", overlay_desc_key); goto error; } list = string_split(overlay, ", "); if (!list) { RARCH_ERR("[Overlay]: Failed to split overlay desc.\n"); goto error; } if (list->size < 6) { RARCH_ERR("[Overlay]: Overlay desc is invalid. Requires at least 6 tokens.\n"); goto error; } key = list->elems[0].data; x = list->elems[1].data; y = list->elems[2].data; box = list->elems[3].data; box_hash = djb2_calculate(box); key_hash = djb2_calculate(key); desc->key_mask = 0; switch (key_hash) { case KEY_ANALOG_LEFT: desc->type = OVERLAY_TYPE_ANALOG_LEFT; break; case KEY_ANALOG_RIGHT: desc->type = OVERLAY_TYPE_ANALOG_RIGHT; break; default: if (strstr(key, "retrok_") == key) { desc->type = OVERLAY_TYPE_KEYBOARD; desc->key_mask = input_config_translate_str_to_rk(key + 7); } else { const char *tmp = NULL; desc->type = OVERLAY_TYPE_BUTTONS; tmp = strtok_r(key, "|", &save); for (; tmp; tmp = strtok_r(NULL, "|", &save)) { if (!string_is_equal(tmp, "nul")) desc->key_mask |= UINT64_C(1) << input_config_translate_str_to_bind_id(tmp); } if (desc->key_mask & (UINT64_C(1) << RARCH_OVERLAY_NEXT)) { char overlay_target_key[64]; snprintf(overlay_target_key, sizeof(overlay_target_key), "overlay%u_desc%u_next_target", ol_idx, desc_idx); config_get_array(conf, overlay_target_key, desc->next_index_name, sizeof(desc->next_index_name)); } } break; } width_mod = 1.0f; height_mod = 1.0f; if (by_pixel) { width_mod /= width; height_mod /= height; } desc->x = (float)strtod(x, NULL) * width_mod; desc->y = (float)strtod(y, NULL) * height_mod; switch (box_hash) { case BOX_RADIAL: desc->hitbox = OVERLAY_HITBOX_RADIAL; break; case BOX_RECT: desc->hitbox = OVERLAY_HITBOX_RECT; break; default: RARCH_ERR("[Overlay]: Hitbox type (%s) is invalid. Use \"radial\" or \"rect\".\n", box); ret = false; goto end; } switch (desc->type) { case OVERLAY_TYPE_ANALOG_LEFT: case OVERLAY_TYPE_ANALOG_RIGHT: { char overlay_analog_saturate_key[64] = {0}; if (desc->hitbox != OVERLAY_HITBOX_RADIAL) { RARCH_ERR("[Overlay]: Analog hitbox type must be \"radial\".\n"); ret = false; goto end; } snprintf(overlay_analog_saturate_key, sizeof(overlay_analog_saturate_key), "overlay%u_desc%u_saturate_pct", ol_idx, desc_idx); if (!config_get_float(conf, overlay_analog_saturate_key, &desc->analog_saturate_pct)) desc->analog_saturate_pct = 1.0f; } break; default: /* OVERLAY_TYPE_BUTTONS - unhandled */ /* OVERLAY_TYPE_KEYBOARD - unhandled */ break; } desc->range_x = (float)strtod(list->elems[4].data, NULL) * width_mod; desc->range_y = (float)strtod(list->elems[5].data, NULL) * height_mod; desc->mod_x = desc->x - desc->range_x; desc->mod_w = 2.0f * desc->range_x; desc->mod_y = desc->y - desc->range_y; desc->mod_h = 2.0f * desc->range_y; snprintf(conf_key, sizeof(conf_key), "overlay%u_desc%u_alpha_mod", ol_idx, desc_idx); desc->alpha_mod = alpha_mod; config_get_float(conf, conf_key, &desc->alpha_mod); snprintf(conf_key, sizeof(conf_key), "overlay%u_desc%u_range_mod", ol_idx, desc_idx); desc->range_mod = range_mod; config_get_float(conf, conf_key, &desc->range_mod); snprintf(conf_key, sizeof(conf_key), "overlay%u_desc%u_movable", ol_idx, desc_idx); desc->movable = false; desc->delta_x = 0.0f; desc->delta_y = 0.0f; config_get_bool(conf, conf_key, &desc->movable); desc->range_x_mod = desc->range_x; desc->range_y_mod = desc->range_y; input_overlay->pos ++; end: if (list) string_list_free(list); return ret; error: if (list) string_list_free(list); return false; } static void rarch_task_overlay_deferred_loading(overlay_loader_t *loader) { size_t i = 0; bool not_done = true; struct overlay *overlay = NULL; overlay = &loader->overlays[loader->pos]; not_done = loader->pos < loader->size; if (!not_done) { loader->state = OVERLAY_STATUS_DEFERRED_LOADING_RESOLVE; return; } switch (loader->loading_status) { case OVERLAY_IMAGE_TRANSFER_NONE: case OVERLAY_IMAGE_TRANSFER_BUSY: loader->loading_status = OVERLAY_IMAGE_TRANSFER_DONE; #if 0 break; #endif case OVERLAY_IMAGE_TRANSFER_DONE: rarch_task_overlay_image_done(&loader->overlays[loader->pos]); loader->loading_status = OVERLAY_IMAGE_TRANSFER_DESC_IMAGE_ITERATE; loader->overlays[loader->pos].pos = 0; break; case OVERLAY_IMAGE_TRANSFER_DESC_IMAGE_ITERATE: for (i = 0; i < overlay->pos_increment; i++) { if (overlay->pos < overlay->size) { rarch_task_overlay_load_desc_image(loader, &overlay->descs[overlay->pos], overlay, loader->pos, overlay->pos); } else { overlay->pos = 0; loader->loading_status = OVERLAY_IMAGE_TRANSFER_DESC_ITERATE; break; } } break; case OVERLAY_IMAGE_TRANSFER_DESC_ITERATE: for (i = 0; i < overlay->pos_increment; i++) { if (overlay->pos < overlay->size) { if (!rarch_task_overlay_load_desc(loader, &overlay->descs[overlay->pos], overlay, loader->pos, overlay->pos, overlay->image.width, overlay->image.height, overlay->config.normalized, overlay->config.alpha_mod, overlay->config.range_mod)) { RARCH_ERR("[Overlay]: Failed to load overlay descs for overlay #%u.\n", (unsigned)overlay->pos); goto error; } } else { overlay->pos = 0; loader->loading_status = OVERLAY_IMAGE_TRANSFER_DESC_DONE; break; } } break; case OVERLAY_IMAGE_TRANSFER_DESC_DONE: if (loader->pos == 0) rarch_task_overlay_resolve_iterate(loader); loader->pos += 1; loader->loading_status = OVERLAY_IMAGE_TRANSFER_NONE; break; case OVERLAY_IMAGE_TRANSFER_ERROR: goto error; } return; error: loader->state = OVERLAY_STATUS_DEFERRED_ERROR; } static void rarch_task_overlay_deferred_load(overlay_loader_t *loader) { unsigned i; config_file_t *conf = loader->conf; for (i = 0; i < loader->pos_increment; i++, loader->pos++) { char conf_key[64] = {0}; char overlay_full_screen_key[64] = {0}; struct overlay *overlay = NULL; bool to_cont = loader->pos < loader->size; if (!to_cont) { loader->pos = 0; loader->state = OVERLAY_STATUS_DEFERRED_LOADING; break; } overlay = &loader->overlays[loader->pos]; if (!overlay) continue; snprintf(overlay->config.descs.key, sizeof(overlay->config.descs.key), "overlay%u_descs", loader->pos); if (!config_get_uint(conf, overlay->config.descs.key, &overlay->config.descs.size)) { RARCH_ERR("[Overlay]: Failed to read number of descs from config key: %s.\n", overlay->config.descs.key); goto error; } overlay->descs = (struct overlay_desc*) calloc(overlay->config.descs.size, sizeof(*overlay->descs)); if (!overlay->descs) { RARCH_ERR("[Overlay]: Failed to allocate descs.\n"); goto error; } overlay->size = overlay->config.descs.size; snprintf(overlay_full_screen_key, sizeof(overlay_full_screen_key), "overlay%u_full_screen", loader->pos); overlay->full_screen = false; config_get_bool(conf, overlay_full_screen_key, &overlay->full_screen); overlay->config.normalized = false; overlay->config.alpha_mod = 1.0f; overlay->config.range_mod = 1.0f; snprintf(conf_key, sizeof(conf_key), "overlay%u_normalized", loader->pos); config_get_bool(conf, conf_key, &overlay->config.normalized); snprintf(conf_key, sizeof(conf_key), "overlay%u_alpha_mod", loader->pos); config_get_float(conf, conf_key, &overlay->config.alpha_mod); snprintf(conf_key, sizeof(conf_key), "overlay%u_range_mod", loader->pos); config_get_float(conf, conf_key, &overlay->config.range_mod); /* Precache load image array for simplicity. */ overlay->load_images = (struct texture_image*) calloc(1 + overlay->size, sizeof(struct texture_image)); if (!overlay->load_images) { RARCH_ERR("[Overlay]: Failed to allocate load_images.\n"); goto error; } snprintf(overlay->config.paths.key, sizeof(overlay->config.paths.key), "overlay%u_overlay", loader->pos); config_get_path(conf, overlay->config.paths.key, overlay->config.paths.path, sizeof(overlay->config.paths.path)); if (!string_is_empty(overlay->config.paths.path)) { char overlay_resolved_path[PATH_MAX_LENGTH] = {0}; fill_pathname_resolve_relative(overlay_resolved_path, loader->overlay_path, overlay->config.paths.path, sizeof(overlay_resolved_path)); if (!rarch_task_overlay_load_texture_image(overlay, &overlay->image, overlay_resolved_path)) { RARCH_ERR("[Overlay]: Failed to load image: %s.\n", overlay_resolved_path); loader->loading_status = OVERLAY_IMAGE_TRANSFER_ERROR; goto error; } } snprintf(overlay->config.names.key, sizeof(overlay->config.names.key), "overlay%u_name", loader->pos); config_get_array(conf, overlay->config.names.key, overlay->name, sizeof(overlay->name)); /* By default, we stretch the overlay out in full. */ overlay->x = overlay->y = 0.0f; overlay->w = overlay->h = 1.0f; snprintf(overlay->config.rect.key, sizeof(overlay->config.rect.key), "overlay%u_rect", loader->pos); if (config_get_array(conf, overlay->config.rect.key, overlay->config.rect.array, sizeof(overlay->config.rect.array))) { struct string_list *list = string_split(overlay->config.rect.array, ", "); if (!list || list->size < 4) { RARCH_ERR("[Overlay]: Failed to split rect \"%s\" into at least four tokens.\n", overlay->config.rect.array); string_list_free(list); goto error; } overlay->x = (float)strtod(list->elems[0].data, NULL); overlay->y = (float)strtod(list->elems[1].data, NULL); overlay->w = (float)strtod(list->elems[2].data, NULL); overlay->h = (float)strtod(list->elems[3].data, NULL); string_list_free(list); } /* Assume for now that scaling center is in the middle. * TODO: Make this configurable. */ overlay->block_scale = false; overlay->center_x = overlay->x + 0.5f * overlay->w; overlay->center_y = overlay->y + 0.5f * overlay->h; } return; error: loader->pos = 0; loader->state = OVERLAY_STATUS_DEFERRED_ERROR; } static ssize_t rarch_task_overlay_find_index(const struct overlay *ol, const char *name, size_t size) { size_t i; if (!ol) return -1; for (i = 0; i < size; i++) { if (string_is_equal(ol[i].name, name)) return i; } return -1; } static bool rarch_task_overlay_resolve_targets(struct overlay *ol, size_t idx, size_t size) { size_t i; struct overlay *current = NULL; current = (struct overlay*)&ol[idx]; for (i = 0; i < current->size; i++) { const char *next = current->descs[i].next_index_name; if (*next) { ssize_t next_idx = rarch_task_overlay_find_index(ol, next, size); if (next_idx < 0) { RARCH_ERR("[Overlay]: Couldn't find overlay called: \"%s\".\n", next); return false; } current->descs[i].next_index = next_idx; } else current->descs[i].next_index = (idx + 1) % size; } return true; } static void rarch_task_overlay_resolve_iterate(overlay_loader_t *loader) { bool not_done = true; not_done = loader->resolve_pos < loader->size; if (!not_done) { loader->state = OVERLAY_STATUS_DEFERRED_DONE; return; } if (!rarch_task_overlay_resolve_targets(loader->overlays, loader->resolve_pos, loader->size)) { RARCH_ERR("[Overlay]: Failed to resolve next targets.\n"); goto error; } if (loader->resolve_pos == 0) { loader->active = &loader->overlays[0]; #if 0 /* TODO: MOVE TO MAIN THREAD / CALLBACK */ input_overlay_load_active(loader->deferred.opacity); input_overlay_enable(loader->deferred.enable); #endif } loader->resolve_pos += 1; return; error: loader->state = OVERLAY_STATUS_DEFERRED_ERROR; } static void rarch_task_overlay_handler(rarch_task_t *task) { overlay_loader_t *loader = (overlay_loader_t*)task->state; overlay_task_data_t *data; switch (loader->state) { case OVERLAY_STATUS_DEFERRED_LOADING: rarch_task_overlay_deferred_loading(loader); break; case OVERLAY_STATUS_DEFERRED_LOAD: rarch_task_overlay_deferred_load(loader); break; case OVERLAY_STATUS_DEFERRED_LOADING_RESOLVE: rarch_task_overlay_resolve_iterate(loader); break; case OVERLAY_STATUS_DEFERRED_ERROR: task->error = strdup("Failed to load the overlay."); case OVERLAY_STATUS_DEFERRED_DONE: default: case OVERLAY_STATUS_NONE: goto task_finished; } if (task->cancelled) goto task_finished; return; task_finished: task->finished = true; if (task->cancelled) { unsigned i; if (task->error) free(task->error); task->error = strdup("Task cancelled."); for (i = 0; i < loader->size; i++) input_overlay_free_overlay(&loader->overlays[i]); free(loader->overlays); } else { data = (overlay_task_data_t*)calloc(1, sizeof(*data)); data->overlays = loader->overlays; data->size = loader->size; data->active = loader->active; task->task_data = data; } if (loader->overlay_path) free(loader->overlay_path); if (loader->conf) config_file_free(loader->conf); free(loader); } static bool rarch_task_push_overlay_load(const char *overlay_path, rarch_task_callback_t cb, void *user_data) { rarch_task_t *t = NULL; overlay_loader_t *loader = (overlay_loader_t*)calloc(1, sizeof(*loader)); config_file_t *conf = config_file_new(overlay_path); if (!conf || !loader) goto error; if (!config_get_uint(conf, "overlays", &loader->size)) { RARCH_ERR("overlays variable not defined in config.\n"); goto error; } loader->overlays = (struct overlay*) calloc(loader->size, sizeof(*loader->overlays)); if (!loader->overlays) goto error; loader->overlay_path = strdup(overlay_path); loader->conf = conf; loader->state = OVERLAY_STATUS_DEFERRED_LOAD; loader->pos_increment = (loader->size / 4) ? (loader->size / 4) : 4; t = (rarch_task_t*)calloc(1, sizeof(*t)); if (!t) goto error; t->handler = rarch_task_overlay_handler; t->state = loader; t->callback = cb; t->user_data = user_data; task_ctl(TASK_CTL_PUSH, t); return true; error: if (conf) config_file_free(conf); if (loader) { if (loader->overlay_path) free(loader->overlay_path); if (loader->overlays) free(loader->overlays); free(loader); } return false; } bool rarch_task_push_overlay_load_default( rarch_task_callback_t cb, void *user_data) { settings_t *settings = config_get_ptr(); bool osk_enable = input_driver_ctl(RARCH_INPUT_CTL_IS_OSK_ENABLED, NULL); if (osk_enable) { if (!*settings->osk.overlay) return false; } else { if (!*settings->input.overlay) return false; } return rarch_task_push_overlay_load( osk_enable ? settings->osk.overlay : settings->input.overlay, cb, user_data); }