~drizzle-trunk/drizzle/development

1 by brian
clean slate
1
/* Copyright (C) 2000 MySQL AB
2
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.
6
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.
11
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 */
15
16
/* readline for batch mode */
17
212.5.39 by Monty Taylor
Phew. Moved my_base and my_global.
18
#include "client_priv.h"
212.5.13 by Monty Taylor
Moved my_sys/my_pthread/my_nosys and mysys_err to mysys.
19
#include <mysys/my_sys.h>
212.5.18 by Monty Taylor
Moved m_ctype, m_string and my_bitmap. Removed t_ctype.
20
#include <mystrings/m_string.h>
1 by brian
clean slate
21
#include "my_readline.h"
22
288 by Brian Aker
ulong cleanp in client apps
23
static bool init_line_buffer(LINE_BUFFER *buffer,File file,uint32_t size,
24
			    uint32_t max_size);
1 by brian
clean slate
25
static bool init_line_buffer_from_string(LINE_BUFFER *buffer,char * str);
26
static size_t fill_buffer(LINE_BUFFER *buffer);
288 by Brian Aker
ulong cleanp in client apps
27
static char *intern_read_line(LINE_BUFFER *buffer,uint32_t *out_length);
28
29
30
LINE_BUFFER *batch_readline_init(uint32_t max_size,FILE *file)
1 by brian
clean slate
31
{
32
  LINE_BUFFER *line_buff;
33
  if (!(line_buff=(LINE_BUFFER*)
34
        my_malloc(sizeof(*line_buff),MYF(MY_WME | MY_ZEROFILL))))
35
    return 0;
36
  if (init_line_buffer(line_buff,fileno(file),IO_SIZE,max_size))
37
  {
38
    my_free(line_buff,MYF(0));
39
    return 0;
40
  }
41
  return line_buff;
42
}
43
44
45
char *batch_readline(LINE_BUFFER *line_buff)
46
{
47
  char *pos;
288 by Brian Aker
ulong cleanp in client apps
48
  uint32_t out_length;
1 by brian
clean slate
49
50
  if (!(pos=intern_read_line(line_buff,&out_length)))
51
    return 0;
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;
56
  pos[out_length]=0;
57
  return pos;
58
}
59
60
61
void batch_readline_end(LINE_BUFFER *line_buff)
62
{
63
  if (line_buff)
64
  {
65
    my_free(line_buff->buffer,MYF(MY_ALLOW_ZERO_PTR));
66
    my_free(line_buff,MYF(0));
67
  }
68
}
69
70
71
LINE_BUFFER *batch_readline_command(LINE_BUFFER *line_buff, char * str)
72
{
73
  if (!line_buff)
74
    if (!(line_buff=(LINE_BUFFER*)
75
          my_malloc(sizeof(*line_buff),MYF(MY_WME | MY_ZEROFILL))))
76
      return 0;
77
  if (init_line_buffer_from_string(line_buff,str))
78
  {
79
    my_free(line_buff,MYF(0));
80
    return 0;
81
  }
82
  return line_buff;
83
}
84
85
86
/*****************************************************************************
87
      Functions to handle buffered readings of lines from a stream
88
******************************************************************************/
89
90
static bool
288 by Brian Aker
ulong cleanp in client apps
91
init_line_buffer(LINE_BUFFER *buffer,File file,uint32_t size,uint32_t max_buffer)
1 by brian
clean slate
92
{
93
  buffer->file=file;
94
  buffer->bufread=size;
95
  buffer->max_size=max_buffer;
96
  if (!(buffer->buffer = (char*) my_malloc(buffer->bufread+1,
97
					   MYF(MY_WME | MY_FAE))))
98
    return 1;
99
  buffer->end_of_line=buffer->end=buffer->buffer;
100
  buffer->buffer[0]=0;				/* For easy start test */
101
  return 0;
102
}
103
104
/*
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
108
*/
109
static bool init_line_buffer_from_string(LINE_BUFFER *buffer,char * str)
110
{
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))))
116
    return 1;
117
  buffer->end= buffer->buffer + old_length;
118
  if (old_length)
119
    buffer->end[-1]=' ';
120
  memcpy(buffer->end, str, length);
121
  buffer->end[length]= '\n';
122
  buffer->end[length+1]= 0;
123
  buffer->end+= length+1;
124
  buffer->eof=1;
125
  buffer->max_size=1;
126
  return 0;
127
}
128
129
130
/*
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.
134
*/
135
136
static size_t fill_buffer(LINE_BUFFER *buffer)
137
{
138
  size_t read_count;
139
  uint bufbytes= (uint) (buffer->end - buffer->start_of_line);
140
141
  if (buffer->eof)
142
    return 0;					/* Everything read */
143
144
  /* See if we need to grow the buffer. */
145
146
  for (;;)
147
  {
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))
151
      break;
152
    buffer->bufread *= 2;
153
    if (!(buffer->buffer = (char*) my_realloc(buffer->buffer,
154
					      buffer->bufread+1,
155
					      MYF(MY_WME | MY_FAE))))
156
      return (uint) -1;
157
    buffer->start_of_line=buffer->buffer+start_offset;
158
    buffer->end=buffer->buffer+bufbytes;
159
  }
160
161
  /* Shift stuff down. */
162
  if (buffer->start_of_line != buffer->buffer)
163
  {
212.6.3 by Mats Kindahl
Removing deprecated functions from code and replacing them with C99 equivalents:
164
    memcpy(buffer->buffer, buffer->start_of_line, (uint) bufbytes);
1 by brian
clean slate
165
    buffer->end=buffer->buffer+bufbytes;
166
  }
167
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)
171
    return (size_t) -1;
172
173
  /* Kludge to pretend every nonempty file ends with a newline. */
174
  if (!read_count && bufbytes && buffer->end[-1] != '\n')
175
  {
176
    buffer->eof = read_count = 1;
177
    *buffer->end = '\n';
178
  }
179
  buffer->end_of_line=(buffer->start_of_line=buffer->buffer)+bufbytes;
180
  buffer->end+=read_count;
181
  *buffer->end=0;				/* Sentinel */
182
  return read_count;
183
}
184
185
186
288 by Brian Aker
ulong cleanp in client apps
187
char *intern_read_line(LINE_BUFFER *buffer,uint32_t *out_length)
1 by brian
clean slate
188
{
189
  char *pos;
190
  size_t length;
142.1.2 by Patrick
All DBUG_x removed from client/
191
1 by brian
clean slate
192
193
  buffer->start_of_line=buffer->end_of_line;
194
  for (;;)
195
  {
196
    pos=buffer->end_of_line;
197
    while (*pos != '\n' && *pos)
198
      pos++;
199
    if (pos == buffer->end)
200
    {
201
      if ((uint) (pos - buffer->start_of_line) < buffer->max_size)
202
      {
203
	if (!(length=fill_buffer(buffer)) || length == (size_t) -1)
142.1.2 by Patrick
All DBUG_x removed from client/
204
	  return(0);
1 by brian
clean slate
205
	continue;
206
      }
207
      pos--;					/* break line here */
208
    }
209
    buffer->end_of_line=pos+1;
288 by Brian Aker
ulong cleanp in client apps
210
    *out_length=(uint32_t) (pos + 1 - buffer->eof - buffer->start_of_line);
142.1.2 by Patrick
All DBUG_x removed from client/
211
    return(buffer->start_of_line);
1 by brian
clean slate
212
  }
213
}