1
/* Copyright (C) 2000 MySQL AB
3
This program is free software; you can redistribute it and/or modify
4
it under the terms of the GNU General Public License as published by
5
the Free Software Foundation; version 2 of the License.
7
This program is distributed in the hope that it will be useful,
8
but WITHOUT ANY WARRANTY; without even the implied warranty of
9
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
GNU General Public License for more details.
12
You should have received a copy of the GNU General Public License
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
16
/* readline for batch mode */
18
#include <my_global.h>
21
#include "my_readline.h"
23
static bool init_line_buffer(LINE_BUFFER *buffer,File file,ulong size,
25
static bool init_line_buffer_from_string(LINE_BUFFER *buffer,char * str);
26
static size_t fill_buffer(LINE_BUFFER *buffer);
27
static char *intern_read_line(LINE_BUFFER *buffer,ulong *out_length);
30
LINE_BUFFER *batch_readline_init(ulong max_size,FILE *file)
32
LINE_BUFFER *line_buff;
33
if (!(line_buff=(LINE_BUFFER*)
34
my_malloc(sizeof(*line_buff),MYF(MY_WME | MY_ZEROFILL))))
36
if (init_line_buffer(line_buff,fileno(file),IO_SIZE,max_size))
38
my_free(line_buff,MYF(0));
45
char *batch_readline(LINE_BUFFER *line_buff)
50
if (!(pos=intern_read_line(line_buff,&out_length)))
52
if (out_length && pos[out_length-1] == '\n')
53
if (--out_length && pos[out_length-1] == '\r') /* Remove '\n' */
54
out_length--; /* Remove '\r' */
55
line_buff->read_length=out_length;
61
void batch_readline_end(LINE_BUFFER *line_buff)
65
my_free(line_buff->buffer,MYF(MY_ALLOW_ZERO_PTR));
66
my_free(line_buff,MYF(0));
71
LINE_BUFFER *batch_readline_command(LINE_BUFFER *line_buff, char * str)
74
if (!(line_buff=(LINE_BUFFER*)
75
my_malloc(sizeof(*line_buff),MYF(MY_WME | MY_ZEROFILL))))
77
if (init_line_buffer_from_string(line_buff,str))
79
my_free(line_buff,MYF(0));
86
/*****************************************************************************
87
Functions to handle buffered readings of lines from a stream
88
******************************************************************************/
91
init_line_buffer(LINE_BUFFER *buffer,File file,ulong size,ulong max_buffer)
95
buffer->max_size=max_buffer;
96
if (!(buffer->buffer = (char*) my_malloc(buffer->bufread+1,
97
MYF(MY_WME | MY_FAE))))
99
buffer->end_of_line=buffer->end=buffer->buffer;
100
buffer->buffer[0]=0; /* For easy start test */
105
init_line_buffer_from_string can be called on the same buffer
106
several times. the resulting buffer will contain a
107
concatenation of all strings separated by spaces
109
static bool init_line_buffer_from_string(LINE_BUFFER *buffer,char * str)
111
uint old_length=(uint)(buffer->end - buffer->buffer);
112
uint length= (uint) strlen(str);
113
if (!(buffer->buffer= buffer->start_of_line= buffer->end_of_line=
114
(char*) my_realloc((uchar*) buffer->buffer, old_length+length+2,
115
MYF(MY_FAE|MY_ALLOW_ZERO_PTR))))
117
buffer->end= buffer->buffer + old_length;
120
memcpy(buffer->end, str, length);
121
buffer->end[length]= '\n';
122
buffer->end[length+1]= 0;
123
buffer->end+= length+1;
131
Fill the buffer retaining the last n bytes at the beginning of the
132
newly filled buffer (for backward context). Returns the number of new
133
bytes read from disk.
136
static size_t fill_buffer(LINE_BUFFER *buffer)
139
uint bufbytes= (uint) (buffer->end - buffer->start_of_line);
142
return 0; /* Everything read */
144
/* See if we need to grow the buffer. */
148
uint start_offset=(uint) (buffer->start_of_line - buffer->buffer);
149
read_count=(buffer->bufread - bufbytes)/IO_SIZE;
150
if ((read_count*=IO_SIZE))
152
buffer->bufread *= 2;
153
if (!(buffer->buffer = (char*) my_realloc(buffer->buffer,
155
MYF(MY_WME | MY_FAE))))
157
buffer->start_of_line=buffer->buffer+start_offset;
158
buffer->end=buffer->buffer+bufbytes;
161
/* Shift stuff down. */
162
if (buffer->start_of_line != buffer->buffer)
164
bmove(buffer->buffer,buffer->start_of_line,(uint) bufbytes);
165
buffer->end=buffer->buffer+bufbytes;
168
/* Read in new stuff. */
169
if ((read_count= my_read(buffer->file, (uchar*) buffer->end, read_count,
170
MYF(MY_WME))) == MY_FILE_ERROR)
173
DBUG_PRINT("fill_buff", ("Got %lu bytes", (ulong) read_count));
175
/* Kludge to pretend every nonempty file ends with a newline. */
176
if (!read_count && bufbytes && buffer->end[-1] != '\n')
178
buffer->eof = read_count = 1;
181
buffer->end_of_line=(buffer->start_of_line=buffer->buffer)+bufbytes;
182
buffer->end+=read_count;
183
*buffer->end=0; /* Sentinel */
189
char *intern_read_line(LINE_BUFFER *buffer,ulong *out_length)
193
DBUG_ENTER("intern_read_line");
195
buffer->start_of_line=buffer->end_of_line;
198
pos=buffer->end_of_line;
199
while (*pos != '\n' && *pos)
201
if (pos == buffer->end)
203
if ((uint) (pos - buffer->start_of_line) < buffer->max_size)
205
if (!(length=fill_buffer(buffer)) || length == (size_t) -1)
209
pos--; /* break line here */
211
buffer->end_of_line=pos+1;
212
*out_length=(ulong) (pos + 1 - buffer->eof - buffer->start_of_line);
213
DBUG_RETURN(buffer->start_of_line);