2014-09-15 16:49:46 +00:00
|
|
|
/* RetroArch - A frontend for libretro.
|
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
|
|
|
* Copyright (C) 2011-2014 - 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2014-10-16 06:20:09 +00:00
|
|
|
#include "string_list.h"
|
2014-09-15 16:49:46 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include "miscellaneous.h"
|
|
|
|
#include "compat/strl.h"
|
2014-09-16 00:30:04 +00:00
|
|
|
#include "compat/posix_string.h"
|
2014-09-15 16:49:46 +00:00
|
|
|
|
|
|
|
void string_list_free(struct string_list *list)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
if (!list)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < list->size; i++)
|
|
|
|
free(list->elems[i].data);
|
|
|
|
free(list->elems);
|
|
|
|
free(list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool string_list_capacity(struct string_list *list, size_t cap)
|
|
|
|
{
|
2014-10-01 14:15:27 +00:00
|
|
|
struct string_list_elem *new_data = NULL;
|
2014-09-15 16:49:46 +00:00
|
|
|
rarch_assert(cap > list->size);
|
|
|
|
|
2014-10-01 14:15:27 +00:00
|
|
|
new_data = (struct string_list_elem*)
|
2014-09-15 16:49:46 +00:00
|
|
|
realloc(list->elems, cap * sizeof(*new_data));
|
2014-10-01 14:15:27 +00:00
|
|
|
|
2014-09-15 16:49:46 +00:00
|
|
|
if (!new_data)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
list->elems = new_data;
|
|
|
|
list->cap = cap;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct string_list *string_list_new(void)
|
|
|
|
{
|
2014-10-01 14:15:27 +00:00
|
|
|
struct string_list *list = (struct string_list*)
|
|
|
|
calloc(1, sizeof(*list));
|
|
|
|
|
2014-09-15 16:49:46 +00:00
|
|
|
if (!list)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!string_list_capacity(list, 32))
|
|
|
|
{
|
|
|
|
string_list_free(list);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_list_append(struct string_list *list, const char *elem,
|
|
|
|
union string_list_elem_attr attr)
|
|
|
|
{
|
2014-10-01 14:15:27 +00:00
|
|
|
char *dup;
|
2014-09-15 16:49:46 +00:00
|
|
|
if (list->size >= list->cap &&
|
|
|
|
!string_list_capacity(list, list->cap * 2))
|
|
|
|
return false;
|
|
|
|
|
2014-10-01 14:15:27 +00:00
|
|
|
dup = strdup(elem);
|
2014-09-15 16:49:46 +00:00
|
|
|
if (!dup)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
list->elems[list->size].data = dup;
|
|
|
|
list->elems[list->size].attr = attr;
|
|
|
|
|
|
|
|
list->size++;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-10-01 14:15:27 +00:00
|
|
|
void string_list_set(struct string_list *list,
|
|
|
|
unsigned index, const char *str)
|
2014-09-15 16:49:46 +00:00
|
|
|
{
|
|
|
|
free(list->elems[index].data);
|
|
|
|
rarch_assert(list->elems[index].data = strdup(str));
|
|
|
|
}
|
|
|
|
|
|
|
|
void string_list_join_concat(char *buffer, size_t size,
|
|
|
|
const struct string_list *list, const char *sep)
|
|
|
|
{
|
2014-10-01 14:15:27 +00:00
|
|
|
size_t i, len = strlen(buffer);
|
2014-09-15 16:49:46 +00:00
|
|
|
rarch_assert(len < size);
|
|
|
|
buffer += len;
|
|
|
|
size -= len;
|
|
|
|
|
|
|
|
for (i = 0; i < list->size; i++)
|
|
|
|
{
|
|
|
|
strlcat(buffer, list->elems[i].data, size);
|
|
|
|
if ((i + 1) < list->size)
|
|
|
|
strlcat(buffer, sep, size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct string_list *string_split(const char *str, const char *delim)
|
|
|
|
{
|
2014-10-01 14:15:27 +00:00
|
|
|
char *save = NULL;
|
2014-09-15 16:49:46 +00:00
|
|
|
char *copy = NULL;
|
|
|
|
const char *tmp = NULL;
|
|
|
|
struct string_list *list = string_list_new();
|
2014-10-01 14:15:27 +00:00
|
|
|
|
2014-09-15 16:49:46 +00:00
|
|
|
if (!list)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
copy = strdup(str);
|
|
|
|
if (!copy)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
tmp = strtok_r(copy, delim, &save);
|
|
|
|
while (tmp)
|
|
|
|
{
|
|
|
|
union string_list_elem_attr attr;
|
|
|
|
memset(&attr, 0, sizeof(attr));
|
|
|
|
|
|
|
|
if (!string_list_append(list, tmp, attr))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
tmp = strtok_r(NULL, delim, &save);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(copy);
|
|
|
|
return list;
|
|
|
|
|
|
|
|
error:
|
|
|
|
string_list_free(list);
|
|
|
|
free(copy);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_list_find_elem(const struct string_list *list, const char *elem)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
if (!list)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (i = 0; i < list->size; i++)
|
|
|
|
{
|
|
|
|
if (strcasecmp(list->elems[i].data, elem) == 0)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_list_find_elem_prefix(const struct string_list *list,
|
|
|
|
const char *prefix, const char *elem)
|
|
|
|
{
|
|
|
|
size_t i;
|
2014-10-01 14:15:27 +00:00
|
|
|
char prefixed[PATH_MAX];
|
|
|
|
|
2014-09-15 16:49:46 +00:00
|
|
|
if (!list)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
snprintf(prefixed, sizeof(prefixed), "%s%s", prefix, elem);
|
|
|
|
|
|
|
|
for (i = 0; i < list->size; i++)
|
|
|
|
{
|
|
|
|
if (strcasecmp(list->elems[i].data, elem) == 0 ||
|
|
|
|
strcasecmp(list->elems[i].data, prefixed) == 0)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|