efl/legacy/eet/src/lib/eet_dictionary.c

413 lines
9.3 KiB
C

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif /* ifdef HAVE_CONFIG_H */
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <Eina.h>
#include "Eet.h"
#include "Eet_private.h"
Eet_Dictionary *
eet_dictionary_add(void)
{
Eet_Dictionary *new;
new = calloc(1, sizeof (Eet_Dictionary));
if (!new)
return NULL;
memset(new->hash, -1, sizeof (int) * 256);
return new;
} /* eet_dictionary_add */
void
eet_dictionary_free(Eet_Dictionary *ed)
{
if (ed)
{
int i;
for (i = 0; i < ed->count; ++i)
if (ed->all[i].str)
free(ed->all[i].str);
if (ed->all)
free(ed->all);
free(ed);
}
} /* eet_dictionary_free */
static int
_eet_dictionary_lookup(Eet_Dictionary *ed,
const char *string,
int hash)
{
int prev = -1;
int current;
current = ed->hash[hash];
while (current != -1)
{
if (ed->all[current].str)
if (strcmp(ed->all[current].str, string) >= 0)
break;
if (ed->all[current].mmap)
if (strcmp(ed->all[current].mmap, string) >= 0)
break;
prev = current;
current = ed->all[current].next;
}
if (current == -1)
return prev;
return current;
} /* _eet_dictionary_lookup */
int
eet_dictionary_string_add(Eet_Dictionary *ed,
const char *string)
{
Eet_String *current;
char *str;
int hash;
int idx;
int len;
if (!ed)
return -1;
hash = _eet_hash_gen(string, 8);
idx = _eet_dictionary_lookup(ed, string, hash);
if (idx != -1)
{
if (ed->all[idx].str)
if (strcmp(ed->all[idx].str, string) == 0)
return idx;
if (ed->all[idx].mmap)
if (strcmp(ed->all[idx].mmap, string) == 0)
return idx;
}
if (ed->total == ed->count)
{
Eet_String *new;
int total;
total = ed->total + 8;
new = realloc(ed->all, sizeof (Eet_String) * total);
if (!new)
return -1;
ed->all = new;
ed->total = total;
}
len = strlen(string) + 1;
str = strdup(string);
if (!str)
return -1;
current = ed->all + ed->count;
current->type = EET_D_NOT_CONVERTED;
current->hash = hash;
current->str = str;
current->len = len;
current->mmap = NULL;
if (idx == -1)
{
current->next = ed->hash[hash];
current->prev = -1;
ed->hash[hash] = ed->count;
}
else
{
current->next = idx;
current->prev = ed->all[idx].prev;
if (current->next != -1)
ed->all[current->next].prev = ed->count;
if (current->prev != -1)
ed->all[current->prev].next = ed->count;
else
ed->hash[hash] = ed->count;
}
return ed->count++;
} /* eet_dictionary_string_add */
int
eet_dictionary_string_get_size(const Eet_Dictionary *ed,
int idx)
{
if (!ed)
return 0;
if (idx < 0)
return 0;
if (idx < ed->count)
return ed->all[idx].len;
return 0;
} /* eet_dictionary_string_get_size */
int
eet_dictionary_string_get_hash(const Eet_Dictionary *ed,
int idx)
{
if (!ed)
return -1;
if (idx < 0)
return -1;
if (idx < ed->count)
return ed->all[idx].hash;
return -1;
} /* eet_dictionary_string_get_hash */
const char *
eet_dictionary_string_get_char(const Eet_Dictionary *ed,
int idx)
{
if (!ed)
return NULL;
if (idx < 0)
return NULL;
if (idx < ed->count)
{
#ifdef _WIN32
/* Windows file system could change the mmaped file when replacing a file. So we need to copy all string in memory to avoid bugs. */
if (!ed->all[idx].str)
{
ed->all[idx].str = strdup(ed->all[idx].mmap);
ed->all[idx].mmap = NULL;
}
#else /* ifdef _WIN32 */
if (ed->all[idx].mmap)
return ed->all[idx].mmap;
#endif /* ifdef _WIN32 */
return ed->all[idx].str;
}
return NULL;
} /* eet_dictionary_string_get_char */
static inline Eina_Bool
_eet_dictionary_string_get_me_cache(const char *s,
int len,
int *mantisse,
int *exponent)
{
if ((len == 6) && (s[0] == '0') && (s[1] == 'x') && (s[3] == 'p'))
{
*mantisse = (s[2] >= 'a') ? (s[2] - 'a' + 10) : (s[2] - '0');
*exponent = (s[5] - '0');
return EINA_TRUE;
}
return EINA_FALSE;
} /* _eet_dictionary_string_get_me_cache */
static inline Eina_Bool
_eet_dictionary_string_get_float_cache(const char *s,
int len,
float *result)
{
int mantisse;
int exponent;
if (_eet_dictionary_string_get_me_cache(s, len, &mantisse, &exponent))
{
if (s[4] == '+')
*result = (float)(mantisse << exponent);
else
*result = (float)mantisse / (float)(1 << exponent);
return EINA_TRUE;
}
return EINA_FALSE;
} /* _eet_dictionary_string_get_float_cache */
static inline Eina_Bool
_eet_dictionary_string_get_double_cache(const char *s,
int len,
double *result)
{
int mantisse;
int exponent;
if (_eet_dictionary_string_get_me_cache(s, len, &mantisse, &exponent))
{
if (s[4] == '+')
*result = (double)(mantisse << exponent);
else
*result = (double)mantisse / (float)(1 << exponent);
return EINA_TRUE;
}
return EINA_FALSE;
} /* _eet_dictionary_string_get_double_cache */
static inline Eina_Bool
_eet_dictionary_test(const Eet_Dictionary *ed,
int idx,
void *result)
{
if (!result)
return EINA_FALSE;
if (!ed)
return EINA_FALSE;
if (idx < 0)
return EINA_FALSE;
if (!(idx < ed->count))
return EINA_FALSE;
return EINA_TRUE;
} /* _eet_dictionary_test */
Eina_Bool
eet_dictionary_string_get_float(const Eet_Dictionary *ed,
int idx,
float *result)
{
if (!_eet_dictionary_test(ed, idx, result))
return EINA_FALSE;
if (!(ed->all[idx].type & EET_D_FLOAT))
{
const char *str;
str = ed->all[idx].str ? ed->all[idx].str : ed->all[idx].mmap;
if (!_eet_dictionary_string_get_float_cache(str, ed->all[idx].len,
&ed->all[idx].f))
{
long long mantisse = 0;
long exponent = 0;
if (eina_convert_atod(str, ed->all[idx].len, &mantisse,
&exponent) == EINA_FALSE)
return EINA_FALSE;
ed->all[idx].f = ldexpf((float)mantisse, exponent);
}
ed->all[idx].type |= EET_D_FLOAT;
}
*result = ed->all[idx].f;
return EINA_TRUE;
} /* eet_dictionary_string_get_float */
Eina_Bool
eet_dictionary_string_get_double(const Eet_Dictionary *ed,
int idx,
double *result)
{
if (!_eet_dictionary_test(ed, idx, result))
return EINA_FALSE;
if (!(ed->all[idx].type & EET_D_DOUBLE))
{
const char *str;
str = ed->all[idx].str ? ed->all[idx].str : ed->all[idx].mmap;
if (!_eet_dictionary_string_get_double_cache(str, ed->all[idx].len,
&ed->all[idx].d))
{
long long mantisse = 0;
long exponent = 0;
if (eina_convert_atod(str, ed->all[idx].len, &mantisse,
&exponent) == EINA_FALSE)
return EINA_FALSE;
ed->all[idx].d = ldexp((double)mantisse, exponent);
}
ed->all[idx].type |= EET_D_DOUBLE;
}
*result = ed->all[idx].d;
return EINA_TRUE;
} /* eet_dictionary_string_get_double */
Eina_Bool
eet_dictionary_string_get_fp(const Eet_Dictionary *ed,
int idx,
Eina_F32p32 *result)
{
if (!_eet_dictionary_test(ed, idx, result))
return EINA_FALSE;
if (!(ed->all[idx].type & EET_D_FIXED_POINT))
{
const char *str;
Eina_F32p32 fp;
str = ed->all[idx].str ? ed->all[idx].str : ed->all[idx].mmap;
if (!eina_convert_atofp(str, ed->all[idx].len, &fp))
return EINA_FALSE;
ed->all[idx].fp = fp;
ed->all[idx].type |= EET_D_FIXED_POINT;
}
*result = ed->all[idx].fp;
return EINA_TRUE;
} /* eet_dictionary_string_get_fp */
EAPI int
eet_dictionary_string_check(Eet_Dictionary *ed,
const char *string)
{
int i;
if ((!ed) || (!string))
return 0;
if ((ed->start <= string) && (string < ed->end))
return 1;
for (i = 0; i < ed->count; ++i)
if (ed->all[i].str == string)
return 1;
return 0;
} /* eet_dictionary_string_check */