mirror of
https://github.com/subsurface/subsurface.git
synced 2024-11-30 22:20:21 +00:00
Fix totally broken put_vformat() implementation
I'm ashamed. put_vbuffer() worked perfectly fine for the normal case when everything fit in our simple buffer on-stack, but the fallback case was broken in so many ways that I'm just going to go sit in a corner and cry myself to sleep. And dammit, I _knew_ how to do it right. I knew you had to do a "va_copy()" and couldn't just keep re-using 'args'. I've done this before. But I half-arsed it, and nobody ever noticed, because you wouldn't do C style format strings for big strings. "128 bytes is enough for everybody". And as penance for this idiocy, I just spent too much time trying to figure out what was wrong in my git loading code when my debug printouts caused SIGSEGV's. Sigh. Anyway, now it should hopefully be correct, and the code is smarter about things too, not having that extra buffer since we already *have* a buffer in the "struct membuffer" we are working with. Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org> Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
This commit is contained in:
parent
cc3a184adf
commit
989c8f0110
1 changed files with 20 additions and 12 deletions
28
membuffer.c
28
membuffer.c
|
@ -76,20 +76,28 @@ void put_string(struct membuffer *b, const char *str)
|
||||||
|
|
||||||
void put_vformat(struct membuffer *b, const char *fmt, va_list args)
|
void put_vformat(struct membuffer *b, const char *fmt, va_list args)
|
||||||
{
|
{
|
||||||
/* Handle the common case on the stack */
|
int room = 128;
|
||||||
char buffer[128], *p;
|
|
||||||
int len;
|
|
||||||
|
|
||||||
len = vsnprintf(buffer, sizeof(buffer), fmt, args);
|
for (;;) {
|
||||||
if (len <= sizeof(buffer)) {
|
int len;
|
||||||
put_bytes(b, buffer, len);
|
va_list copy;
|
||||||
|
char *target;
|
||||||
|
|
||||||
|
make_room(b, room);
|
||||||
|
room = b->alloc - b->len;
|
||||||
|
target = b->buffer + b->len;
|
||||||
|
|
||||||
|
va_copy(copy, args);
|
||||||
|
len = vsnprintf(target, room, fmt, copy);
|
||||||
|
va_end(copy);
|
||||||
|
|
||||||
|
if (len < room) {
|
||||||
|
b->len += len;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
p = malloc(len);
|
room = len+1;
|
||||||
len = vsnprintf(p, len, fmt, args);
|
}
|
||||||
put_bytes(b, p, len);
|
|
||||||
free(p);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void put_format(struct membuffer *b, const char *fmt, ...)
|
void put_format(struct membuffer *b, const char *fmt, ...)
|
||||||
|
|
Loading…
Reference in a new issue