blob: c39038723ed4a90f99f70a694eae58ba488556b6 [file] [log] [blame]
David Barr3bbaec02010-08-09 17:39:43 -05001/*
2 * Licensed under a two-clause BSD-style license.
3 * See LICENSE for details.
4 */
5
6#include "git-compat-util.h"
7#include "line_buffer.h"
Jonathan Niederdeadcef2010-11-06 12:01:28 -05008#include "strbuf.h"
David Barr3bbaec02010-08-09 17:39:43 -05009
David Barr3bbaec02010-08-09 17:39:43 -050010#define COPY_BUFFER_LEN 4096
David Barr3bbaec02010-08-09 17:39:43 -050011
Jonathan Niedere5e45ca2010-10-10 21:41:06 -050012int buffer_init(struct line_buffer *buf, const char *filename)
David Barr3bbaec02010-08-09 17:39:43 -050013{
Jonathan Niederd3508222010-10-10 21:39:21 -050014 buf->infile = filename ? fopen(filename, "r") : stdin;
15 if (!buf->infile)
David Barr3bbaec02010-08-09 17:39:43 -050016 return -1;
17 return 0;
18}
19
Jonathan Niedercb3f87c2011-01-02 21:09:38 -060020int buffer_fdinit(struct line_buffer *buf, int fd)
21{
22 buf->infile = fdopen(fd, "r");
23 if (!buf->infile)
24 return -1;
25 return 0;
26}
27
Jonathan Niederb1c9b792011-01-02 21:10:59 -060028int buffer_tmpfile_init(struct line_buffer *buf)
29{
30 buf->infile = tmpfile();
31 if (!buf->infile)
32 return -1;
33 return 0;
34}
35
Jonathan Niedere5e45ca2010-10-10 21:41:06 -050036int buffer_deinit(struct line_buffer *buf)
David Barr3bbaec02010-08-09 17:39:43 -050037{
38 int err;
Jonathan Niederd3508222010-10-10 21:39:21 -050039 if (buf->infile == stdin)
40 return ferror(buf->infile);
41 err = ferror(buf->infile);
42 err |= fclose(buf->infile);
David Barr3bbaec02010-08-09 17:39:43 -050043 return err;
44}
45
Jonathan Niederb1c9b792011-01-02 21:10:59 -060046FILE *buffer_tmpfile_rewind(struct line_buffer *buf)
47{
48 rewind(buf->infile);
49 return buf->infile;
50}
51
52long buffer_tmpfile_prepare_to_read(struct line_buffer *buf)
53{
54 long pos = ftell(buf->infile);
55 if (pos < 0)
56 return error("ftell error: %s", strerror(errno));
57 if (fseek(buf->infile, 0, SEEK_SET))
58 return error("seek error: %s", strerror(errno));
59 return pos;
60}
61
Jonathan Niederefc749b2010-10-10 21:51:21 -050062int buffer_ferror(struct line_buffer *buf)
63{
64 return ferror(buf->infile);
65}
66
Jonathan Niedercc193f12011-01-02 21:06:32 -060067int buffer_read_char(struct line_buffer *buf)
68{
69 return fgetc(buf->infile);
70}
71
David Barr3bbaec02010-08-09 17:39:43 -050072/* Read a line without trailing newline. */
Jonathan Niedere5e45ca2010-10-10 21:41:06 -050073char *buffer_read_line(struct line_buffer *buf)
David Barr3bbaec02010-08-09 17:39:43 -050074{
75 char *end;
Jonathan Niederd3508222010-10-10 21:39:21 -050076 if (!fgets(buf->line_buffer, sizeof(buf->line_buffer), buf->infile))
David Barr3bbaec02010-08-09 17:39:43 -050077 /* Error or data exhausted. */
78 return NULL;
Jonathan Niederd3508222010-10-10 21:39:21 -050079 end = buf->line_buffer + strlen(buf->line_buffer);
David Barr3bbaec02010-08-09 17:39:43 -050080 if (end[-1] == '\n')
81 end[-1] = '\0';
Jonathan Niederd3508222010-10-10 21:39:21 -050082 else if (feof(buf->infile))
David Barr3bbaec02010-08-09 17:39:43 -050083 ; /* No newline at end of file. That's fine. */
84 else
85 /*
86 * Line was too long.
87 * There is probably a saner way to deal with this,
88 * but for now let's return an error.
89 */
90 return NULL;
Jonathan Niederd3508222010-10-10 21:39:21 -050091 return buf->line_buffer;
David Barr3bbaec02010-08-09 17:39:43 -050092}
93
Jonathan Niedere832f432011-01-02 21:05:46 -060094void buffer_read_binary(struct line_buffer *buf,
95 struct strbuf *sb, uint32_t size)
96{
97 strbuf_fread(sb, size, buf->infile);
98}
99
Jonathan Nieder26557fc2010-12-28 04:26:17 -0600100off_t buffer_copy_bytes(struct line_buffer *buf, off_t nbytes)
David Barr3bbaec02010-08-09 17:39:43 -0500101{
Jonathan Nieder4d21bec2010-10-10 21:37:10 -0500102 char byte_buffer[COPY_BUFFER_LEN];
Jonathan Nieder26557fc2010-12-28 04:26:17 -0600103 off_t done = 0;
104 while (done < nbytes && !feof(buf->infile) && !ferror(buf->infile)) {
105 off_t len = nbytes - done;
106 size_t in = len < COPY_BUFFER_LEN ? len : COPY_BUFFER_LEN;
Jonathan Niederd3508222010-10-10 21:39:21 -0500107 in = fread(byte_buffer, 1, in, buf->infile);
Jonathan Nieder26557fc2010-12-28 04:26:17 -0600108 done += in;
David Barr3bbaec02010-08-09 17:39:43 -0500109 fwrite(byte_buffer, 1, in, stdout);
Jonathan Nieder26557fc2010-12-28 04:26:17 -0600110 if (ferror(stdout))
111 return done + buffer_skip_bytes(buf, nbytes - done);
David Barr3bbaec02010-08-09 17:39:43 -0500112 }
Jonathan Nieder26557fc2010-12-28 04:26:17 -0600113 return done;
David Barr3bbaec02010-08-09 17:39:43 -0500114}
115
Jonathan Niederd234f542010-10-10 21:44:21 -0500116off_t buffer_skip_bytes(struct line_buffer *buf, off_t nbytes)
David Barr3bbaec02010-08-09 17:39:43 -0500117{
Jonathan Nieder4d21bec2010-10-10 21:37:10 -0500118 char byte_buffer[COPY_BUFFER_LEN];
Jonathan Niederd234f542010-10-10 21:44:21 -0500119 off_t done = 0;
120 while (done < nbytes && !feof(buf->infile) && !ferror(buf->infile)) {
121 off_t len = nbytes - done;
122 size_t in = len < COPY_BUFFER_LEN ? len : COPY_BUFFER_LEN;
123 done += fread(byte_buffer, 1, in, buf->infile);
David Barr3bbaec02010-08-09 17:39:43 -0500124 }
Jonathan Niederd234f542010-10-10 21:44:21 -0500125 return done;
David Barr3bbaec02010-08-09 17:39:43 -0500126}
127
Jonathan Niedere5e45ca2010-10-10 21:41:06 -0500128void buffer_reset(struct line_buffer *buf)
David Barr3bbaec02010-08-09 17:39:43 -0500129{
David Barr3bbaec02010-08-09 17:39:43 -0500130}