2010-08-08 10:06:34 +02:00
|
|
|
/*
|
2015-01-30 22:10:29 +01:00
|
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
2014-08-24 22:54:40 +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.
|
|
|
|
**/
|
|
|
|
|
2017-12-29 06:13:28 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <platform.h>
|
|
|
|
#endif
|
2016-08-29 15:49:31 +02:00
|
|
|
#include "strings.h"
|
2016-08-29 10:20:08 +02:00
|
|
|
#include "assert.h"
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
/* libc includes */
|
2017-12-11 19:23:56 +01:00
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdio.h>
|
2014-08-24 22:54:40 +02:00
|
|
|
#include <assert.h>
|
2015-05-18 08:59:38 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2017-12-31 21:33:31 +01:00
|
|
|
#ifdef HAVE_LIBBSD
|
|
|
|
#include <bsd/string.h>
|
|
|
|
#else
|
|
|
|
#include <string.h>
|
|
|
|
#endif
|
|
|
|
|
2017-12-30 13:04:00 +01:00
|
|
|
size_t str_strlcpy(char *dst, const char *src, size_t len)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_BSDSTRING
|
|
|
|
return strlcpy(dst, src, len);
|
|
|
|
#else
|
|
|
|
register char *d = dst;
|
|
|
|
register const char *s = src;
|
|
|
|
register size_t n = len;
|
|
|
|
|
|
|
|
assert(src);
|
|
|
|
assert(dst);
|
|
|
|
/* Copy as many bytes as will fit */
|
|
|
|
if (n != 0 && --n != 0) {
|
|
|
|
do {
|
|
|
|
if ((*d++ = *s++) == 0)
|
|
|
|
break;
|
|
|
|
} while (--n != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not enough room in dst, add NUL and traverse rest of src */
|
|
|
|
if (n == 0) {
|
|
|
|
if (len != 0)
|
|
|
|
*d = '\0'; /* NUL-terminate dst */
|
|
|
|
while (*s++);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (s - src - 1); /* count does not include NUL */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t str_strlcat(char *dst, const char *src, size_t len)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_BSDSTRING
|
|
|
|
return strlcat(dst, src, len);
|
|
|
|
#else
|
|
|
|
register char *d = dst;
|
|
|
|
register const char *s = src;
|
|
|
|
register size_t n = len;
|
|
|
|
size_t dlen;
|
|
|
|
|
|
|
|
/* Find the end of dst and adjust bytes left but don't go past end */
|
|
|
|
while (*d != '\0' && n-- != 0)
|
|
|
|
d++;
|
|
|
|
dlen = d - dst;
|
|
|
|
n = len - dlen;
|
|
|
|
|
|
|
|
if (n == 0)
|
|
|
|
return (dlen + strlen(s));
|
|
|
|
while (*s != '\0') {
|
|
|
|
if (n != 1) {
|
|
|
|
*d++ = *s;
|
|
|
|
n--;
|
|
|
|
}
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
*d = '\0';
|
|
|
|
|
|
|
|
return (dlen + (s - src)); /* count does not include NUL */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-12-11 19:23:56 +01:00
|
|
|
size_t str_slprintf(char * dst, size_t size, const char * format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
va_start(args, format);
|
|
|
|
result = vsnprintf(dst, size, format, args);
|
|
|
|
va_end(args);
|
|
|
|
if (result < 0 || result >= (int)size) {
|
|
|
|
dst[size - 1] = '\0';
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (size_t)result;
|
|
|
|
}
|
|
|
|
|
2017-12-11 18:20:21 +01:00
|
|
|
void str_replace(char *buffer, size_t size, const char *tmpl, const char *var, const char *value)
|
|
|
|
{
|
|
|
|
size_t val_len = strlen(value);
|
|
|
|
size_t var_len = strlen(var);
|
|
|
|
char *s = buffer;
|
|
|
|
while (buffer + size > s) {
|
|
|
|
char *p = strstr(tmpl, var);
|
|
|
|
size_t len;
|
|
|
|
if (p) {
|
|
|
|
len = p - tmpl;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
len = strlen(tmpl);
|
|
|
|
}
|
|
|
|
if (len < size) {
|
|
|
|
memmove(s, tmpl, len);
|
|
|
|
tmpl += len;
|
|
|
|
s += len;
|
|
|
|
size -= len;
|
|
|
|
if (p && val_len < size) {
|
|
|
|
tmpl += var_len;
|
|
|
|
memmove(s, value, val_len);
|
|
|
|
s += val_len;
|
|
|
|
size -= val_len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!p) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*s = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int str_hash(const char *s)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-24 22:54:40 +02:00
|
|
|
unsigned int key = 0;
|
2015-08-05 10:25:25 +02:00
|
|
|
assert(s);
|
2014-08-24 22:54:40 +02:00
|
|
|
while (*s) {
|
|
|
|
key = key * 37 + *s++;
|
|
|
|
}
|
|
|
|
return key & 0x7FFFFFFF;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-12-11 18:20:21 +01:00
|
|
|
const char *str_escape(const char *str, char *buffer,
|
2015-05-15 20:35:36 +02:00
|
|
|
size_t len)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-24 22:54:40 +02:00
|
|
|
const char *start = strchr(str, '\"');
|
|
|
|
if (!start) start = strchr(str, '\\');
|
|
|
|
assert(buffer);
|
|
|
|
if (start) {
|
|
|
|
const char *p;
|
|
|
|
char *o;
|
|
|
|
size_t skip = start - str;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
2014-12-12 11:28:37 +01:00
|
|
|
if (skip > len) {
|
|
|
|
skip = len;
|
|
|
|
}
|
2014-08-24 22:54:40 +02:00
|
|
|
memcpy(buffer, str, skip);
|
|
|
|
o = buffer + skip;
|
|
|
|
p = str + skip;
|
2014-12-12 11:28:37 +01:00
|
|
|
len -= skip;
|
2014-08-24 22:54:40 +02:00
|
|
|
do {
|
|
|
|
if (*p == '\"' || *p == '\\') {
|
|
|
|
if (len < 2) {
|
|
|
|
*o = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
(*o++) = '\\';
|
|
|
|
len -= 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (len < 1) {
|
|
|
|
*o = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
--len;
|
|
|
|
}
|
|
|
|
(*o++) = (*p);
|
|
|
|
} while (*p++);
|
|
|
|
return buffer;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2014-08-24 22:54:40 +02:00
|
|
|
return str;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-03-15 22:51:57 +01:00
|
|
|
unsigned int jenkins_hash(unsigned int a)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-24 22:54:40 +02:00
|
|
|
a = (a + 0x7ed55d16) + (a << 12);
|
|
|
|
a = (a ^ 0xc761c23c) ^ (a >> 19);
|
|
|
|
a = (a + 0x165667b1) + (a << 5);
|
|
|
|
a = (a + 0xd3a2646c) ^ (a << 9);
|
|
|
|
a = (a + 0xfd7046c5) + (a << 3);
|
|
|
|
a = (a ^ 0xb55a4f09) ^ (a >> 16);
|
|
|
|
return a;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-03-15 22:51:57 +01:00
|
|
|
unsigned int wang_hash(unsigned int a)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-24 22:54:40 +02:00
|
|
|
a = ~a + (a << 15); /* a = (a << 15) - a - 1; */
|
|
|
|
a = a ^ (a >> 12);
|
|
|
|
a = a + (a << 2);
|
|
|
|
a = a ^ (a >> 4);
|
|
|
|
a = a * 2057; /* a = (a + (a << 3)) + (a << 11); */
|
|
|
|
a = a ^ (a >> 16);
|
|
|
|
return a;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2017-12-28 18:29:40 +01:00
|
|
|
|
|
|
|
char *str_strdup(const char *s) {
|
2017-12-31 21:33:31 +01:00
|
|
|
#ifdef HAVE_STRDUP
|
|
|
|
return strdup(s);
|
|
|
|
#elif defined(_MSC_VER)
|
2017-12-28 18:29:40 +01:00
|
|
|
return _strdup(s);
|
|
|
|
#else
|
2017-12-31 21:33:31 +01:00
|
|
|
size_t len = strlen(s);
|
|
|
|
char *dup = malloc(len+1);
|
|
|
|
memcpy(dup, s, len+1);
|
|
|
|
return dup;
|
2017-12-28 18:29:40 +01:00
|
|
|
#endif
|
|
|
|
}
|
2017-12-31 18:07:29 +01:00
|
|
|
|
|
|
|
void sbs_init(struct sbstring *sbs, char *buffer, size_t size)
|
|
|
|
{
|
|
|
|
assert(sbs);
|
|
|
|
assert(size>0);
|
|
|
|
sbs->begin = buffer;
|
|
|
|
sbs->size = size;
|
|
|
|
sbs->end = buffer;
|
|
|
|
buffer[0] = '\0';
|
|
|
|
}
|
|
|
|
|
2017-12-31 20:07:40 +01:00
|
|
|
void sbs_strncat(struct sbstring *sbs, const char *str, size_t size)
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
assert(sbs);
|
|
|
|
len = sbs->size - (sbs->end - sbs->begin) - 1;
|
|
|
|
if (len < size) {
|
|
|
|
size = len;
|
|
|
|
}
|
|
|
|
memcpy(sbs->end, str, size);
|
|
|
|
sbs->end[size] = '\0';
|
|
|
|
sbs->end += size;
|
|
|
|
}
|
|
|
|
|
2017-12-31 18:07:29 +01:00
|
|
|
void sbs_strcat(struct sbstring *sbs, const char *str)
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
assert(sbs);
|
|
|
|
len = sbs->size - (sbs->end - sbs->begin);
|
|
|
|
len = str_strlcpy(sbs->end, str, len);
|
|
|
|
sbs->end += len;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sbs_strcpy(struct sbstring *sbs, const char *str)
|
|
|
|
{
|
|
|
|
size_t len = str_strlcpy(sbs->begin, str, sbs->size);
|
2017-12-31 18:51:56 +01:00
|
|
|
if (len >= sbs->size) {
|
|
|
|
len = sbs->size - 1;
|
|
|
|
}
|
|
|
|
sbs->end = sbs->begin + len;
|
2017-12-31 18:07:29 +01:00
|
|
|
}
|
2018-05-17 16:53:56 +02:00
|
|
|
|
|
|
|
char *str_unescape(char *str) {
|
|
|
|
char *read = str, *write = str;
|
|
|
|
while (*read) {
|
|
|
|
char * pos = strchr(read, '\\');
|
|
|
|
if (pos) {
|
|
|
|
size_t len = pos - read;
|
|
|
|
memmove(write, read, len);
|
|
|
|
write += len;
|
|
|
|
read += (len + 1);
|
|
|
|
switch (read[0]) {
|
|
|
|
case 'r':
|
|
|
|
*write++ = '\r';
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
*write++ = '\n';
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
*write++ = '\t';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*write++ = read[0];
|
|
|
|
}
|
|
|
|
*write = 0;
|
|
|
|
++read;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
size_t len = strlen(read);
|
|
|
|
memmove(write, read, len);
|
|
|
|
write[len] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|