/* deadbeef config file manager http://deadbeef.sourceforge.net Copyright (C) 2009-2013 Alexey Yakovenko 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. Alexey Yakovenko waker@users.sourceforge.net */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include #include #include #include #include #include #include #if HAVE_SYS_CDEFS_H #include #endif #if HAVE_SYS_SYSLIMITS_H #include #endif #include "conf.h" #include "threading.h" #define min(x,y) ((x)<(y)?(x):(y)) static DB_conf_item_t *conf_items; static int changed = 0; static uintptr_t mutex; void conf_init (void) { mutex = mutex_create (); } void conf_lock (void) { mutex_lock (mutex); } void conf_unlock (void) { mutex_unlock (mutex); } void conf_free (void) { mutex_lock (mutex); DB_conf_item_t *next = NULL; for (DB_conf_item_t *it = conf_items; it; it = next) { next = it->next; conf_item_free (it); } conf_items = NULL; changed = 0; mutex_free (mutex); mutex = 0; } int conf_load (void) { extern char dbconfdir[1024]; // $HOME/.config/deadbeef char str[1024]; snprintf (str, 1024, "%s/config", dbconfdir); FILE *fp = fopen (str, "rt"); if (!fp) { fprintf (stderr, "failed to load config file\n"); return -1; } conf_lock (); int line = 0; while (fgets (str, 1024, fp) != NULL) { line++; if (str[0] == '#' || str[0] <= 0x20) { continue; } uint8_t *p = (uint8_t *)str; while (*p && *p > 0x20) { p++; } if (!*p) { fprintf (stderr, "error in config file line %d\n", line); continue; } *p = 0; p++; // skip whitespace while (*p && *p <= 0x20) { p++; } char *value = p; // remove trailing trash while (*p && *p >= 0x20) { p++; } *p = 0; // new items are appended, to preserve order conf_set_str (str, value); } fclose (fp); changed = 0; conf_unlock (); return 0; } int conf_save (void) { extern char dbconfdir[1024]; // $HOME/.config/deadbeef char tempfile[PATH_MAX]; char str[PATH_MAX]; FILE *fp; int err; if (!changed) { return 0; } snprintf (tempfile, sizeof (tempfile), "%s/config.tmp", dbconfdir); snprintf (str, sizeof (str), "%s/config", dbconfdir); conf_lock (); changed = 0; fp = fopen (tempfile, "w+t"); if (!fp) { fprintf (stderr, "failed to open config file for writing\n"); conf_unlock (); return -1; } for (DB_conf_item_t *it = conf_items; it; it = it->next) { if (fprintf (fp, "%s %s\n", it->key, it->value) < 0) { fprintf (stderr, "failed to write to file %s (%s)\n", tempfile, strerror (errno)); fclose (fp); conf_unlock (); return -1; } } fclose (fp); err = rename (tempfile, str); if (err != 0) { fprintf (stderr, "config rename %s -> %s failed: %s\n", tempfile, str, strerror (errno)); } conf_unlock (); return 0; } void conf_item_free (DB_conf_item_t *it) { conf_lock (); if (it) { if (it->key) { free (it->key); } if (it->value) { free (it->value); } free (it); } conf_unlock (); } const char * conf_get_str_fast (const char *key, const char *def) { for (DB_conf_item_t *it = conf_items; it; it = it->next) { if (!strcasecmp (key, it->key)) { return it->value; } } return def; } void conf_get_str (const char *key, const char *def, char *buffer, int buffer_size) { conf_lock (); const char *out = conf_get_str_fast (key, def); if (out) { int n = strlen (out)+1; n = min (n, buffer_size); memcpy (buffer, out, n); buffer[buffer_size-1] = 0; } else { *buffer = 0; } conf_unlock (); } float conf_get_float (const char *key, float def) { conf_lock (); const char *v = conf_get_str_fast (key, NULL); conf_unlock (); return v ? atof (v) : def; } int conf_get_int (const char *key, int def) { conf_lock (); const char *v = conf_get_str_fast (key, NULL); conf_unlock (); return v ? atoi (v) : def; } int64_t conf_get_int64 (const char *key, int64_t def) { conf_lock (); const char *v = conf_get_str_fast (key, NULL); conf_unlock (); return v ? atoll (v) : def; } DB_conf_item_t * conf_find (const char *group, DB_conf_item_t *prev) { int l = strlen (group); for (DB_conf_item_t *it = prev ? prev->next : conf_items; it; it = it->next) { if (!strncasecmp (group, it->key, l)) { return it; } } return NULL; } void conf_set_str (const char *key, const char *val) { conf_lock (); DB_conf_item_t *prev = NULL; for (DB_conf_item_t *it = conf_items; it; it = it->next) { int cmp = strcasecmp (key, it->key); if (!cmp) { if (!strcmp (it->value, val)) { conf_unlock (); return; } free (it->value); it->value = strdup (val); conf_unlock (); changed = 1; return; } else if (cmp < 0) { break; } prev = it; } if (!val) { conf_unlock (); return; } DB_conf_item_t *it = malloc (sizeof (DB_conf_item_t)); memset (it, 0, sizeof (DB_conf_item_t)); it->key = strdup (key); it->value = strdup (val); changed = 1; if (prev) { DB_conf_item_t *next = prev->next; prev->next = it; it->next = next; } else { it->next = conf_items; conf_items = it; } conf_unlock (); } void conf_set_int (const char *key, int val) { char s[10]; snprintf (s, sizeof (s), "%d", val); conf_set_str (key, s); } void conf_set_int64 (const char *key, int64_t val) { char s[20]; snprintf (s, sizeof (s), "%"PRId64, val); conf_set_str (key, s); } void conf_set_float (const char *key, float val) { char s[10]; snprintf (s, sizeof (s), "%0.7f", val); conf_set_str (key, s); } int conf_ischanged (void) { return changed; } void conf_setchanged (int c) { changed = c; } void conf_remove_items (const char *key) { int l = strlen (key); conf_lock (); DB_conf_item_t *prev = NULL; DB_conf_item_t *it; for (it = conf_items; it; prev = it, it = it->next) { if (!strncasecmp (key, it->key, l)) { break; } } DB_conf_item_t *next = NULL; while (it) { next = it->next; conf_item_free (it); it = next; if (!it || strncasecmp (key, it->key, l)) { break; } } if (prev) { prev->next = next; } else { conf_items = next; } conf_unlock (); }