2010-08-08 10:06:34 +02:00
|
|
|
/*
|
2015-01-30 20:37:14 +01:00
|
|
|
Copyright (c) 1998-2015, Enno Rehling Rehling <enno@eressea.de>
|
2014-08-23 09:17:58 +02:00
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
#include "base36.h"
|
2014-09-29 18:32:12 +02:00
|
|
|
#include "log.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int atoi36(const char *str)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-23 09:17:58 +02:00
|
|
|
/* cannot use strtol, because invalid strings will cause crash */
|
|
|
|
const unsigned char *s = (const unsigned char *)str;
|
|
|
|
int i = 0, sign = 1;
|
|
|
|
assert(s);
|
|
|
|
if (!(*s))
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-08-23 09:17:58 +02:00
|
|
|
while (isxspace(*(unsigned char *)s))
|
|
|
|
++s;
|
|
|
|
if (*s == '-') {
|
|
|
|
sign = -1;
|
|
|
|
++s;
|
|
|
|
}
|
|
|
|
while (isalnum(*(unsigned char *)s)) {
|
|
|
|
if (isupper(*(unsigned char *)s))
|
|
|
|
i = i * 36 + (*s) - 'A' + 10;
|
|
|
|
else if (islower(*(unsigned char *)s))
|
|
|
|
i = i * 36 + (*s) - 'a' + 10;
|
|
|
|
else if (isdigit(*(unsigned char *)s))
|
|
|
|
i = i * 36 + (*s) - '0';
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
++s;
|
|
|
|
}
|
|
|
|
if (i < 0)
|
|
|
|
return 0;
|
|
|
|
return i * sign;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *itoab(int i, int base)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-09-29 18:32:12 +02:00
|
|
|
const int maxlen = 20;
|
2014-08-24 22:54:40 +02:00
|
|
|
static char **as = NULL; // FIXME: static return value
|
2014-08-23 09:17:58 +02:00
|
|
|
char *s, *dst;
|
|
|
|
static int index = 0; /* STATIC_XCALL: used across calls */
|
|
|
|
int neg = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-08-23 09:17:58 +02:00
|
|
|
if (!as) {
|
|
|
|
int j;
|
2014-09-29 18:32:12 +02:00
|
|
|
char *x = (char *)calloc(sizeof(char), maxlen * 4); /* STATIC_LEAK: malloc in static variable */
|
2014-08-23 09:17:58 +02:00
|
|
|
as = (char **)calloc(sizeof(char *), 4);
|
|
|
|
for (j = 0; j != 4; ++j)
|
2014-09-29 18:32:12 +02:00
|
|
|
as[j] = x + j * maxlen;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2014-08-23 09:17:58 +02:00
|
|
|
s = as[index];
|
|
|
|
index = (index + 1) & 3; /* quick for % 4 */
|
2014-09-29 18:32:12 +02:00
|
|
|
dst = s + maxlen - 1;
|
2014-08-23 09:17:58 +02:00
|
|
|
(*dst--) = 0;
|
|
|
|
if (i != 0) {
|
|
|
|
if (i < 0) {
|
|
|
|
i = -i;
|
|
|
|
neg = 1;
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
while (i && dst >= s) {
|
2014-08-23 09:17:58 +02:00
|
|
|
int x = i % base;
|
|
|
|
i = i / base;
|
2014-09-29 18:32:12 +02:00
|
|
|
if (x < 10) {
|
2014-08-23 09:17:58 +02:00
|
|
|
*(dst--) = (char)('0' + x);
|
2014-09-29 18:32:12 +02:00
|
|
|
}
|
|
|
|
else if ('a' + x - 10 == 'l') {
|
2014-08-23 09:17:58 +02:00
|
|
|
*(dst--) = 'L';
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2014-09-29 18:32:12 +02:00
|
|
|
else {
|
2014-08-23 09:17:58 +02:00
|
|
|
*(dst--) = (char)('a' + (x - 10));
|
2014-09-29 18:32:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (dst > s) {
|
|
|
|
if (neg) {
|
|
|
|
*(dst) = '-';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
++dst;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("static buffer exhauset, itoab(%d, %d)", i, base);
|
|
|
|
assert(i == 0 || !"itoab: static buffer exhausted");
|
2014-08-23 09:17:58 +02:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
else
|
2014-08-23 09:17:58 +02:00
|
|
|
*dst = '0';
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-08-23 09:17:58 +02:00
|
|
|
return dst;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *itoa36(int i)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-23 09:17:58 +02:00
|
|
|
return itoab(i, 36);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *itoa10(int i)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-23 09:17:58 +02:00
|
|
|
return itoab(i, 10);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int i10toi36(int i)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-23 09:17:58 +02:00
|
|
|
int r = 0;
|
|
|
|
while (i) {
|
|
|
|
r = r * 36 + i % 10;
|
|
|
|
i = i / 10;
|
|
|
|
}
|
|
|
|
return r;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|