X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Flibpspp%2Fstr.c;h=71f54474d57f995f8c5b73572e1d8f761b7e934e;hb=ebd0dcdfe65da02a6e541ceea0226c9b0b8d7a47;hp=8243aa0690aad24a9b519db9bd1838724d32d729;hpb=7c08a6e1009cf60847e770a77a73c650e9326379;p=pspp diff --git a/src/libpspp/str.c b/src/libpspp/str.c index 8243aa0690..71f54474d5 100644 --- a/src/libpspp/str.c +++ b/src/libpspp/str.c @@ -1,5 +1,5 @@ /* PSPP - a program for statistical analysis. - Copyright (C) 1997-9, 2000, 2006, 2009 Free Software Foundation, Inc. + Copyright (C) 1997-9, 2000, 2006, 2009, 2010 Free Software Foundation, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -1222,6 +1222,17 @@ ds_cstr (const struct string *st_) return st->ss.string; } +/* Returns the value of ST as a null-terminated string and then + reinitialized ST as an empty string. The caller must free the + returned string with free(). */ +char * +ds_steal_cstr (struct string *st) +{ + char *s = ds_cstr (st); + ds_init_empty (st); + return s; +} + /* Reads characters from STREAM and appends them to ST, stopping after MAX_LENGTH characters, after appending a newline, or after an I/O error or end of file was encountered, whichever @@ -1229,9 +1240,9 @@ ds_cstr (const struct string *st_) to ST, false if no characters were read before an I/O error or end of file (or if MAX_LENGTH was 0). - This function accepts LF, CR LF, and CR sequences as new-line, - and translates each of them to a single '\n' new-line - character in ST. */ + This function treats LF and CR LF sequences as new-line, + translating each of them to a single '\n' new-line character + in ST. */ bool ds_read_line (struct string *st, FILE *stream, size_t max_length) { @@ -1240,21 +1251,36 @@ ds_read_line (struct string *st, FILE *stream, size_t max_length) for (length = 0; length < max_length; length++) { int c = getc (stream); - if (c == EOF) - break; - - if (c == '\r') + switch (c) { + case EOF: + return length > 0; + + case '\n': + ds_put_char (st, c); + return true; + + case '\r': c = getc (stream); - if (c != '\n') + if (c == '\n') { + /* CR followed by LF is special: translate to \n. */ + ds_put_char (st, '\n'); + return true; + } + else + { + /* CR followed by anything else is just CR. */ + ds_put_char (st, '\r'); + if (c == EOF) + return true; ungetc (c, stream); - c = '\n'; } + break; + + default: + ds_put_char (st, c); } - ds_put_char (st, c); - if (c == '\n') - return true; } return length > 0;