__getdelim.c
来自「一个C源代码分析器」· C语言 代码 · 共 171 行
C
171 行
/* Copyright (C) 1991, 1992 Free Software Foundation, Inc.This file is part of the GNU C Library.The GNU C Library is free software; you can redistribute it and/ormodify it under the terms of the GNU Library General Public License aspublished by the Free Software Foundation; either version 2 of theLicense, or (at your option) any later version.The GNU C Library is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNULibrary General Public License for more details.You should have received a copy of the GNU Library General PublicLicense along with the GNU C Library; see the file COPYING.LIB. Ifnot, write to the Free Software Foundation, Inc., 675 Mass Ave,Cambridge, MA 02139, USA. */#include <ansidecl.h>#include <stddef.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <limits.h>/* Read up to (and including) a TERMINATOR from STREAM into *LINEPTR (and null-terminate it). *LINEPTR is a pointer returned from malloc (or NULL), pointing to *N characters of space. It is realloc'd as necessary. Returns the number of characters read (not including the null terminator), or -1 on error or EOF. */ssize_tDEFUN(__getdelim, (lineptr, n, terminator, stream), char **lineptr AND size_t *n AND int terminator AND FILE *stream){ char *line, *p; size_t size, copy; if (!__validfp (stream) || lineptr == NULL || n == NULL) { errno = EINVAL; return -1; } if (ferror (stream)) return -1; /* Make sure we have a line buffer to start with. */ if (*lineptr == NULL || *n < 2) /* !seen and no buf yet need 2 chars. */ {#ifndef MAX_CANON#define MAX_CANON 256#endif line = realloc (*lineptr, MAX_CANON); if (line == NULL) return -1; *lineptr = line; *n = MAX_CANON; } line = *lineptr; size = *n; copy = size; p = line; if (stream->__buffer == NULL && stream->__userbuf) { /* Unbuffered stream. Not much optimization to do. */ while (1) { size_t len; while (--copy > 0) { register int c = getc (stream); if (c == EOF) goto lose; else if ((*p++ = c) == terminator) goto win; } /* Need to enlarge the line buffer. */ len = p - line; size *= 2; line = realloc (line, size); if (line == NULL) goto lose; *lineptr = line; *n = size; p = line + len; copy = size - len; } } else { /* Leave space for the terminating null. */ --copy; if (!stream->__seen || stream->__buffer == NULL || stream->__pushed_back) { /* Do one with getc to allocate a buffer. */ int c = getc (stream); if (c == EOF) goto lose; *p++ = c; if (c == terminator) goto win; --copy; } while (1) { size_t i; char *found; i = stream->__get_limit - stream->__bufp; if (i == 0) { /* Refill the buffer. */ int c = __fillbf (stream); if (c == EOF) goto lose; *p++ = c; if (c == terminator) goto win; i = stream->__get_limit - stream->__bufp; } if (i > copy) i = copy; found = (char *) __memccpy ((PTR) p, stream->__bufp, terminator, i); if (found != NULL) { stream->__bufp += found - p; p = found; goto win; } stream->__bufp += i; p += i; copy -= i; if (copy == 0) { /* Need to enlarge the line buffer. */ size_t len = p - line; size *= 2; line = realloc (line, size); if (line == NULL) goto lose; *lineptr = line; *n = size; p = line + len; copy = size - len; /* Leave space for the terminating null. */ --copy; } } } lose: if (p == *lineptr) return -1; /* Return a partial line since we got an error in the middle. */ win: *p = '\0'; return p - *lineptr;}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?