/* RetroArch - A frontend for libretro. * Copyright (C) 2011-2017 - 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 #ifdef HAVE_CONFIG_H #include "../../config.h" #endif #include "../menu_driver.h" #include "../menu_entries.h" #include "../menu_cbs.h" #include "../menu_input.h" #include "../menu_setting.h" #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) #include "../menu_shader.h" #endif #include "../../configuration.h" #include "../../core.h" #include "../../core_info.h" #ifdef HAVE_CHEATS #include "../../cheat_manager.h" #endif #include "../../file_path_special.h" #include "../../retroarch.h" #include "../../audio/audio_driver.h" #include "../../verbosity.h" #include "../../ui/ui_companion_driver.h" #ifdef HAVE_NETWORKING #include "../../network/netplay/netplay.h" #endif #include "../../playlist.h" #include "../../manual_content_scan.h" #include "../misc/cpufreq/cpufreq.h" #ifndef BIND_ACTION_RIGHT #define BIND_ACTION_RIGHT(cbs, name) (cbs)->action_right = (name) #endif /* Forward declarations */ int action_ok_core_lock(const char *path, const char *label, unsigned type, size_t idx, size_t entry_idx); int action_ok_core_set_standalone_exempt(const char *path, const char *label, unsigned type, size_t idx, size_t entry_idx); extern struct key_desc key_descriptors[RARCH_MAX_KEYS]; #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) static int generic_shader_action_parameter_right(struct video_shader_parameter *param, unsigned type, const char *label, bool wraparound) { param->current += param->step; param->current = MIN(MAX(param->minimum, param->current), param->maximum); return 0; } static int generic_shader_action_parameter_right_internal(unsigned type, const char *label, bool wraparound, unsigned offset) { video_shader_ctx_t shader_info; struct video_shader *shader = menu_shader_get(); struct video_shader_parameter *param_menu = NULL; struct video_shader_parameter *param_prev = NULL; int ret = 0; video_shader_driver_get_current_shader(&shader_info); param_prev = &shader_info.data->parameters[type - offset]; param_menu = shader ? &shader->parameters [type - offset] : NULL; if (!param_prev || !param_menu) return -1; ret = generic_shader_action_parameter_right(param_prev, type, label, wraparound); param_menu->current = param_prev->current; shader->flags |= SHDR_FLAG_MODIFIED; return ret; } int shader_action_parameter_right(unsigned type, const char *label, bool wraparound) { return generic_shader_action_parameter_right_internal(type, label, wraparound, MENU_SETTINGS_SHADER_PARAMETER_0); } int shader_action_preset_parameter_right(unsigned type, const char *label, bool wraparound) { return generic_shader_action_parameter_right_internal(type, label, wraparound, MENU_SETTINGS_SHADER_PRESET_PARAMETER_0); } #endif #ifdef HAVE_CHEATS int generic_action_cheat_toggle(size_t idx, unsigned type, const char *label, bool wraparound) { settings_t *settings = config_get_ptr(); bool apply_cheats_after_toggle = settings->bools.apply_cheats_after_toggle; cheat_manager_toggle_index( apply_cheats_after_toggle, (unsigned)idx); return 0; } int action_right_cheat(unsigned type, const char *label, bool wraparound) { size_t idx = type - MENU_SETTINGS_CHEAT_BEGIN; return generic_action_cheat_toggle(idx, type, label, wraparound); } static int action_right_cheat_num_passes(unsigned type, const char *label, bool wraparound) { bool refresh = false; unsigned new_size = 0; new_size = cheat_manager_get_size() + 1; menu_entries_ctl(MENU_ENTRIES_CTL_SET_REFRESH, &refresh); menu_driver_ctl(RARCH_MENU_CTL_SET_PREVENT_POPULATE, NULL); cheat_manager_realloc(new_size, CHEAT_HANDLER_TYPE_EMU); return 0; } #endif static int action_right_input_desc_kbd(unsigned type, const char *label, bool wraparound) { unsigned key_id, user_idx, btn_idx; unsigned remap_id; settings_t *settings = config_get_ptr(); if (!settings) return 0; user_idx = (type - MENU_SETTINGS_INPUT_DESC_KBD_BEGIN) / RARCH_ANALOG_BIND_LIST_END; btn_idx = (type - MENU_SETTINGS_INPUT_DESC_KBD_BEGIN) - RARCH_ANALOG_BIND_LIST_END * user_idx; remap_id = settings->uints.input_keymapper_ids[user_idx][btn_idx]; for (key_id = 0; key_id < RARCH_MAX_KEYS; key_id++) { if (remap_id == key_descriptors[key_id].key) break; } if (key_id < (RARCH_MAX_KEYS - 1)) key_id++; else key_id = 0; settings->uints.input_keymapper_ids[user_idx][btn_idx] = key_descriptors[key_id].key; return 0; } /* TODO/FIXME: incomplete, lacks error checking */ static int action_right_input_desc(unsigned type, const char *label, bool wraparound) { settings_t *settings = config_get_ptr(); rarch_system_info_t *system = &runloop_state_get_ptr()->system; if (settings && system) { unsigned bind_idx; unsigned user_idx = (type - MENU_SETTINGS_INPUT_DESC_BEGIN) / (RARCH_FIRST_CUSTOM_BIND + 8); unsigned btn_idx = (type - MENU_SETTINGS_INPUT_DESC_BEGIN) - (RARCH_FIRST_CUSTOM_BIND + 8) * user_idx; unsigned mapped_port = settings->uints.input_remap_ports[user_idx]; unsigned remap_idx = settings->uints.input_remap_ids[user_idx][btn_idx]; for (bind_idx = 0; bind_idx < RARCH_ANALOG_BIND_LIST_END; bind_idx++) { if (input_config_bind_order[bind_idx] == remap_idx) break; } if (bind_idx < RARCH_CUSTOM_BIND_LIST_END - 1) { if (bind_idx > RARCH_ANALOG_BIND_LIST_END) settings->uints.input_remap_ids[user_idx][btn_idx]++; else { if (bind_idx < RARCH_ANALOG_BIND_LIST_END - 1) { bind_idx++; bind_idx = input_config_bind_order[bind_idx]; } else if (bind_idx == RARCH_ANALOG_BIND_LIST_END - 1) bind_idx = RARCH_UNMAPPED; else bind_idx = input_config_bind_order[0]; settings->uints.input_remap_ids[user_idx][btn_idx] = bind_idx; } } else if (bind_idx == RARCH_CUSTOM_BIND_LIST_END - 1) settings->uints.input_remap_ids[user_idx][btn_idx] = RARCH_UNMAPPED; else settings->uints.input_remap_ids[user_idx][btn_idx] = input_config_bind_order[0]; remap_idx = settings->uints.input_remap_ids[user_idx][btn_idx]; /* skip the not used buttons (unless they are at the end by calling the right desc function recursively also skip all the axes until analog remapping is implemented */ if (remap_idx != RARCH_UNMAPPED) { if ((string_is_empty(system->input_desc_btn[mapped_port][remap_idx]) && remap_idx < RARCH_CUSTOM_BIND_LIST_END)) action_right_input_desc(type, label, wraparound); } } return 0; } static int action_right_scroll(unsigned type, const char *label, bool wraparound) { size_t scroll_accel = 0; if (menu_driver_ctl(MENU_NAVIGATION_CTL_GET_SCROLL_ACCEL, &scroll_accel)) { size_t selection = menu_navigation_get_selection(); unsigned scroll_speed = (unsigned)((MAX(scroll_accel, 2) - 2) / 4 + 1); unsigned fast_scroll_speed = 10 * scroll_speed; if (selection + fast_scroll_speed < (menu_entries_get_size())) { size_t idx = selection + fast_scroll_speed; menu_navigation_set_selection(idx); menu_driver_navigation_set(true); } else { if ((menu_entries_get_size() > 0)) menu_driver_ctl(MENU_NAVIGATION_CTL_SET_LAST, NULL); } } return 0; } #ifdef HAVE_AUDIOMIXER static int audio_mixer_stream_volume_right(unsigned type, const char *label, bool wraparound) { unsigned offset = (type - MENU_SETTINGS_AUDIO_MIXER_STREAM_ACTIONS_VOLUME_BEGIN); if (offset < AUDIO_MIXER_MAX_STREAMS) { float orig_volume = audio_driver_mixer_get_stream_volume(offset); orig_volume = orig_volume + 1.00f; audio_driver_mixer_set_stream_volume(offset, orig_volume); } return 0; } #endif static int action_right_goto_tab(void) { menu_ctx_list_t list_info; file_list_t *selection_buf = menu_entries_get_selection_buf_ptr(0); list_info.type = MENU_LIST_HORIZONTAL; list_info.action = MENU_ACTION_RIGHT; menu_driver_list_cache(&list_info); return menu_driver_deferred_push_content_list(selection_buf); } static int action_right_mainmenu(unsigned type, const char *label, bool wraparound) { menu_ctx_list_t list_info; settings_t *settings = config_get_ptr(); bool menu_nav_wraparound_enable = settings->bools.menu_navigation_wraparound_enable; const char *menu_ident = menu_driver_ident(); menu_driver_list_get_selection(&list_info); list_info.type = MENU_LIST_PLAIN; menu_driver_list_get_size(&list_info); /* Tab switching functionality only applies * to XMB */ if ((list_info.size == 1) && string_is_equal(menu_ident, "xmb")) { menu_ctx_list_t list_horiz_info; menu_ctx_list_t list_tabs_info; list_horiz_info.type = MENU_LIST_HORIZONTAL; list_tabs_info.type = MENU_LIST_TABS; menu_driver_list_get_size(&list_horiz_info); menu_driver_list_get_size(&list_tabs_info); if ((list_info.selection != (list_horiz_info.size + list_tabs_info.size)) || menu_nav_wraparound_enable) return action_right_goto_tab(); } else action_right_scroll(0, "", false); return 0; } #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) static int action_right_shader_scale_pass(unsigned type, const char *label, bool wraparound) { unsigned current_scale, delta; unsigned pass = type - MENU_SETTINGS_SHADER_PASS_SCALE_0; struct video_shader *shader = menu_shader_get(); struct video_shader_pass *shader_pass = shader ? &shader->pass[pass] : NULL; if (!shader_pass) return -1; /* A 20x scale is used to support scaling handheld border shaders up to 8K resolutions */ current_scale = shader_pass->fbo.scale_x; delta = 1; current_scale = (current_scale + delta) % 21; shader_pass->fbo.scale_x = shader_pass->fbo.scale_y = current_scale; if (current_scale) shader_pass->fbo.flags |= FBO_SCALE_FLAG_VALID; else shader_pass->fbo.flags &= ~FBO_SCALE_FLAG_VALID; shader->flags |= SHDR_FLAG_MODIFIED; return 0; } static int action_right_shader_filter_pass(unsigned type, const char *label, bool wraparound) { unsigned pass = type - MENU_SETTINGS_SHADER_PASS_FILTER_0; unsigned delta = 1; struct video_shader *shader = menu_shader_get(); struct video_shader_pass *shader_pass = shader ? &shader->pass[pass] : NULL; if (!shader_pass) return -1; shader_pass->filter = ((shader_pass->filter + delta) % 3); shader->flags |= SHDR_FLAG_MODIFIED; return 0; } static int action_right_shader_filter_default(unsigned type, const char *label, bool wraparound) { rarch_setting_t *setting = menu_setting_find_enum(MENU_ENUM_LABEL_VIDEO_SMOOTH); if (!setting) return -1; return menu_action_handle_setting(setting, setting->type, MENU_ACTION_RIGHT, wraparound); } static int action_right_shader_num_passes(unsigned type, const char *label, bool wraparound) { bool refresh = false; struct video_shader *shader = menu_shader_get(); unsigned pass_count = shader ? shader->passes : 0; if (!shader) return -1; if (pass_count < GFX_MAX_SHADERS) shader->passes++; menu_entries_ctl(MENU_ENTRIES_CTL_SET_REFRESH, &refresh); menu_driver_ctl(RARCH_MENU_CTL_SET_PREVENT_POPULATE, NULL); video_shader_resolve_parameters(shader); shader->flags |= SHDR_FLAG_MODIFIED; return 0; } #endif static int action_right_video_gpu_index(unsigned type, const char *label, bool wraparound) { enum gfx_ctx_api api = video_context_driver_get_api(); switch (api) { #ifdef HAVE_VULKAN case GFX_CTX_VULKAN_API: { struct string_list *list = video_driver_get_gpu_api_devices(api); if (list) { settings_t *settings = config_get_ptr(); if (settings->ints.vulkan_gpu_index <(int)(list->size - 1)) settings->ints.vulkan_gpu_index++; else if (settings->ints.vulkan_gpu_index == (int)(list->size - 1)) settings->ints.vulkan_gpu_index = 0; } break; } #endif #ifdef HAVE_D3D10 case GFX_CTX_DIRECT3D10_API: { struct string_list *list = video_driver_get_gpu_api_devices(api); if (list) { settings_t *settings = config_get_ptr(); if (settings->ints.d3d10_gpu_index < (int)(list->size - 1)) settings->ints.d3d10_gpu_index++; else if (settings->ints.d3d10_gpu_index == (int)(list->size - 1)) settings->ints.d3d10_gpu_index = 0; } break; } #endif #ifdef HAVE_D3D11 case GFX_CTX_DIRECT3D11_API: { struct string_list *list = video_driver_get_gpu_api_devices(api); if (list) { settings_t *settings = config_get_ptr(); if (settings->ints.d3d11_gpu_index < (int)(list->size - 1)) settings->ints.d3d11_gpu_index++; else if (settings->ints.d3d11_gpu_index == (int)(list->size - 1)) settings->ints.d3d11_gpu_index = 0; } break; } #endif #ifdef HAVE_D3D12 case GFX_CTX_DIRECT3D12_API: { struct string_list *list = video_driver_get_gpu_api_devices(api); if (list) { settings_t *settings = config_get_ptr(); if (settings->ints.d3d12_gpu_index < (int)(list->size - 1)) settings->ints.d3d12_gpu_index++; else if (settings->ints.d3d12_gpu_index == (int)(list->size - 1)) settings->ints.d3d12_gpu_index = 0; } break; } #endif default: break; } return 0; } static int action_right_video_resolution(unsigned type, const char *label, bool wraparound) { video_driver_get_next_video_out(); return 0; } static int playlist_association_right(unsigned type, const char *label, bool wraparound) { char core_filename[PATH_MAX_LENGTH]; size_t i, next, current = 0; core_info_list_t *core_info_list = NULL; core_info_t *core_info = NULL; playlist_t *playlist = playlist_get_cached(); const char *default_core_path = playlist_get_default_core_path(playlist); bool default_core_set = false; core_filename[0] = '\0'; if (!playlist) return -1; core_info_get_list(&core_info_list); if (!core_info_list) return -1; /* Get current core path association */ if (!string_is_empty(default_core_path) && !string_is_equal(default_core_path, "DETECT")) { const char *default_core_filename = path_basename(default_core_path); if (!string_is_empty(default_core_filename)) { strlcpy(core_filename, default_core_filename, sizeof(core_filename)); default_core_set = true; } } /* Sort cores alphabetically */ core_info_qsort(core_info_list, CORE_INFO_LIST_SORT_DISPLAY_NAME); /* If a core is currently associated... */ if (default_core_set) { /* ...get its index */ for (i = 0; i < core_info_list->count; i++) { core_info = NULL; core_info = core_info_get(core_info_list, i); if (!core_info) continue; if (string_starts_with(core_filename, core_info->core_file_id.str)) current = i; } /* ...then increment it */ next = current + 1; if (next >= core_info_list->count) { if (wraparound || (core_info_list->count < 1)) { /* Unset core association (DETECT) */ next = 0; default_core_set = false; } else next = core_info_list->count - 1; } } /* If a core is *not* currently associated, * select first core in the list */ else { next = 0; default_core_set = true; } /* If a core is now associated, get new core info */ core_info = NULL; if (default_core_set) core_info = core_info_get(core_info_list, next); /* Update playlist */ playlist_set_default_core_path(playlist, core_info ? core_info->path : "DETECT"); playlist_set_default_core_name(playlist, core_info ? core_info->display_name : "DETECT"); playlist_write_file(playlist); return 0; } static int playlist_label_display_mode_right(unsigned type, const char *label, bool wraparound) { enum playlist_label_display_mode label_display_mode; playlist_t *playlist = playlist_get_cached(); if (!playlist) return -1; label_display_mode = playlist_get_label_display_mode(playlist); if (label_display_mode != LABEL_DISPLAY_MODE_KEEP_REGION_AND_DISC_INDEX) label_display_mode = (enum playlist_label_display_mode)((int)label_display_mode + 1); else if (wraparound) label_display_mode = LABEL_DISPLAY_MODE_DEFAULT; playlist_set_label_display_mode(playlist, label_display_mode); playlist_write_file(playlist); return 0; } static void playlist_thumbnail_mode_right(playlist_t *playlist, enum playlist_thumbnail_id thumbnail_id, bool wraparound) { enum playlist_thumbnail_mode thumbnail_mode = playlist_get_thumbnail_mode(playlist, thumbnail_id); if (thumbnail_mode < PLAYLIST_THUMBNAIL_MODE_BOXARTS) thumbnail_mode = (enum playlist_thumbnail_mode)((unsigned)thumbnail_mode + 1); else if (wraparound) thumbnail_mode = PLAYLIST_THUMBNAIL_MODE_DEFAULT; playlist_set_thumbnail_mode(playlist, thumbnail_id, thumbnail_mode); playlist_write_file(playlist); } static int playlist_right_thumbnail_mode_right(unsigned type, const char *label, bool wraparound) { playlist_t *playlist = playlist_get_cached(); if (!playlist) return -1; playlist_thumbnail_mode_right(playlist, PLAYLIST_THUMBNAIL_RIGHT, wraparound); return 0; } static int playlist_left_thumbnail_mode_right(unsigned type, const char *label, bool wraparound) { playlist_t *playlist = playlist_get_cached(); if (!playlist) return -1; playlist_thumbnail_mode_right(playlist, PLAYLIST_THUMBNAIL_LEFT, wraparound); return 0; } static int playlist_sort_mode_right(unsigned type, const char *label, bool wraparound) { enum playlist_sort_mode sort_mode; playlist_t *playlist = playlist_get_cached(); if (!playlist) return -1; sort_mode = playlist_get_sort_mode(playlist); if (sort_mode < PLAYLIST_SORT_MODE_OFF) sort_mode = (enum playlist_sort_mode)((int)sort_mode + 1); else if (wraparound) sort_mode = PLAYLIST_SORT_MODE_DEFAULT; playlist_set_sort_mode(playlist, sort_mode); playlist_write_file(playlist); return 0; } static int manual_content_scan_system_name_right(unsigned type, const char *label, bool wraparound) { settings_t *settings = config_get_ptr(); bool show_hidden_files = settings->bools.show_hidden_files; #ifdef HAVE_LIBRETRODB const char *path_content_database = settings->paths.path_content_database; struct string_list *system_name_list = manual_content_scan_get_menu_system_name_list( path_content_database, show_hidden_files); #else struct string_list *system_name_list = manual_content_scan_get_menu_system_name_list(NULL, show_hidden_files); #endif const char *current_system_name = NULL; enum manual_content_scan_system_name_type next_system_name_type = MANUAL_CONTENT_SCAN_SYSTEM_NAME_DATABASE; const char *next_system_name = NULL; unsigned current_index = 0; unsigned next_index = 0; unsigned i; if (!system_name_list) return -1; /* Get currently selected system name */ if (manual_content_scan_get_menu_system_name(¤t_system_name)) { /* Get index of currently selected system name */ for (i = 0; i < system_name_list->size; i++) { const char *system_name = system_name_list->elems[i].data; if (string_is_equal(current_system_name, system_name)) { current_index = i; break; } } /* Increment index */ next_index = current_index + 1; if (next_index >= system_name_list->size) { next_index = 0; if (!wraparound && system_name_list->size > 0) next_index = (unsigned)(system_name_list->size - 1); } } /* Get new system name parameters */ if (next_index == (unsigned)MANUAL_CONTENT_SCAN_SYSTEM_NAME_CONTENT_DIR) next_system_name_type = MANUAL_CONTENT_SCAN_SYSTEM_NAME_CONTENT_DIR; else if (next_index == (unsigned)MANUAL_CONTENT_SCAN_SYSTEM_NAME_CUSTOM) next_system_name_type = MANUAL_CONTENT_SCAN_SYSTEM_NAME_CUSTOM; next_system_name = system_name_list->elems[next_index].data; /* Set system name */ manual_content_scan_set_menu_system_name( next_system_name_type, next_system_name); /* Clean up */ string_list_free(system_name_list); return 0; } static int manual_content_scan_core_name_right(unsigned type, const char *label, bool wraparound) { struct string_list *core_name_list = manual_content_scan_get_menu_core_name_list(); const char *current_core_name = NULL; enum manual_content_scan_core_type next_core_type = MANUAL_CONTENT_SCAN_CORE_SET; const char *next_core_name = NULL; unsigned current_index = 0; unsigned next_index = 0; unsigned i; if (!core_name_list) return -1; /* Get currently selected core name */ if (manual_content_scan_get_menu_core_name(¤t_core_name)) { /* Get index of currently selected core name */ for (i = 0; i < core_name_list->size; i++) { const char *core_name = core_name_list->elems[i].data; if (string_is_equal(current_core_name, core_name)) { current_index = i; break; } } /* Increment index */ next_index = current_index + 1; if (next_index >= core_name_list->size) { next_index = 0; if (!wraparound && core_name_list->size > 0) next_index = (unsigned)(core_name_list->size - 1); } } /* Get new core name parameters */ if (next_index == (unsigned)MANUAL_CONTENT_SCAN_CORE_DETECT) next_core_type = MANUAL_CONTENT_SCAN_CORE_DETECT; next_core_name = core_name_list->elems[next_index].data; /* Set core name */ manual_content_scan_set_menu_core_name( next_core_type, next_core_name); /* Clean up */ string_list_free(core_name_list); return 0; } #ifndef HAVE_LAKKA_SWITCH #ifdef HAVE_LAKKA static int cpu_policy_mode_change(unsigned type, const char *label, bool wraparound) { bool refresh = false; enum cpu_scaling_mode mode = get_cpu_scaling_mode(NULL); if (mode != CPUSCALING_MANUAL) mode++; set_cpu_scaling_mode(mode, NULL); menu_entries_ctl(MENU_ENTRIES_CTL_SET_REFRESH, &refresh); return 0; } static int cpu_policy_freq_managed_tweak(unsigned type, const char *label, bool wraparound) { bool refresh = false; cpu_scaling_opts_t opts; enum cpu_scaling_mode mode = get_cpu_scaling_mode(&opts); switch (type) { case MENU_SETTINGS_CPU_MANAGED_SET_MINFREQ: opts.min_freq = get_cpu_scaling_next_frequency_limit( opts.min_freq, 1); set_cpu_scaling_mode(mode, &opts); break; case MENU_SETTINGS_CPU_MANAGED_SET_MAXFREQ: opts.max_freq = get_cpu_scaling_next_frequency_limit( opts.max_freq, 1); set_cpu_scaling_mode(mode, &opts); break; } return 0; } static int cpu_policy_freq_managed_gov(unsigned type, const char *label, bool wraparound) { int pidx; bool refresh = false; cpu_scaling_opts_t opts; enum cpu_scaling_mode mode = get_cpu_scaling_mode(&opts); cpu_scaling_driver_t **drivers = get_cpu_scaling_drivers(false); /* Using drivers[0] governors, should be improved */ if (!drivers || !drivers[0]) return -1; switch (atoi(label)) { case 0: pidx = string_list_find_elem(drivers[0]->available_governors, opts.main_policy); if (pidx && pidx + 1 < drivers[0]->available_governors->size) { strlcpy(opts.main_policy, drivers[0]->available_governors->elems[pidx].data, sizeof(opts.main_policy)); set_cpu_scaling_mode(mode, &opts); } break; case 1: pidx = string_list_find_elem(drivers[0]->available_governors, opts.menu_policy); if (pidx && pidx + 1 < drivers[0]->available_governors->size) { strlcpy(opts.menu_policy, drivers[0]->available_governors->elems[pidx].data, sizeof(opts.menu_policy)); set_cpu_scaling_mode(mode, &opts); } break; } return 0; } static int cpu_policy_freq_tweak(unsigned type, const char *label, bool wraparound) { cpu_scaling_driver_t **drivers = get_cpu_scaling_drivers(false); if (drivers) { uint32_t next_freq; unsigned policyid = atoi(label); switch (type) { case MENU_SETTINGS_CPU_POLICY_SET_MINFREQ: next_freq = get_cpu_scaling_next_frequency(drivers[policyid], drivers[policyid]->min_policy_freq, 1); set_cpu_scaling_min_frequency(drivers[policyid], next_freq); break; case MENU_SETTINGS_CPU_POLICY_SET_MAXFREQ: next_freq = get_cpu_scaling_next_frequency(drivers[policyid], drivers[policyid]->max_policy_freq, 1); set_cpu_scaling_max_frequency(drivers[policyid], next_freq); break; case MENU_SETTINGS_CPU_POLICY_SET_GOVERNOR: { int pidx = string_list_find_elem(drivers[policyid]->available_governors, drivers[policyid]->scaling_governor); if (pidx && pidx + 1 < drivers[policyid]->available_governors->size) { set_cpu_scaling_governor(drivers[policyid], drivers[policyid]->available_governors->elems[pidx].data); } break; } } } return 0; } #endif #endif int core_setting_right(unsigned type, const char *label, bool wraparound) { unsigned idx = type - MENU_SETTINGS_CORE_OPTION_START; retroarch_ctl(RARCH_CTL_CORE_OPTION_NEXT, &idx); return 0; } static int action_right_core_lock(unsigned type, const char *label, bool wraparound) { return action_ok_core_lock(label, label, type, 0, 0); } static int action_right_core_set_standalone_exempt(unsigned type, const char *label, bool wraparound) { return action_ok_core_set_standalone_exempt(label, label, type, 0, 0); } static int disk_options_disk_idx_right(unsigned type, const char *label, bool wraparound) { /* Note: Menu itself provides visual feedback - no * need to print info message to screen */ bool print_log = false; command_event(CMD_EVENT_DISK_NEXT, &print_log); return 0; } static int action_right_state_slot(unsigned type, const char *label, bool wraparound) { settings_t *settings = config_get_ptr(); settings->ints.state_slot++; if (settings->ints.state_slot > 999) settings->ints.state_slot = -1; menu_driver_ctl(RARCH_MENU_CTL_UPDATE_SAVESTATE_THUMBNAIL_PATH, NULL); menu_driver_ctl(RARCH_MENU_CTL_UPDATE_SAVESTATE_THUMBNAIL_IMAGE, NULL); return 0; } int bind_right_generic(unsigned type, const char *label, bool wraparound) { return menu_setting_set(type, MENU_ACTION_RIGHT, wraparound); } static int menu_cbs_init_bind_right_compare_type(menu_file_list_cbs_t *cbs, unsigned type, const char *menu_label) { #ifdef HAVE_CHEATS if (type >= MENU_SETTINGS_CHEAT_BEGIN && type <= MENU_SETTINGS_CHEAT_END) { BIND_ACTION_RIGHT(cbs, action_right_cheat); } else #endif #ifdef HAVE_AUDIOMIXER if (type >= MENU_SETTINGS_AUDIO_MIXER_STREAM_ACTIONS_VOLUME_BEGIN && type <= MENU_SETTINGS_AUDIO_MIXER_STREAM_ACTIONS_VOLUME_END) { BIND_ACTION_RIGHT(cbs, audio_mixer_stream_volume_right); } else #endif #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) if (type >= MENU_SETTINGS_SHADER_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PARAMETER_LAST) { BIND_ACTION_RIGHT(cbs, shader_action_parameter_right); } else if (type >= MENU_SETTINGS_SHADER_PRESET_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PRESET_PARAMETER_LAST) { BIND_ACTION_RIGHT(cbs, shader_action_preset_parameter_right); } else #endif if (type >= MENU_SETTINGS_INPUT_DESC_BEGIN && type <= MENU_SETTINGS_INPUT_DESC_END) { BIND_ACTION_RIGHT(cbs, action_right_input_desc); } else if (type >= MENU_SETTINGS_INPUT_DESC_KBD_BEGIN && type <= MENU_SETTINGS_INPUT_DESC_KBD_END) { BIND_ACTION_RIGHT(cbs, action_right_input_desc_kbd); } else if ((type >= MENU_SETTINGS_CORE_OPTION_START) && (type < MENU_SETTINGS_CHEEVOS_START)) { BIND_ACTION_RIGHT(cbs, core_setting_right); } else { switch (type) { case MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_INDEX: BIND_ACTION_RIGHT(cbs, disk_options_disk_idx_right); break; case FILE_TYPE_PLAIN: case FILE_TYPE_DIRECTORY: case FILE_TYPE_PARENT_DIRECTORY: case FILE_TYPE_CARCHIVE: case FILE_TYPE_IN_CARCHIVE: case FILE_TYPE_CORE: case FILE_TYPE_RDB: case FILE_TYPE_RDB_ENTRY: case FILE_TYPE_RPL_ENTRY: case FILE_TYPE_SHADER: case FILE_TYPE_SHADER_PRESET: case FILE_TYPE_IMAGE: case FILE_TYPE_OVERLAY: #ifdef HAVE_VIDEO_LAYOUT case FILE_TYPE_VIDEO_LAYOUT: #endif case FILE_TYPE_VIDEOFILTER: case FILE_TYPE_AUDIOFILTER: case FILE_TYPE_CONFIG: case FILE_TYPE_USE_DIRECTORY: case FILE_TYPE_PLAYLIST_ENTRY: case MENU_INFO_MESSAGE: case FILE_TYPE_DOWNLOAD_CORE: case FILE_TYPE_CHEAT: case FILE_TYPE_REMAP: case FILE_TYPE_MOVIE: case FILE_TYPE_MUSIC: case FILE_TYPE_IMAGEVIEWER: case FILE_TYPE_PLAYLIST_COLLECTION: case FILE_TYPE_DOWNLOAD_CORE_CONTENT: case FILE_TYPE_DOWNLOAD_CORE_SYSTEM_FILES: case FILE_TYPE_DOWNLOAD_THUMBNAIL_CONTENT: case FILE_TYPE_DOWNLOAD_URL: case FILE_TYPE_SCAN_DIRECTORY: case FILE_TYPE_MANUAL_SCAN_DIRECTORY: case FILE_TYPE_FONT: case FILE_TYPE_VIDEO_FONT: case MENU_SETTING_GROUP: case MENU_SETTINGS_CORE_INFO_NONE: if ( string_ends_with_size(menu_label, "_tab", strlen(menu_label), STRLEN_CONST("_tab")) || string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_HORIZONTAL_MENU)) ) { BIND_ACTION_RIGHT(cbs, action_right_mainmenu); break; } case MENU_SETTING_ACTION_RUN: case MENU_SETTING_ACTION_CLOSE: case MENU_SETTING_ACTION_CLOSE_HORIZONTAL: case MENU_SETTING_ACTION_DELETE_ENTRY: case MENU_SETTING_ACTION_CORE_OPTIONS: case MENU_SETTING_ACTION_CORE_DISK_OPTIONS: case MENU_SETTING_ACTION_SCREENSHOT: case MENU_SETTING_ACTION_FAVORITES_DIR: case MENU_SETTING_ACTION_CORE_MANAGER_OPTIONS: case MENU_SETTING_DROPDOWN_ITEM_INPUT_DESCRIPTION: case MENU_SETTING_DROPDOWN_ITEM_INPUT_DESCRIPTION_KBD: BIND_ACTION_RIGHT(cbs, action_right_scroll); break; case MENU_SETTING_ACTION: case FILE_TYPE_CONTENTLIST_ENTRY: BIND_ACTION_RIGHT(cbs, action_right_mainmenu); break; case MENU_SETTING_ACTION_CORE_LOCK: BIND_ACTION_RIGHT(cbs, action_right_core_lock); break; case MENU_SETTING_ACTION_CORE_SET_STANDALONE_EXEMPT: BIND_ACTION_RIGHT(cbs, action_right_core_set_standalone_exempt); break; case MENU_SETTING_ACTION_SAVESTATE: case MENU_SETTING_ACTION_LOADSTATE: BIND_ACTION_RIGHT(cbs, action_right_state_slot); break; default: return -1; } } return 0; } static int menu_cbs_init_bind_right_compare_label(menu_file_list_cbs_t *cbs, const char *label, const char *menu_label) { if (cbs->setting) { const char *parent_group = cbs->setting->parent_group; if (string_is_equal(parent_group, msg_hash_to_str(MENU_ENUM_LABEL_MAIN_MENU)) && (cbs->setting->type == ST_GROUP)) { BIND_ACTION_RIGHT(cbs, action_right_scroll); return 0; } } if ( string_starts_with_size(label, "input_player", STRLEN_CONST("input_player")) && string_ends_with_size(label, "_joypad_index", strlen(label), STRLEN_CONST("_joypad_index"))) { unsigned i; for (i = 0; i < MAX_USERS; i++) { char label_setting[128]; label_setting[0] = '\0'; snprintf(label_setting, sizeof(label_setting), "input_player%d_joypad_index", i + 1); if (!string_is_equal(label, label_setting)) continue; BIND_ACTION_RIGHT(cbs, bind_right_generic); return 0; } } if (string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_PLAYLISTS_TAB))) { BIND_ACTION_RIGHT(cbs, action_right_mainmenu); return 0; } if ( strstr(label, "rdb_entry") || string_starts_with_size(label, "content_info", STRLEN_CONST("content_info"))) { BIND_ACTION_RIGHT(cbs, action_right_scroll); } else { if (cbs->enum_idx != MSG_UNKNOWN) { switch (cbs->enum_idx) { case MENU_ENUM_LABEL_SUBSYSTEM_ADD: case MENU_ENUM_LABEL_SUBSYSTEM_LOAD: case MENU_ENUM_LABEL_CONNECT_NETPLAY_ROOM: case MENU_ENUM_LABEL_EXPLORE_ITEM: case MENU_ENUM_LABEL_CONTENTLESS_CORE: case MENU_ENUM_LABEL_NO_SETTINGS_FOUND: BIND_ACTION_RIGHT(cbs, action_right_mainmenu); break; case MENU_ENUM_LABEL_VIDEO_SHADER_SCALE_PASS: #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) BIND_ACTION_RIGHT(cbs, action_right_shader_scale_pass); #endif break; case MENU_ENUM_LABEL_VIDEO_SHADER_FILTER_PASS: #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) BIND_ACTION_RIGHT(cbs, action_right_shader_filter_pass); #endif break; case MENU_ENUM_LABEL_VIDEO_SHADER_DEFAULT_FILTER: #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) BIND_ACTION_RIGHT(cbs, action_right_shader_filter_default); #endif break; case MENU_ENUM_LABEL_VIDEO_SHADER_NUM_PASSES: #if defined(HAVE_CG) || defined(HAVE_GLSL) || defined(HAVE_SLANG) || defined(HAVE_HLSL) BIND_ACTION_RIGHT(cbs, action_right_shader_num_passes); #endif break; case MENU_ENUM_LABEL_CHEAT_NUM_PASSES: #ifdef HAVE_CHEATS BIND_ACTION_RIGHT(cbs, action_right_cheat_num_passes); #endif break; case MENU_ENUM_LABEL_SCREEN_RESOLUTION: BIND_ACTION_RIGHT(cbs, action_right_video_resolution); break; case MENU_ENUM_LABEL_OPEN_ARCHIVE_DETECT_CORE: case MENU_ENUM_LABEL_LOAD_ARCHIVE_DETECT_CORE: BIND_ACTION_RIGHT(cbs, action_right_scroll); break; case MENU_ENUM_LABEL_NO_ITEMS: case MENU_ENUM_LABEL_NO_PLAYLIST_ENTRIES_AVAILABLE: case MENU_ENUM_LABEL_NO_CORES_AVAILABLE: case MENU_ENUM_LABEL_EXPLORE_INITIALISING_LIST: if ( string_ends_with_size(menu_label, "_tab", strlen(menu_label), STRLEN_CONST("_tab") ) || string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_MAIN_MENU)) || string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_HORIZONTAL_MENU)) ) { BIND_ACTION_RIGHT(cbs, action_right_mainmenu); } else { BIND_ACTION_RIGHT(cbs, action_right_scroll); } break; case MENU_ENUM_LABEL_VIDEO_GPU_INDEX: BIND_ACTION_RIGHT(cbs, action_right_video_gpu_index); break; case MENU_ENUM_LABEL_PLAYLIST_MANAGER_DEFAULT_CORE: BIND_ACTION_RIGHT(cbs, playlist_association_right); break; case MENU_ENUM_LABEL_PLAYLIST_MANAGER_LABEL_DISPLAY_MODE: BIND_ACTION_RIGHT(cbs, playlist_label_display_mode_right); break; case MENU_ENUM_LABEL_PLAYLIST_MANAGER_RIGHT_THUMBNAIL_MODE: BIND_ACTION_RIGHT(cbs, playlist_right_thumbnail_mode_right); break; case MENU_ENUM_LABEL_PLAYLIST_MANAGER_LEFT_THUMBNAIL_MODE: BIND_ACTION_RIGHT(cbs, playlist_left_thumbnail_mode_right); break; case MENU_ENUM_LABEL_PLAYLIST_MANAGER_SORT_MODE: BIND_ACTION_RIGHT(cbs, playlist_sort_mode_right); break; case MENU_ENUM_LABEL_MANUAL_CONTENT_SCAN_SYSTEM_NAME: BIND_ACTION_RIGHT(cbs, manual_content_scan_system_name_right); break; case MENU_ENUM_LABEL_MANUAL_CONTENT_SCAN_CORE_NAME: BIND_ACTION_RIGHT(cbs, manual_content_scan_core_name_right); break; #ifndef HAVE_LAKKA_SWITCH #ifdef HAVE_LAKKA case MENU_ENUM_LABEL_CPU_PERF_MODE: BIND_ACTION_RIGHT(cbs, cpu_policy_mode_change); break; case MENU_ENUM_LABEL_CPU_POLICY_MAX_FREQ: case MENU_ENUM_LABEL_CPU_POLICY_MIN_FREQ: case MENU_ENUM_LABEL_CPU_POLICY_GOVERNOR: BIND_ACTION_RIGHT(cbs, cpu_policy_freq_tweak); break; case MENU_ENUM_LABEL_CPU_MANAGED_MIN_FREQ: case MENU_ENUM_LABEL_CPU_MANAGED_MAX_FREQ: BIND_ACTION_RIGHT(cbs, cpu_policy_freq_managed_tweak); break; case MENU_ENUM_LABEL_CPU_POLICY_CORE_GOVERNOR: case MENU_ENUM_LABEL_CPU_POLICY_MENU_GOVERNOR: BIND_ACTION_RIGHT(cbs, cpu_policy_freq_managed_gov); break; #endif #endif default: return -1; } } else return -1; } return 0; } int menu_cbs_init_bind_right(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx, const char *menu_label) { if (!cbs) return -1; BIND_ACTION_RIGHT(cbs, bind_right_generic); if (type == MENU_SETTING_NO_ITEM) { if ( string_ends_with_size(menu_label, "_tab", strlen(menu_label), STRLEN_CONST("_tab")) || string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_MAIN_MENU)) || string_is_equal(menu_label, msg_hash_to_str(MENU_ENUM_LABEL_HORIZONTAL_MENU)) ) { BIND_ACTION_RIGHT(cbs, action_right_mainmenu); return 0; } } if (menu_cbs_init_bind_right_compare_label(cbs, label, menu_label ) == 0) return 0; if (menu_cbs_init_bind_right_compare_type(cbs, type, menu_label ) == 0) return 0; return -1; }