2009-02-04 14:00:44 +03:00
|
|
|
#ifndef MHL_STRING_H
|
|
|
|
#define MHL_STRING_H
|
2009-01-15 16:10:45 +03:00
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <stdarg.h>
|
2009-01-30 10:31:28 +03:00
|
|
|
#include <assert.h>
|
2009-01-19 04:31:04 +03:00
|
|
|
#include <mhl/memory.h>
|
2009-01-15 16:10:45 +03:00
|
|
|
|
|
|
|
#define mhl_str_dup(str) ((str ? strdup(str) : strdup("")))
|
|
|
|
#define mhl_str_ndup(str,len) ((str ? strndup(str,len) : strdup("")))
|
|
|
|
#define mhl_str_len(str) ((str ? strlen(str) : 0))
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
#define ISSPACE(c) isspace((unsigned char)(c))
|
|
|
|
#define TOUPPER(c) toupper((unsigned char)(c))
|
|
|
|
|
2009-01-25 14:10:37 +03:00
|
|
|
static inline char * mhl_str_dup_range(const char * s_start, const char * s_bound)
|
|
|
|
{
|
|
|
|
return mhl_str_ndup(s_start, s_bound - s_start);
|
|
|
|
}
|
|
|
|
|
2009-01-15 16:10:45 +03:00
|
|
|
static inline char* mhl_str_trim(char* str)
|
|
|
|
{
|
2009-01-28 00:04:12 +03:00
|
|
|
if (!str) return NULL; /* NULL string ?! bail out. */
|
2009-01-15 16:10:45 +03:00
|
|
|
|
2009-01-28 00:04:12 +03:00
|
|
|
/* find the first non-space */
|
2009-02-04 14:00:44 +03:00
|
|
|
char* start; for (start=str; ((*str) && (!ISSPACE(*str))); str++);
|
2009-01-15 16:10:45 +03:00
|
|
|
|
2009-01-28 00:04:12 +03:00
|
|
|
/* only spaces ? */
|
2009-01-15 16:10:45 +03:00
|
|
|
if (!(*str)) { *str = 0; return str; }
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
/* get the size (cannot be empty - caught above) */
|
2009-01-15 16:10:45 +03:00
|
|
|
size_t _sz = strlen(str);
|
|
|
|
|
2009-01-28 00:04:12 +03:00
|
|
|
/* find the proper end */
|
2009-01-15 16:10:45 +03:00
|
|
|
char* end;
|
2009-02-04 14:00:44 +03:00
|
|
|
for (end=(str+_sz-1); ((end>str) && (ISSPACE(*end))); end--);
|
2009-01-28 00:04:12 +03:00
|
|
|
end[1] = 0; /* terminate, just to be sure */
|
2009-01-15 16:10:45 +03:00
|
|
|
|
2009-01-28 00:04:12 +03:00
|
|
|
/* if we have no leading spaces, just trucate */
|
2009-01-15 16:10:45 +03:00
|
|
|
if (start==str) { end++; *end = 0; return str; }
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
/* if it's only one char, dont need memmove for that */
|
2009-01-15 16:10:45 +03:00
|
|
|
if (start==end) { str[0]=*start; str[1]=0; return str; }
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
/* by here we have a (non-empty) region between start and end */
|
2009-01-15 16:10:45 +03:00
|
|
|
memmove(str,start,(end-start+1));
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void mhl_str_toupper(char* str)
|
|
|
|
{
|
|
|
|
if (str)
|
|
|
|
for (;*str;str++)
|
2009-02-04 14:00:44 +03:00
|
|
|
*str = TOUPPER(*str);
|
2009-01-15 16:10:45 +03:00
|
|
|
}
|
|
|
|
|
2009-02-04 13:51:30 +03:00
|
|
|
/* note: we use ((char*)(1)) as terminator - NULL is a valid argument ! */
|
|
|
|
static const char * mhl_s_c_sep__ = (const char *)1;
|
2009-01-15 16:10:45 +03:00
|
|
|
/* _NEVER_ call this function directly ! */
|
2009-02-04 13:51:30 +03:00
|
|
|
static inline char* mhl_str_concat_hlp__(const char* va_start_dummy, ...)
|
2009-01-15 16:10:45 +03:00
|
|
|
{
|
2009-02-04 13:51:30 +03:00
|
|
|
char * result;
|
|
|
|
size_t result_len = 0;
|
|
|
|
char * p;
|
|
|
|
const char * chunk;
|
2009-01-15 16:10:45 +03:00
|
|
|
|
|
|
|
va_list args;
|
2009-02-04 13:51:30 +03:00
|
|
|
va_start(args,va_start_dummy);
|
|
|
|
while ((chunk = va_arg(args, const char*)) != mhl_s_c_sep__)
|
2009-01-15 16:10:45 +03:00
|
|
|
{
|
2009-02-04 13:51:30 +03:00
|
|
|
if (chunk)
|
2009-01-15 16:10:45 +03:00
|
|
|
{
|
2009-02-04 13:51:30 +03:00
|
|
|
result_len += strlen (chunk);
|
2009-01-15 16:10:45 +03:00
|
|
|
}
|
|
|
|
}
|
2009-02-03 22:09:40 +03:00
|
|
|
va_end(args);
|
2009-01-15 16:10:45 +03:00
|
|
|
|
2009-02-04 13:51:30 +03:00
|
|
|
if (result_len == 0)
|
2009-01-15 16:10:45 +03:00
|
|
|
return mhl_str_dup("");
|
|
|
|
|
2009-02-04 13:51:30 +03:00
|
|
|
/* now as we know how much to copy, allocate the buffer + '\0'*/
|
|
|
|
result = (char*)mhl_mem_alloc_u (result_len + 1);
|
|
|
|
|
|
|
|
p = result;
|
|
|
|
|
|
|
|
va_start(args,va_start_dummy);
|
|
|
|
while ((chunk = va_arg(args, const char*)) != mhl_s_c_sep__)
|
2009-01-15 16:10:45 +03:00
|
|
|
{
|
2009-02-04 13:51:30 +03:00
|
|
|
if (chunk)
|
|
|
|
{
|
|
|
|
size_t chunk_len = strlen (chunk);
|
|
|
|
memcpy (p, chunk, chunk_len);
|
|
|
|
p += chunk_len;
|
|
|
|
}
|
2009-01-15 16:10:45 +03:00
|
|
|
}
|
2009-02-04 13:51:30 +03:00
|
|
|
va_end(args);
|
2009-01-15 16:10:45 +03:00
|
|
|
|
2009-02-04 13:51:30 +03:00
|
|
|
*p = '\0';
|
|
|
|
return result;
|
2009-01-15 16:10:45 +03:00
|
|
|
}
|
|
|
|
|
2009-02-04 13:51:30 +03:00
|
|
|
#define mhl_str_concat(...) (mhl_str_concat_hlp__(mhl_s_c_sep__, __VA_ARGS__, mhl_s_c_sep__))
|
2009-01-15 16:10:45 +03:00
|
|
|
|
|
|
|
static inline char* mhl_str_reverse(char* ptr)
|
|
|
|
{
|
2009-01-28 00:04:12 +03:00
|
|
|
if (!ptr) return NULL; /* missing string */
|
|
|
|
if (!(ptr[0] && ptr[1])) return ptr; /* empty or 1-ch string */
|
2009-01-15 16:10:45 +03:00
|
|
|
|
|
|
|
size_t _sz = strlen(ptr);
|
|
|
|
char* start = ptr;
|
|
|
|
char* end = ptr+_sz-1;
|
|
|
|
|
|
|
|
while (start<end)
|
|
|
|
{
|
|
|
|
char c = *start;
|
|
|
|
*start = *end;
|
|
|
|
*end = c;
|
|
|
|
start++;
|
|
|
|
end--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2009-01-30 10:31:28 +03:00
|
|
|
/*
|
|
|
|
* strcpy is unsafe on overlapping memory areas, so define memmove-alike
|
|
|
|
* string function. Has sense only when dest <= src.
|
|
|
|
*/
|
|
|
|
static inline char * mhl_strmove(char * dest, const char * src)
|
|
|
|
{
|
|
|
|
size_t n = strlen (src) + 1; /* + '\0' */
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
/* strictly speaking, this invokes undefined behavior as soon as dest and src are pointers into different objects. */
|
2009-01-30 10:31:28 +03:00
|
|
|
assert (dest<=src);
|
|
|
|
|
|
|
|
return memmove(dest, src, n);
|
|
|
|
}
|
2009-01-31 23:18:19 +03:00
|
|
|
|
2009-01-31 18:51:58 +03:00
|
|
|
static inline char* mhl_str_dir_plus_file(const char* dirname, const char* filename)
|
|
|
|
{
|
|
|
|
/* make sure we have valid strings */
|
|
|
|
if (!dirname)
|
|
|
|
dirname="";
|
|
|
|
|
|
|
|
if (!filename)
|
|
|
|
filename="";
|
|
|
|
|
|
|
|
/* skip leading slashes on filename */
|
|
|
|
while (*filename == '/')
|
|
|
|
filename++;
|
|
|
|
|
|
|
|
/* skip trailing slashes on dirname */
|
2009-02-01 23:01:49 +03:00
|
|
|
size_t dnlen = strlen(dirname);
|
|
|
|
while ((dnlen != 0) && (dirname[dnlen-1]=='/'))
|
2009-01-31 18:51:58 +03:00
|
|
|
dnlen--;
|
|
|
|
|
2009-02-01 23:01:49 +03:00
|
|
|
size_t fnlen = strlen(filename);
|
2009-01-31 18:51:58 +03:00
|
|
|
char* buffer = mhl_mem_alloc_z(dnlen+fnlen+2); /* enough space for dirname, /, filename, zero */
|
|
|
|
char* ptr = buffer;
|
|
|
|
|
|
|
|
memcpy(ptr, dirname, dnlen);
|
|
|
|
ptr+=dnlen;
|
|
|
|
*ptr = '/';
|
|
|
|
ptr++;
|
|
|
|
memcpy(ptr, filename, fnlen);
|
|
|
|
ptr+=fnlen;
|
|
|
|
*ptr = 0;
|
|
|
|
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2009-02-04 14:00:44 +03:00
|
|
|
#endif /* MHL_STRING_H */
|