|  | /* 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 | 
|  | runeFmtStrFlush(Fmt *f) | 
|  | { | 
|  | Rune *s; | 
|  | int n; | 
|  |  | 
|  | if(f->start == nil) | 
|  | return 0; | 
|  | n = (uintptr)f->farg; | 
|  | n *= 2; | 
|  | s = (Rune*)f->start; | 
|  | f->start = realloc(s, sizeof(Rune)*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 = (Rune*)f->start + ((Rune*)f->to - s); | 
|  | f->stop = (Rune*)f->start + n - 1; | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | int | 
|  | runefmtstrinit(Fmt *f) | 
|  | { | 
|  | int n; | 
|  |  | 
|  | memset(f, 0, sizeof *f); | 
|  | f->runes = 1; | 
|  | n = 32; | 
|  | f->start = malloc(sizeof(Rune)*n); | 
|  | if(f->start == nil) | 
|  | return -1; | 
|  | f->to = f->start; | 
|  | f->stop = (Rune*)f->start + n - 1; | 
|  | f->flush = runeFmtStrFlush; | 
|  | f->farg = (void*)(uintptr)n; | 
|  | f->nfmt = 0; | 
|  | fmtlocaleinit(f, nil, nil, nil); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * print into an allocated string buffer | 
|  | */ | 
|  | Rune* | 
|  | runevsmprint(char *fmt, va_list args) | 
|  | { | 
|  | Fmt f; | 
|  | int n; | 
|  |  | 
|  | if(runefmtstrinit(&f) < 0) | 
|  | return nil; | 
|  | VA_COPY(f.args,args); | 
|  | n = dofmt(&f, fmt); | 
|  | VA_END(f.args); | 
|  | if(f.start == nil) | 
|  | return nil; | 
|  | if(n < 0){ | 
|  | free(f.start); | 
|  | return nil; | 
|  | } | 
|  | *(Rune*)f.to = '\0'; | 
|  | return (Rune*)f.start; | 
|  | } |