425 lines
8.9 KiB
C
425 lines
8.9 KiB
C
/****************************************************************************
|
|
* apps/interpreters/bas/bas_str.c
|
|
* Dynamic strings.
|
|
*
|
|
* Copyright (c) 1999-2014 Michael Haardt
|
|
*
|
|
* 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.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <assert.h>
|
|
#include <ctype.h>
|
|
#include <stdarg.h>
|
|
#include <stddef.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "bas_str.h"
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
int cistrcmp(const char *s, const char *r)
|
|
{
|
|
assert(s != (char *)0);
|
|
assert(r != (char *)0);
|
|
while (*s && tolower(*s) == tolower(*r))
|
|
{
|
|
++s;
|
|
++r;
|
|
}
|
|
|
|
return ((tolower(*s) - tolower(*r)));
|
|
}
|
|
|
|
struct String *String_new(struct String *self)
|
|
{
|
|
assert(self != (struct String *)0);
|
|
self->length = 0;
|
|
self->character = (char *)0;
|
|
self->field = (struct StringField *)0;
|
|
return self;
|
|
}
|
|
|
|
void String_destroy(struct String *self)
|
|
{
|
|
assert(self != (struct String *)0);
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
if (self->length)
|
|
{
|
|
free(self->character);
|
|
}
|
|
}
|
|
|
|
int String_joinField(struct String *self, struct StringField *field,
|
|
char *character, size_t length)
|
|
{
|
|
struct String **n;
|
|
|
|
assert(self != (struct String *)0);
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
self->field = field;
|
|
if ((n =
|
|
(struct String **)realloc(field->refStrings,
|
|
sizeof(struct String *) * (field->refCount +
|
|
1))) ==
|
|
(struct String **)0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
field->refStrings = n;
|
|
field->refStrings[field->refCount] = self;
|
|
++field->refCount;
|
|
if (self->length)
|
|
{
|
|
free(self->character);
|
|
}
|
|
|
|
self->character = character;
|
|
self->length = length;
|
|
return 0;
|
|
}
|
|
|
|
void String_leaveField(struct String *self)
|
|
{
|
|
struct StringField *field;
|
|
int i;
|
|
struct String **ref;
|
|
|
|
assert(self != (struct String *)0);
|
|
field = self->field;
|
|
assert(field != (struct StringField *)0);
|
|
for (i = 0, ref = field->refStrings; i < field->refCount; ++i, ++ref)
|
|
{
|
|
if (*ref == self)
|
|
{
|
|
int further = --field->refCount - i;
|
|
|
|
if (further)
|
|
{
|
|
memmove(ref, ref + 1, further * sizeof(struct String **));
|
|
}
|
|
|
|
self->character = (char *)0;
|
|
self->length = 0;
|
|
self->field = (struct StringField *)0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
assert(0);
|
|
}
|
|
|
|
struct String *String_clone(struct String *self, const struct String *original)
|
|
{
|
|
assert(self != (struct String *)0);
|
|
String_new(self);
|
|
String_appendString(self, original);
|
|
return self;
|
|
}
|
|
|
|
int String_size(struct String *self, size_t length)
|
|
{
|
|
char *n;
|
|
|
|
assert(self != (struct String *)0);
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
if (length)
|
|
{
|
|
if (length > self->length)
|
|
{
|
|
if ((n = realloc(self->character, length + 1)) == (char *)0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
self->character = n;
|
|
}
|
|
|
|
self->character[length] = '\0';
|
|
}
|
|
else
|
|
{
|
|
if (self->length)
|
|
{
|
|
free(self->character);
|
|
}
|
|
|
|
self->character = (char *)0;
|
|
}
|
|
|
|
self->length = length;
|
|
return 0;
|
|
}
|
|
|
|
int String_appendString(struct String *self, const struct String *app)
|
|
{
|
|
size_t oldlength = self->length;
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
if (app->length == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (String_size(self, self->length + app->length) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
memcpy(self->character + oldlength, app->character, app->length);
|
|
return 0;
|
|
}
|
|
|
|
int String_appendChar(struct String *self, char ch)
|
|
{
|
|
size_t oldlength = self->length;
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
if (String_size(self, self->length + 1) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
self->character[oldlength] = ch;
|
|
return 0;
|
|
}
|
|
|
|
int String_appendChars(struct String *self, const char *ch)
|
|
{
|
|
size_t oldlength = self->length;
|
|
size_t chlen = strlen(ch);
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
if (String_size(self, self->length + chlen) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
memcpy(self->character + oldlength, ch, chlen);
|
|
return 0;
|
|
}
|
|
|
|
int String_appendPrintf(struct String *self, const char *fmt, ...)
|
|
{
|
|
char buf[1024];
|
|
size_t l, j;
|
|
va_list ap;
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
va_start(ap, fmt);
|
|
l = vsprintf(buf, fmt, ap);
|
|
va_end(ap);
|
|
j = self->length;
|
|
if (String_size(self, j + l) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
memcpy(self->character + j, buf, l);
|
|
return 0;
|
|
}
|
|
|
|
int String_insertChar(struct String *self, size_t where, char ch)
|
|
{
|
|
size_t oldlength = self->length;
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
assert(where < oldlength);
|
|
if (String_size(self, self->length + 1) == -1)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
memmove(self->character + where + 1, self->character + where,
|
|
oldlength - where);
|
|
self->character[where] = ch;
|
|
return 0;
|
|
}
|
|
|
|
int String_delete(struct String *self, size_t where, size_t len)
|
|
{
|
|
size_t oldlength = self->length;
|
|
|
|
if (self->field)
|
|
{
|
|
String_leaveField(self);
|
|
}
|
|
|
|
assert(where < oldlength);
|
|
assert(len > 0);
|
|
if ((where + len) < oldlength)
|
|
{
|
|
memmove(self->character + where, self->character + where + len,
|
|
oldlength - where - len);
|
|
}
|
|
|
|
self->character[self->length -= len] = '\0';
|
|
return 0;
|
|
}
|
|
|
|
void String_ucase(struct String *self)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < self->length; ++i)
|
|
{
|
|
self->character[i] = toupper(self->character[i]);
|
|
}
|
|
}
|
|
|
|
void String_lcase(struct String *self)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < self->length; ++i)
|
|
{
|
|
self->character[i] = tolower(self->character[i]);
|
|
}
|
|
}
|
|
|
|
int String_cmp(const struct String *self, const struct String *s)
|
|
{
|
|
size_t pos;
|
|
int res;
|
|
const char *thisch, *sch;
|
|
|
|
for (pos = 0, thisch = self->character, sch = s->character;
|
|
pos < self->length && pos < s->length; ++pos, ++thisch, ++sch)
|
|
{
|
|
if ((res = (*thisch - *sch)))
|
|
{
|
|
return res;
|
|
}
|
|
}
|
|
|
|
return (self->length - s->length);
|
|
}
|
|
|
|
void String_lset(struct String *self, const struct String *s)
|
|
{
|
|
size_t copy;
|
|
|
|
copy = (self->length < s->length ? self->length : s->length);
|
|
if (copy)
|
|
{
|
|
memcpy(self->character, s->character, copy);
|
|
}
|
|
|
|
if (copy < self->length)
|
|
{
|
|
memset(self->character + copy, ' ', self->length - copy);
|
|
}
|
|
}
|
|
|
|
void String_rset(struct String *self, const struct String *s)
|
|
{
|
|
size_t copy;
|
|
|
|
copy = (self->length < s->length ? self->length : s->length);
|
|
if (copy)
|
|
{
|
|
memcpy(self->character + self->length - copy, s->character, copy);
|
|
}
|
|
|
|
if (copy < self->length)
|
|
{
|
|
memset(self->character, ' ', self->length - copy);
|
|
}
|
|
}
|
|
|
|
void String_set(struct String *self, size_t pos, const struct String *s,
|
|
size_t length)
|
|
{
|
|
if (self->length >= pos)
|
|
{
|
|
if (self->length < (pos + length))
|
|
{
|
|
length = self->length - pos;
|
|
}
|
|
|
|
if (length)
|
|
{
|
|
memcpy(self->character + pos, s->character, length);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct StringField *StringField_new(struct StringField *self)
|
|
{
|
|
self->refStrings = (struct String **)0;
|
|
self->refCount = 0;
|
|
return self;
|
|
}
|
|
|
|
void StringField_destroy(struct StringField *self)
|
|
{
|
|
int i;
|
|
|
|
for (i = self->refCount; i > 0;)
|
|
{
|
|
String_leaveField(self->refStrings[--i]);
|
|
}
|
|
|
|
self->refCount = -1;
|
|
free(self->refStrings);
|
|
}
|