diff options
author | Taru Karttunen <taruti@taruti.net> | 2011-03-30 15:46:40 +0300 |
---|---|---|
committer | Taru Karttunen <taruti@taruti.net> | 2011-03-30 15:46:40 +0300 |
commit | e5888a1ffdae813d7575f5fb02275c6bb07e5199 (patch) | |
tree | d8d51eac403f07814b9e936eed0c9a79195e2450 /sys/src/libstdio/_IO_putc.c |
Import sources from 2011-03-30 iso image
Diffstat (limited to 'sys/src/libstdio/_IO_putc.c')
-rwxr-xr-x | sys/src/libstdio/_IO_putc.c | 105 |
1 files changed, 105 insertions, 0 deletions
diff --git a/sys/src/libstdio/_IO_putc.c b/sys/src/libstdio/_IO_putc.c new file mode 100755 index 000000000..4c3a74238 --- /dev/null +++ b/sys/src/libstdio/_IO_putc.c @@ -0,0 +1,105 @@ +/* + * pANS stdio -- _IO_putc, _IO_cleanup + */ +#include "iolib.h" +void _IO_cleanup(void){ + fflush(NULL); +} +/* + * Look this over for simplification + */ +int _IO_putc(int c, FILE *f){ + int cnt; + static int first=1; + switch(f->state){ + case RD: + f->state=ERR; + case ERR: + case CLOSED: + return EOF; + case OPEN: + _IO_setvbuf(f); + /* fall through */ + case RDWR: + case END: + f->rp=f->buf+f->bufl; + if(f->flags&LINEBUF){ + f->wp=f->rp; + f->lp=f->buf; + } + else + f->wp=f->buf; + break; + } + if(first){ + atexit(_IO_cleanup); + first=0; + } + if(f->flags&STRING){ + f->rp=f->buf+f->bufl; + if(f->wp==f->rp){ + if(f->flags&BALLOC) + f->buf=realloc(f->buf, f->bufl+BUFSIZ); + else{ + f->state=ERR; + return EOF; + } + if(f->buf==NULL){ + f->state=ERR; + return EOF; + } + f->rp=f->buf+f->bufl; + f->bufl+=BUFSIZ; + } + *f->wp++=c; + } + else if(f->flags&LINEBUF){ + if(f->lp==f->rp){ + cnt=f->lp-f->buf; + if(f->flags&APPEND) seek(f->fd, 0L, 2); + if(cnt!=0 && write(f->fd, f->buf, cnt)!=cnt){ + f->state=ERR; + return EOF; + } + f->lp=f->buf; + } + *f->lp++=c; + if(c=='\n'){ + cnt=f->lp-f->buf; + if(f->flags&APPEND) seek(f->fd, 0L, 2); + if(cnt!=0 && write(f->fd, f->buf, cnt)!=cnt){ + f->state=ERR; + return EOF; + } + f->lp=f->buf; + } + } + else if(f->buf==f->unbuf){ + f->unbuf[0]=c; + if(f->flags&APPEND) seek(f->fd, 0L, 2); + if(write(f->fd, f->buf, 1)!=1){ + f->state=ERR; + return EOF; + } + } + else{ + if(f->wp==f->rp){ + cnt=f->wp-f->buf; + if(f->flags&APPEND) seek(f->fd, 0L, 2); + if(cnt!=0 && write(f->fd, f->buf, cnt)!=cnt){ + f->state=ERR; + return EOF; + } + f->wp=f->buf; + f->rp=f->buf+f->bufl; + } + *f->wp++=c; + } + f->state=WR; + /* + * Make sure EOF looks different from putc(-1) + * Should be able to cast to unsigned char, but + * there's a vc bug preventing that from working + */ + return c&0xff; +} |