strbuf.c 2.2 KB
Newer Older
1
#include "cache.h"
2
#include "strbuf.h"
3 4

void strbuf_init(struct strbuf *sb) {
5
	memset(sb, 0, sizeof(*sb));
6 7
}

8
void strbuf_release(struct strbuf *sb) {
9
	free(sb->buf);
10 11 12 13 14 15 16 17 18 19 20
	memset(sb, 0, sizeof(*sb));
}

void strbuf_reset(struct strbuf *sb) {
	if (sb->len)
		strbuf_setlen(sb, 0);
	sb->eof = 0;
}

char *strbuf_detach(struct strbuf *sb) {
	char *res = sb->buf;
21
	strbuf_init(sb);
22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
	return res;
}

void strbuf_grow(struct strbuf *sb, size_t extra) {
	if (sb->len + extra + 1 <= sb->len)
		die("you want to use way too much memory");
	ALLOC_GROW(sb->buf, sb->len + extra + 1, sb->alloc);
}

void strbuf_add(struct strbuf *sb, const void *data, size_t len) {
	strbuf_grow(sb, len);
	memcpy(sb->buf + sb->len, data, len);
	strbuf_setlen(sb, sb->len + len);
}

void strbuf_addf(struct strbuf *sb, const char *fmt, ...) {
	int len;
	va_list ap;

	va_start(ap, fmt);
	len = vsnprintf(sb->buf + sb->len, sb->alloc - sb->len, fmt, ap);
	va_end(ap);
	if (len < 0) {
		len = 0;
	}
	if (len >= strbuf_avail(sb)) {
		strbuf_grow(sb, len);
		va_start(ap, fmt);
		len = vsnprintf(sb->buf + sb->len, sb->alloc - sb->len, fmt, ap);
		va_end(ap);
		if (len >= strbuf_avail(sb)) {
			die("this should not happen, your snprintf is broken");
		}
	}
	strbuf_setlen(sb, sb->len + len);
57 58
}

59 60 61 62 63 64 65
size_t strbuf_fread(struct strbuf *sb, size_t size, FILE *f) {
	size_t res;

	strbuf_grow(sb, size);
	res = fread(sb->buf + sb->len, 1, size, f);
	if (res > 0) {
		strbuf_setlen(sb, sb->len + res);
66
	}
67
	return res;
68 69
}

70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
ssize_t strbuf_read(struct strbuf *sb, int fd)
{
	size_t oldlen = sb->len;

	for (;;) {
		ssize_t cnt;

		strbuf_grow(sb, 8192);
		cnt = xread(fd, sb->buf + sb->len, sb->alloc - sb->len - 1);
		if (cnt < 0) {
			strbuf_setlen(sb, oldlen);
			return -1;
		}
		if (!cnt)
			break;
		sb->len += cnt;
	}

	sb->buf[sb->len] = '\0';
	return sb->len - oldlen;
90 91 92 93 94
}

void read_line(struct strbuf *sb, FILE *fp, int term) {
	int ch;
	if (feof(fp)) {
95
		strbuf_release(sb);
96 97 98
		sb->eof = 1;
		return;
	}
99 100

	strbuf_reset(sb);
101 102 103
	while ((ch = fgetc(fp)) != EOF) {
		if (ch == term)
			break;
104 105
		strbuf_grow(sb, 1);
		sb->buf[sb->len++] = ch;
106
	}
107 108
	if (ch == EOF && sb->len == 0) {
		strbuf_release(sb);
J
Junio C Hamano 已提交
109
		sb->eof = 1;
110 111 112 113
	}

	strbuf_grow(sb, 1);
	sb->buf[sb->len] = '\0';
114
}