replace sbs_cut with more powerful sbs_substr

This commit is contained in:
Enno Rehling 2018-11-27 20:50:58 +01:00
parent 413a83c1ec
commit cd0ba434b0
4 changed files with 58 additions and 13 deletions

View file

@ -1550,7 +1550,7 @@ void pump_paragraph(sbstring *sbp, stream *out, size_t maxlen, bool isfinal)
return; return;
} }
else if (next > begin + maxlen) { else if (next > begin + maxlen) {
size_t len = pos - begin; ptrdiff_t len = pos - begin;
swrite(begin, 1, len, out); swrite(begin, 1, len, out);
newline(out); newline(out);
@ -1558,8 +1558,8 @@ void pump_paragraph(sbstring *sbp, stream *out, size_t maxlen, bool isfinal)
++pos; ++pos;
++len; ++len;
} }
assert(len <= INT_MAX); assert(len <= SIZE_MAX);
sbs_cut(sbp, (int)len); sbs_substr(sbp, len, SIZE_MAX);
break; break;
} }
pos = next; pos = next;

View file

@ -302,19 +302,24 @@ void sbs_strcpy(struct sbstring *sbs, const char *str)
assert(sbs->begin + sbs->size >= sbs->end); assert(sbs->begin + sbs->size >= sbs->end);
} }
void sbs_cut(sbstring *sbs, int bytes) void sbs_substr(sbstring *sbs, ptrdiff_t pos, size_t len)
{ {
if (bytes > 0) { if (pos > sbs->end - sbs->begin) {
size_t len = sbs_length(sbs) - bytes; /* starting past end of string, do nothing */
memmove(sbs->begin, sbs->begin + bytes, len + 1); sbs->end = sbs->begin;
sbs->end = sbs->begin + len;
} }
else if (bytes < 0) { if (pos >= 0) {
size_t len = sbs_length(sbs) + bytes; size_t sz = sbs->end - (sbs->begin + pos);
sbs->end = sbs->begin + len; if (len > sz) len = sz;
if (len - pos > 0) {
memmove(sbs->begin, sbs->begin + pos, len);
}
else {
memcpy(sbs->begin, sbs->begin + pos, len);
}
sbs->end = sbs->begin + len;
sbs->end[0] = '\0';
} }
assert(sbs->begin + sbs->size >= sbs->end);
assert(sbs->end[0] == '\0');
} }
size_t sbs_length(const struct sbstring *sbs) size_t sbs_length(const struct sbstring *sbs)

View file

@ -53,7 +53,7 @@ extern "C" {
void sbs_strcat(struct sbstring *sbs, const char *str); void sbs_strcat(struct sbstring *sbs, const char *str);
void sbs_strncat(struct sbstring *sbs, const char *str, size_t size); void sbs_strncat(struct sbstring *sbs, const char *str, size_t size);
void sbs_strcpy(struct sbstring *sbs, const char *str); void sbs_strcpy(struct sbstring *sbs, const char *str);
void sbs_cut(struct sbstring *sbp, int bytes); void sbs_substr(struct sbstring *sbp, ptrdiff_t pos, size_t len);
size_t sbs_length(const struct sbstring *sbs); size_t sbs_length(const struct sbstring *sbs);
/* benchmark for units: /* benchmark for units:

View file

@ -183,6 +183,45 @@ static void test_sbs_strcat(CuTest * tc)
CuAssertStrEquals(tc, "1234123", sbs.begin); CuAssertStrEquals(tc, "1234123", sbs.begin);
} }
static void test_sbs_substr(CuTest * tc)
{
char buffer[10];
sbstring sbs;
buffer[9] = 'A';
sbs_init(&sbs, buffer, sizeof(buffer));
sbs_strcpy(&sbs, "12345678");
CuAssertStrEquals(tc, "12345678", buffer);
sbs_substr(&sbs, 0, 4);
CuAssertStrEquals(tc, "1234", buffer);
CuAssertIntEquals(tc, 4, (int)sbs_length(&sbs));
sbs_strcpy(&sbs, "12345678");
CuAssertStrEquals(tc, "12345678", buffer);
sbs_substr(&sbs, 4, 4);
CuAssertStrEquals(tc, "5678", buffer);
CuAssertIntEquals(tc, 4, (int)sbs_length(&sbs));
sbs_strcpy(&sbs, "12345678");
CuAssertStrEquals(tc, "12345678", buffer);
sbs_substr(&sbs, 2, 4);
CuAssertStrEquals(tc, "3456", buffer);
CuAssertIntEquals(tc, 4, (int)sbs_length(&sbs));
sbs_strcpy(&sbs, "12345678");
CuAssertStrEquals(tc, "12345678", buffer);
sbs_substr(&sbs, 4, 4);
CuAssertStrEquals(tc, "5678", buffer);
CuAssertIntEquals(tc, 4, (int)sbs_length(&sbs));
sbs_strcpy(&sbs, "12345678");
CuAssertStrEquals(tc, "12345678", buffer);
sbs_substr(&sbs, 4, 8);
CuAssertStrEquals(tc, "5678", buffer);
CuAssertIntEquals(tc, 4, (int)sbs_length(&sbs));
CuAssertIntEquals(tc, 'A', buffer[9]);
}
CuSuite *get_strings_suite(void) CuSuite *get_strings_suite(void)
{ {
CuSuite *suite = CuSuiteNew(); CuSuite *suite = CuSuiteNew();
@ -196,5 +235,6 @@ CuSuite *get_strings_suite(void)
SUITE_ADD_TEST(suite, test_str_strlcpy); SUITE_ADD_TEST(suite, test_str_strlcpy);
SUITE_ADD_TEST(suite, test_sbstring); SUITE_ADD_TEST(suite, test_sbstring);
SUITE_ADD_TEST(suite, test_sbs_strcat); SUITE_ADD_TEST(suite, test_sbs_strcat);
SUITE_ADD_TEST(suite, test_sbs_substr);
return suite; return suite;
} }