summaryrefslogtreecommitdiff
path: root/gnu/lib/libg++/iostream/sgetline.C
blob: 04745969223a8db61da021ee25f70c0d3d1646d8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
//    This is part of the iostream library, providing input/output for C++.
//    Copyright (C) 1991 Per Bothner.
//
//    This library is free software; you can redistribute it and/or
//    modify it under the terms of the GNU Library General Public
//    License as published by the Free Software Foundation; either
//    version 2 of the License, or (at your option) any later version.
//
//    This library is distributed in the hope that it will be useful,
//    but WITHOUT ANY WARRANTY; without even the implied warranty of
//    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
//    Library General Public License for more details.
//
//    You should have received a copy of the GNU Library General Public
//    License along with this library; if not, write to the Free
//    Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.

#include "ioprivate.h"

// Algorithm based on that used by Berkeley pre-4.4 fgets implementation.

// Read chars into buf (of size n), until delim is seen.
// Return number of chars read (at most n-1).
// If extract_delim < 0, leave delimiter unread.
// If extract_delim > 0, insert delim in output.

long streambuf::sgetline(char* buf, size_t n, char delim, int extract_delim)
{
    register char *ptr = buf;
    if (n <= 0)
	return EOF;
    n--; // Leave space for final '\0'.
    do {
	int len = egptr() - gptr();
	if (len <= 0)
	    if (underflow() == EOF)
		break;
	    else
		len = egptr() - gptr();
	if (len >= (int)n)
	    len = n;
	char *t = (char*)memchr((void*)_gptr, delim, len);
	if (t != NULL) {
	    size_t old_len = ptr-buf;
	    len = t - _gptr;
	    if (extract_delim >= 0) {
		t++;
		old_len++;
		if (extract_delim > 0)
		    len++;
	    }
	    memcpy((void*)ptr, (void*)_gptr, len);
	    ptr[len] = 0;
	    _gptr = t;
	    return old_len + len;
	}
	memcpy((void*)ptr, (void*)_gptr, len);
	_gptr += len;
	ptr += len;
	n -= len;
    } while (n != 0);
    *ptr = 0;
    return ptr - buf;
}