/* Copyright (C) 2010-2016 The RetroArch team * * --------------------------------------------------------------------------------------- * The following license statement only applies to this file (file_list.c). * --------------------------------------------------------------------------------------- * * Permission is hereby granted, free of charge, * to any person obtaining a copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation the rights to * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, * and to permit persons to whom the Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include #include #include #include #include #include #include /** * file_list_capacity: * @list : pointer to file list * @cap : new capacity for file list. * * Change maximum capacity of file list's size. * * Returns: true (1) if successful, otherwise false (0). **/ static struct item_file *realloc_file_list_capacity(file_list_t *list, size_t cap) { struct item_file *new_data = NULL; retro_assert(cap > list->size); new_data = (struct item_file*)realloc(list->list, cap * sizeof(struct item_file)); if (!new_data) return NULL; if (cap > list->capacity) memset(&new_data[list->capacity], 0, sizeof(*new_data) * (cap - list->capacity)); return new_data; } bool file_list_prepend(file_list_t *list, const char *path, const char *label, unsigned type, size_t directory_ptr, size_t entry_idx) { unsigned i; if (list->size >= list->capacity) { size_t new_capacity = list->capacity * 2 + 1; struct item_file *items = realloc_file_list_capacity( list, new_capacity); if (!items) return false; list->list = items; list->capacity = new_capacity; } list->size++; for (i = list->size -1; i > 0; i--) { struct item_file *copy = (struct item_file*) calloc(1, sizeof(struct item_file)); memcpy(copy, &list->list[i-1], sizeof(struct item_file)); memcpy(&list->list[i-1], &list->list[i], sizeof(struct item_file)); memcpy(&list->list[i], copy, sizeof(struct item_file)); free(copy); } memset(&list->list[0], 0, sizeof(*list->list) * list->capacity); list->list[0].label = NULL; list->list[0].path = NULL; list->list[0].alt = NULL; list->list[0].userdata = NULL; list->list[0].actiondata = NULL; list->list[0].type = type; list->list[0].directory_ptr = directory_ptr; list->list[0].entry_idx = entry_idx; if (label) list->list[0].label = strdup(label); if (path) list->list[0].path = strdup(path); return true; } bool file_list_append(file_list_t *list, const char *path, const char *label, unsigned type, size_t directory_ptr, size_t entry_idx) { if (list->size >= list->capacity) { size_t new_capacity = list->capacity * 2 + 1; struct item_file *items = realloc_file_list_capacity(list, new_capacity); if (!items) return false; list->list = items; list->capacity = new_capacity; } list->list[list->size].label = NULL; list->list[list->size].path = NULL; list->list[list->size].alt = NULL; list->list[list->size].userdata = NULL; list->list[list->size].actiondata = NULL; list->list[list->size].type = type; list->list[list->size].directory_ptr = directory_ptr; list->list[list->size].entry_idx = entry_idx; if (label) list->list[list->size].label = strdup(label); if (path) list->list[list->size].path = strdup(path); list->size++; return true; } size_t file_list_get_size(const file_list_t *list) { if (!list) return 0; return list->size; } size_t file_list_get_directory_ptr(const file_list_t *list) { size_t size = file_list_get_size(list); return list->list[size].directory_ptr; } void file_list_pop(file_list_t *list, size_t *directory_ptr) { if (!list) return; if (list->size != 0) { --list->size; if (list->list[list->size].path) free(list->list[list->size].path); list->list[list->size].path = NULL; if (list->list[list->size].label) free(list->list[list->size].label); list->list[list->size].label = NULL; } if (directory_ptr) *directory_ptr = list->list[list->size].directory_ptr; } void file_list_free(file_list_t *list) { size_t i; if (!list) return; for (i = 0; i < list->size; i++) { file_list_free_userdata(list, i); file_list_free_actiondata(list, i); if (list->list[i].path) free(list->list[i].path); list->list[i].path = NULL; if (list->list[i].label) free(list->list[i].label); list->list[i].label = NULL; if (list->list[i].alt) free(list->list[i].alt); list->list[i].alt = NULL; } if (list->list) free(list->list); list->list = NULL; free(list); } void file_list_clear(file_list_t *list) { size_t i; if (!list) return; for (i = 0; i < list->size; i++) { if (list->list[i].path) free(list->list[i].path); list->list[i].path = NULL; if (list->list[i].label) free(list->list[i].label); list->list[i].label = NULL; if (list->list[i].alt) free(list->list[i].alt); list->list[i].alt = NULL; } list->size = 0; } void file_list_copy(const file_list_t *src, file_list_t *dst) { struct item_file *item; if (!src || !dst) return; if (dst->list) { for (item = dst->list; item < &dst->list[dst->size]; ++item) { if (item->path) free(item->path); if (item->label) free(item->label); if (item->alt) free(item->alt); } free(dst->list); } dst->size = 0; dst->capacity = 0; dst->list = (struct item_file*)malloc(src->size * sizeof(struct item_file)); if (!dst->list) return; dst->size = dst->capacity = src->size; memcpy(dst->list, src->list, dst->size * sizeof(struct item_file)); for (item = dst->list; item < &dst->list[dst->size]; ++item) { if (item->path) item->path = strdup(item->path); if (item->label) item->label = strdup(item->label); if (item->alt) item->alt = strdup(item->alt); } } void file_list_set_label_at_offset(file_list_t *list, size_t idx, const char *label) { if (!list) return; if (list->list[idx].label) free(list->list[idx].label); list->list[idx].alt = NULL; if (label) list->list[idx].label = strdup(label); } void file_list_get_label_at_offset(const file_list_t *list, size_t idx, const char **label) { if (!label || !list) return; *label = list->list[idx].path; if (list->list[idx].label) *label = list->list[idx].label; } void file_list_set_alt_at_offset(file_list_t *list, size_t idx, const char *alt) { if (!list || !alt) return; if (list->list[idx].alt) free(list->list[idx].alt); list->list[idx].alt = NULL; if (alt) list->list[idx].alt = strdup(alt); } void file_list_get_alt_at_offset(const file_list_t *list, size_t idx, const char **alt) { if (!list) return; if (alt) *alt = list->list[idx].alt ? list->list[idx].alt : list->list[idx].path; } static int file_list_alt_cmp(const void *a_, const void *b_) { const struct item_file *a = (const struct item_file*)a_; const struct item_file *b = (const struct item_file*)b_; const char *cmp_a = a->alt ? a->alt : a->path; const char *cmp_b = b->alt ? b->alt : b->path; return strcasecmp(cmp_a, cmp_b); } static int file_list_type_cmp(const void *a_, const void *b_) { const struct item_file *a = (const struct item_file*)a_; const struct item_file *b = (const struct item_file*)b_; if (a->type < b->type) return -1; if (a->type == b->type) return 0; return 1; } void file_list_sort_on_alt(file_list_t *list) { qsort(list->list, list->size, sizeof(list->list[0]), file_list_alt_cmp); } void file_list_sort_on_type(file_list_t *list) { qsort(list->list, list->size, sizeof(list->list[0]), file_list_type_cmp); } void *file_list_get_userdata_at_offset(const file_list_t *list, size_t idx) { if (!list) return NULL; return list->list[idx].userdata; } void file_list_set_userdata(const file_list_t *list, size_t idx, void *ptr) { if (!list || !ptr) return; list->list[idx].userdata = ptr; } void file_list_set_actiondata(const file_list_t *list, size_t idx, void *ptr) { if (!list || !ptr) return; list->list[idx].actiondata = ptr; } void *file_list_get_actiondata_at_offset(const file_list_t *list, size_t idx) { if (!list) return NULL; return list->list[idx].actiondata; } void file_list_free_actiondata(const file_list_t *list, size_t idx) { if (!list) return; if (list->list[idx].actiondata) free(list->list[idx].actiondata); list->list[idx].actiondata = NULL; } void file_list_free_userdata(const file_list_t *list, size_t idx) { if (!list) return; if (list->list[idx].userdata) free(list->list[idx].userdata); list->list[idx].userdata = NULL; } void *file_list_get_last_actiondata(const file_list_t *list) { if (!list) return NULL; return list->list[list->size - 1].actiondata; } void file_list_get_at_offset(const file_list_t *list, size_t idx, const char **path, const char **label, unsigned *file_type, size_t *entry_idx) { if (!list) return; if (path) *path = list->list[idx].path; if (label) *label = list->list[idx].label; if (file_type) *file_type = list->list[idx].type; if (entry_idx) *entry_idx = list->list[idx].entry_idx; } void file_list_get_last(const file_list_t *list, const char **path, const char **label, unsigned *file_type, size_t *entry_idx) { if (!list) return; if (list->size) file_list_get_at_offset(list, list->size - 1, path, label, file_type, entry_idx); } bool file_list_search(const file_list_t *list, const char *needle, size_t *idx) { size_t i; const char *alt; bool ret = false; if (!list) return false; for (i = 0; i < list->size; i++) { const char *str; file_list_get_alt_at_offset(list, i, &alt); if (!alt) { file_list_get_label_at_offset(list, i, &alt); if (!alt) continue; } str = (const char *)strcasestr(alt, needle); if (str == alt) { /* Found match with first chars, best possible match. */ *idx = i; ret = true; break; } else if (str && !ret) { /* Found mid-string match, but try to find a match with * first characters before we settle. */ *idx = i; ret = true; } } return ret; }