|  | /* Copyright (c) 2002-2006 Lucent Technologies; see LICENSE */ | 
|  | /* | 
|  | * Plan 9 port version must include libc.h in order to | 
|  | * get Plan 9 debugging malloc, which sometimes returns | 
|  | * different pointers than the standard malloc. | 
|  | */ | 
|  | #ifdef PLAN9PORT | 
|  | #include <u.h> | 
|  | #include <libc.h> | 
|  | #include "fmtdef.h" | 
|  | #else | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include "plan9.h" | 
|  | #include "fmt.h" | 
|  | #include "fmtdef.h" | 
|  | #endif | 
|  |  | 
|  | static int | 
|  | fmtStrFlush(Fmt *f) | 
|  | { | 
|  | char *s; | 
|  | int n; | 
|  |  | 
|  | if(f->start == nil) | 
|  | return 0; | 
|  | n = (uintptr)f->farg; | 
|  | n *= 2; | 
|  | s = (char*)f->start; | 
|  | f->start = realloc(s, n); | 
|  | if(f->start == nil){ | 
|  | f->farg = nil; | 
|  | f->to = nil; | 
|  | f->stop = nil; | 
|  | free(s); | 
|  | return 0; | 
|  | } | 
|  | f->farg = (void*)(uintptr)n; | 
|  | f->to = (char*)f->start + ((char*)f->to - s); | 
|  | f->stop = (char*)f->start + n - 1; | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | int | 
|  | fmtstrinit(Fmt *f) | 
|  | { | 
|  | int n; | 
|  |  | 
|  | memset(f, 0, sizeof *f); | 
|  | f->runes = 0; | 
|  | n = 32; | 
|  | f->start = malloc(n); | 
|  | if(f->start == nil) | 
|  | return -1; | 
|  | f->to = f->start; | 
|  | f->stop = (char*)f->start + n - 1; | 
|  | f->flush = fmtStrFlush; | 
|  | f->farg = (void*)(uintptr)n; | 
|  | f->nfmt = 0; | 
|  | fmtlocaleinit(f, nil, nil, nil); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * print into an allocated string buffer | 
|  | */ | 
|  | char* | 
|  | vsmprint(char *fmt, va_list args) | 
|  | { | 
|  | Fmt f; | 
|  | int n; | 
|  |  | 
|  | if(fmtstrinit(&f) < 0) | 
|  | return nil; | 
|  | VA_COPY(f.args,args); | 
|  | n = dofmt(&f, fmt); | 
|  | VA_END(f.args); | 
|  | if(n < 0){ | 
|  | free(f.start); | 
|  | return nil; | 
|  | } | 
|  | return fmtstrflush(&f); | 
|  | } |