t2ex/bsd_source/lib/libc/src_bsd/stdio/ungetc.c | bare source | permlink (0.02 seconds) |
1: /* Add __sungetc() for call from locked region */ 2: 3: /* $OpenBSD: ungetc.c,v 1.12 2009/11/09 00:18:27 kurt Exp $ */ 4: /*- 5: * Copyright (c) 1990, 1993 6: * The Regents of the University of California. All rights reserved. 7: * 8: * This code is derived from software contributed to Berkeley by 9: * Chris Torek. 10: * 11: * Redistribution and use in source and binary forms, with or without 12: * modification, are permitted provided that the following conditions 13: * are met: 14: * 1. Redistributions of source code must retain the above copyright 15: * notice, this list of conditions and the following disclaimer. 16: * 2. Redistributions in binary form must reproduce the above copyright 17: * notice, this list of conditions and the following disclaimer in the 18: * documentation and/or other materials provided with the distribution. 19: * 3. Neither the name of the University nor the names of its contributors 20: * may be used to endorse or promote products derived from this software 21: * without specific prior written permission. 22: * 23: * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 24: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 25: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 26: * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 27: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 28: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 29: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 30: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 31: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 32: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 33: * SUCH DAMAGE. 34: */ 35: 36: #include <stdio.h> 37: #include <stdlib.h> 38: #include <string.h> 39: #include "local.h" 40: 41: static int __submore(FILE *); 42: /* 43: * Expand the ungetc buffer `in place'. That is, adjust fp->_p when 44: * the buffer moves, so that it points the same distance from the end, 45: * and move the bytes in the buffer around as necessary so that they 46: * are all at the end (stack-style). 47: */ 48: static int 49: __submore(FILE *fp) 50: { 51: int i; 52: unsigned char *p; 53: 54: if (_UB(fp)._base == fp->_ubuf) { 55: /* 56: * Get a new buffer (rather than expanding the old one). 57: */ 58: if ((p = malloc((size_t)BUFSIZ)) == NULL) 59: return (EOF); 60: _UB(fp)._base = p; 61: _UB(fp)._size = BUFSIZ; 62: p += BUFSIZ - sizeof(fp->_ubuf); 63: for (i = sizeof(fp->_ubuf); --i >= 0;) 64: p[i] = fp->_ubuf[i]; 65: fp->_p = p; 66: return (0); 67: } 68: i = _UB(fp)._size; 69: p = realloc(_UB(fp)._base, i << 1); 70: if (p == NULL) 71: return (EOF); 72: /* no overlap (hence can use memcpy) because we doubled the size */ 73: (void)memcpy((void *)(p + i), (void *)p, (size_t)i); 74: fp->_p = p + i; 75: _UB(fp)._base = p; 76: _UB(fp)._size = i << 1; 77: return (0); 78: } 79: 80: int 81: __sungetc(int c, FILE *fp) 82: { 83: if (c == EOF) 84: return (EOF); 85: if (!__sdidinit) 86: __sinit(); 87: /*FLOCKFILE(fp);*/ 88: _SET_ORIENTATION(fp, -1); 89: if ((fp->_flags & __SRD) == 0) { 90: /* 91: * Not already reading: no good unless reading-and-writing. 92: * Otherwise, flush any current write stuff. 93: */ 94: if ((fp->_flags & __SRW) == 0) { 95: error: /*FUNLOCKFILE(fp);*/ 96: return (EOF); 97: } 98: if (fp->_flags & __SWR) { 99: if (__sflush(fp)) 100: goto error; 101: fp->_flags &= ~__SWR; 102: fp->_w = 0; 103: fp->_lbfsize = 0; 104: } 105: fp->_flags |= __SRD; 106: } 107: c = (unsigned char)c; 108: 109: /* 110: * If we are in the middle of ungetc'ing, just continue. 111: * This may require expanding the current ungetc buffer. 112: */ 113: if (HASUB(fp)) { 114: if (fp->_r >= _UB(fp)._size && __submore(fp)) 115: goto error; 116: *--fp->_p = c; 117: inc_ret: fp->_r++; 118: /*FUNLOCKFILE(fp);*/ 119: return (c); 120: } 121: fp->_flags &= ~__SEOF; 122: 123: /* 124: * If we can handle this by simply backing up, do so, 125: * but never replace the original character. 126: * (This makes sscanf() work when scanning `const' data.) 127: */ 128: if (fp->_bf._base != NULL && fp->_p > fp->_bf._base && 129: fp->_p[-1] == c) { 130: fp->_p--; 131: goto inc_ret; 132: } 133: 134: /* 135: * Create an ungetc buffer. 136: * Initially, we will use the `reserve' buffer. 137: */ 138: fp->_ur = fp->_r; 139: fp->_up = fp->_p; 140: _UB(fp)._base = fp->_ubuf; 141: _UB(fp)._size = sizeof(fp->_ubuf); 142: fp->_ubuf[sizeof(fp->_ubuf) - 1] = c; 143: fp->_p = &fp->_ubuf[sizeof(fp->_ubuf) - 1]; 144: fp->_r = 1; 145: /*FUNLOCKFILE(fp);*/ 146: return (c); 147: } 148: 149: int 150: ungetc(int c, FILE *fp) 151: { 152: FLOCKFILE(fp); 153: c = __sungetc(c, fp); 154: FUNLOCKFILE(fp); 155: return (c); 156: }