123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2025 Sam Lantinga <slouken@libsdl.org>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "SDL_internal.h"
- #include "SDL_hints_c.h"
- typedef struct SDL_HintWatch
- {
- SDL_HintCallback callback;
- void *userdata;
- struct SDL_HintWatch *next;
- } SDL_HintWatch;
- typedef struct SDL_Hint
- {
- char *value;
- SDL_HintPriority priority;
- SDL_HintWatch *callbacks;
- } SDL_Hint;
- static SDL_AtomicU32 SDL_hint_props;
- void SDL_InitHints(void)
- {
- }
- void SDL_QuitHints(void)
- {
- SDL_PropertiesID props;
- do {
- props = SDL_GetAtomicU32(&SDL_hint_props);
- } while (!SDL_CompareAndSwapAtomicU32(&SDL_hint_props, props, 0));
- if (props) {
- SDL_DestroyProperties(props);
- }
- }
- static SDL_PropertiesID GetHintProperties(bool create)
- {
- SDL_PropertiesID props = SDL_GetAtomicU32(&SDL_hint_props);
- if (!props && create) {
- props = SDL_CreateProperties();
- if (!SDL_CompareAndSwapAtomicU32(&SDL_hint_props, 0, props)) {
- // Somebody else created hint properties before us, just use those
- SDL_DestroyProperties(props);
- props = SDL_GetAtomicU32(&SDL_hint_props);
- }
- }
- return props;
- }
- static void SDLCALL CleanupHintProperty(void *userdata, void *value)
- {
- SDL_Hint *hint = (SDL_Hint *) value;
- SDL_free(hint->value);
- SDL_HintWatch *entry = hint->callbacks;
- while (entry) {
- SDL_HintWatch *freeable = entry;
- entry = entry->next;
- SDL_free(freeable);
- }
- SDL_free(hint);
- }
- static const char* GetHintEnvironmentVariable(const char *name)
- {
- const char *result = SDL_getenv(name);
- if (!result && name && *name) {
- // fall back to old (SDL2) names of environment variables that
- // are important to users (e.g. many use SDL_VIDEODRIVER=wayland)
- if (SDL_strcmp(name, SDL_HINT_VIDEO_DRIVER) == 0) {
- result = SDL_getenv("SDL_VIDEODRIVER");
- } else if (SDL_strcmp(name, SDL_HINT_AUDIO_DRIVER) == 0) {
- result = SDL_getenv("SDL_AUDIODRIVER");
- }
- }
- return result;
- }
- bool SDL_SetHintWithPriority(const char *name, const char *value, SDL_HintPriority priority)
- {
- if (!name || !*name) {
- return SDL_InvalidParamError("name");
- }
- const char *env = GetHintEnvironmentVariable(name);
- if (env && (priority < SDL_HINT_OVERRIDE)) {
- return SDL_SetError("An environment variable is taking priority");
- }
- const SDL_PropertiesID hints = GetHintProperties(true);
- if (!hints) {
- return false;
- }
- bool result = false;
- SDL_LockProperties(hints);
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (hint) {
- if (priority >= hint->priority) {
- if (hint->value != value && (!value || !hint->value || SDL_strcmp(hint->value, value) != 0)) {
- char *old_value = hint->value;
- hint->value = value ? SDL_strdup(value) : NULL;
- SDL_HintWatch *entry = hint->callbacks;
- while (entry) {
- // Save the next entry in case this one is deleted
- SDL_HintWatch *next = entry->next;
- entry->callback(entry->userdata, name, old_value, value);
- entry = next;
- }
- SDL_free(old_value);
- }
- hint->priority = priority;
- result = true;
- }
- } else { // Couldn't find the hint? Add a new one.
- hint = (SDL_Hint *)SDL_malloc(sizeof(*hint));
- if (hint) {
- hint->value = value ? SDL_strdup(value) : NULL;
- hint->priority = priority;
- hint->callbacks = NULL;
- result = SDL_SetPointerPropertyWithCleanup(hints, name, hint, CleanupHintProperty, NULL);
- }
- }
- SDL_UnlockProperties(hints);
- return result;
- }
- bool SDL_ResetHint(const char *name)
- {
- if (!name || !*name) {
- return SDL_InvalidParamError("name");
- }
- const char *env = GetHintEnvironmentVariable(name);
- const SDL_PropertiesID hints = GetHintProperties(false);
- if (!hints) {
- return false;
- }
- bool result = false;
- SDL_LockProperties(hints);
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (hint) {
- if ((!env && hint->value) || (env && !hint->value) || (env && SDL_strcmp(env, hint->value) != 0)) {
- for (SDL_HintWatch *entry = hint->callbacks; entry;) {
- // Save the next entry in case this one is deleted
- SDL_HintWatch *next = entry->next;
- entry->callback(entry->userdata, name, hint->value, env);
- entry = next;
- }
- }
- SDL_free(hint->value);
- hint->value = NULL;
- hint->priority = SDL_HINT_DEFAULT;
- result = true;
- }
- SDL_UnlockProperties(hints);
- return result;
- }
- static void SDLCALL ResetHintsCallback(void *userdata, SDL_PropertiesID hints, const char *name)
- {
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (!hint) {
- return; // uh...okay.
- }
- const char *env = GetHintEnvironmentVariable(name);
- if ((!env && hint->value) || (env && !hint->value) || (env && SDL_strcmp(env, hint->value) != 0)) {
- SDL_HintWatch *entry = hint->callbacks;
- while (entry) {
- // Save the next entry in case this one is deleted
- SDL_HintWatch *next = entry->next;
- entry->callback(entry->userdata, name, hint->value, env);
- entry = next;
- }
- }
- SDL_free(hint->value);
- hint->value = NULL;
- hint->priority = SDL_HINT_DEFAULT;
- }
- void SDL_ResetHints(void)
- {
- SDL_EnumerateProperties(GetHintProperties(false), ResetHintsCallback, NULL);
- }
- bool SDL_SetHint(const char *name, const char *value)
- {
- return SDL_SetHintWithPriority(name, value, SDL_HINT_NORMAL);
- }
- const char *SDL_GetHint(const char *name)
- {
- if (!name) {
- return NULL;
- }
- const char *result = GetHintEnvironmentVariable(name);
- const SDL_PropertiesID hints = GetHintProperties(false);
- if (hints) {
- SDL_LockProperties(hints);
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (hint) {
- if (!result || hint->priority == SDL_HINT_OVERRIDE) {
- result = SDL_GetPersistentString(hint->value);
- }
- }
- SDL_UnlockProperties(hints);
- }
- return result;
- }
- int SDL_GetStringInteger(const char *value, int default_value)
- {
- if (!value || !*value) {
- return default_value;
- }
- if (SDL_strcasecmp(value, "false") == 0) {
- return 0;
- }
- if (SDL_strcasecmp(value, "true") == 0) {
- return 1;
- }
- if (*value == '-' || SDL_isdigit(*value)) {
- return SDL_atoi(value);
- }
- return default_value;
- }
- bool SDL_GetStringBoolean(const char *value, bool default_value)
- {
- if (!value || !*value) {
- return default_value;
- }
- if (*value == '0' || SDL_strcasecmp(value, "false") == 0) {
- return false;
- }
- return true;
- }
- bool SDL_GetHintBoolean(const char *name, bool default_value)
- {
- const char *hint = SDL_GetHint(name);
- return SDL_GetStringBoolean(hint, default_value);
- }
- bool SDL_AddHintCallback(const char *name, SDL_HintCallback callback, void *userdata)
- {
- if (!name || !*name) {
- return SDL_InvalidParamError("name");
- } else if (!callback) {
- return SDL_InvalidParamError("callback");
- }
- const SDL_PropertiesID hints = GetHintProperties(true);
- if (!hints) {
- return false;
- }
- SDL_HintWatch *entry = (SDL_HintWatch *)SDL_malloc(sizeof(*entry));
- if (!entry) {
- return false;
- }
- entry->callback = callback;
- entry->userdata = userdata;
- bool result = false;
- SDL_LockProperties(hints);
- SDL_RemoveHintCallback(name, callback, userdata);
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (hint) {
- result = true;
- } else { // Need to add a hint entry for this watcher
- hint = (SDL_Hint *)SDL_malloc(sizeof(*hint));
- if (!hint) {
- SDL_free(entry);
- SDL_UnlockProperties(hints);
- return false;
- } else {
- hint->value = NULL;
- hint->priority = SDL_HINT_DEFAULT;
- hint->callbacks = NULL;
- result = SDL_SetPointerPropertyWithCleanup(hints, name, hint, CleanupHintProperty, NULL);
- }
- }
- // Add it to the callbacks for this hint
- entry->next = hint->callbacks;
- hint->callbacks = entry;
- // Now call it with the current value
- const char *value = SDL_GetHint(name);
- callback(userdata, name, value, value);
- SDL_UnlockProperties(hints);
- return result;
- }
- void SDL_RemoveHintCallback(const char *name, SDL_HintCallback callback, void *userdata)
- {
- if (!name || !*name) {
- return;
- }
- const SDL_PropertiesID hints = GetHintProperties(false);
- if (!hints) {
- return;
- }
- SDL_LockProperties(hints);
- SDL_Hint *hint = (SDL_Hint *)SDL_GetPointerProperty(hints, name, NULL);
- if (hint) {
- SDL_HintWatch *prev = NULL;
- for (SDL_HintWatch *entry = hint->callbacks; entry; entry = entry->next) {
- if ((callback == entry->callback) && (userdata == entry->userdata)) {
- if (prev) {
- prev->next = entry->next;
- } else {
- hint->callbacks = entry->next;
- }
- SDL_free(entry);
- break;
- }
- prev = entry;
- }
- }
- SDL_UnlockProperties(hints);
- }
|