| #include "test-lib.h" |
| #include "strbuf.h" |
| |
| /* wrapper that supplies tests with an empty, initialized strbuf */ |
| static void setup(void (*f)(struct strbuf*, const void*), |
| const void *data) |
| { |
| struct strbuf buf = STRBUF_INIT; |
| |
| f(&buf, data); |
| strbuf_release(&buf); |
| check_uint(buf.len, ==, 0); |
| check_uint(buf.alloc, ==, 0); |
| } |
| |
| /* wrapper that supplies tests with a populated, initialized strbuf */ |
| static void setup_populated(void (*f)(struct strbuf*, const void*), |
| const char *init_str, const void *data) |
| { |
| struct strbuf buf = STRBUF_INIT; |
| |
| strbuf_addstr(&buf, init_str); |
| check_uint(buf.len, ==, strlen(init_str)); |
| f(&buf, data); |
| strbuf_release(&buf); |
| check_uint(buf.len, ==, 0); |
| check_uint(buf.alloc, ==, 0); |
| } |
| |
| static int assert_sane_strbuf(struct strbuf *buf) |
| { |
| /* Initialized strbufs should always have a non-NULL buffer */ |
| if (!check(!!buf->buf)) |
| return 0; |
| /* Buffers should always be NUL-terminated */ |
| if (!check_char(buf->buf[buf->len], ==, '\0')) |
| return 0; |
| /* |
| * Freshly-initialized strbufs may not have a dynamically allocated |
| * buffer |
| */ |
| if (buf->len == 0 && buf->alloc == 0) |
| return 1; |
| /* alloc must be at least one byte larger than len */ |
| return check_uint(buf->len, <, buf->alloc); |
| } |
| |
| static void t_static_init(void) |
| { |
| struct strbuf buf = STRBUF_INIT; |
| |
| check_uint(buf.len, ==, 0); |
| check_uint(buf.alloc, ==, 0); |
| check_char(buf.buf[0], ==, '\0'); |
| } |
| |
| static void t_dynamic_init(void) |
| { |
| struct strbuf buf; |
| |
| strbuf_init(&buf, 1024); |
| check(assert_sane_strbuf(&buf)); |
| check_uint(buf.len, ==, 0); |
| check_uint(buf.alloc, >=, 1024); |
| check_char(buf.buf[0], ==, '\0'); |
| strbuf_release(&buf); |
| } |
| |
| static void t_addch(struct strbuf *buf, const void *data) |
| { |
| const char *p_ch = data; |
| const char ch = *p_ch; |
| size_t orig_alloc = buf->alloc; |
| size_t orig_len = buf->len; |
| |
| if (!check(assert_sane_strbuf(buf))) |
| return; |
| strbuf_addch(buf, ch); |
| if (!check(assert_sane_strbuf(buf))) |
| return; |
| if (!(check_uint(buf->len, ==, orig_len + 1) && |
| check_uint(buf->alloc, >=, orig_alloc))) |
| return; /* avoid de-referencing buf->buf */ |
| check_char(buf->buf[buf->len - 1], ==, ch); |
| check_char(buf->buf[buf->len], ==, '\0'); |
| } |
| |
| static void t_addstr(struct strbuf *buf, const void *data) |
| { |
| const char *text = data; |
| size_t len = strlen(text); |
| size_t orig_alloc = buf->alloc; |
| size_t orig_len = buf->len; |
| |
| if (!check(assert_sane_strbuf(buf))) |
| return; |
| strbuf_addstr(buf, text); |
| if (!check(assert_sane_strbuf(buf))) |
| return; |
| if (!(check_uint(buf->len, ==, orig_len + len) && |
| check_uint(buf->alloc, >=, orig_alloc) && |
| check_uint(buf->alloc, >, orig_len + len) && |
| check_char(buf->buf[orig_len + len], ==, '\0'))) |
| return; |
| check_str(buf->buf + orig_len, text); |
| } |
| |
| int cmd_main(int argc UNUSED, const char **argv UNUSED) |
| { |
| if (!TEST(t_static_init(), "static initialization works")) |
| test_skip_all("STRBUF_INIT is broken"); |
| TEST(t_dynamic_init(), "dynamic initialization works"); |
| TEST(setup(t_addch, "a"), "strbuf_addch adds char"); |
| TEST(setup(t_addch, ""), "strbuf_addch adds NUL char"); |
| TEST(setup_populated(t_addch, "initial value", "a"), |
| "strbuf_addch appends to initial value"); |
| TEST(setup(t_addstr, "hello there"), "strbuf_addstr adds string"); |
| TEST(setup_populated(t_addstr, "initial value", "hello there"), |
| "strbuf_addstr appends string to initial value"); |
| |
| return test_done(); |
| } |