~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to client/linebuffer.cc

Merged in latest plugin-slot-reorg.

Show diffs side-by-side

added added

removed removed

Lines of Context:
15
15
 
16
16
/* readline for batch mode */
17
17
 
18
 
#include "client_priv.h"
 
18
#include <drizzled/global.h>
19
19
#include <mysys/my_sys.h>
20
 
#include <mystrings/m_string.h>
21
 
#include "my_readline.h"
22
 
 
23
 
static bool init_line_buffer(LINE_BUFFER *buffer,File file,uint32_t size,
24
 
                            uint32_t max_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,uint32_t *out_length);
28
 
 
29
 
 
30
 
LINE_BUFFER *batch_readline_init(uint32_t max_size,FILE *file)
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
 
    free(line_buff);
39
 
    return 0;
40
 
  }
41
 
  return line_buff;
42
 
}
43
 
 
44
 
 
45
 
char *batch_readline(LINE_BUFFER *line_buff)
46
 
{
47
 
  char *pos;
48
 
  uint32_t out_length;
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
 
    free(line_buff->buffer);
66
 
    free(line_buff);
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
 
    free(line_buff);
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
91
 
init_line_buffer(LINE_BUFFER *buffer,File file,uint32_t size,uint32_t max_buffer)
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((unsigned char*) 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
 
  {
164
 
    memcpy(buffer->buffer, buffer->start_of_line, (uint) bufbytes);
165
 
    buffer->end=buffer->buffer+bufbytes;
166
 
  }
167
 
 
168
 
  /* Read in new stuff. */
169
 
  if ((read_count= my_read(buffer->file, (unsigned char*) 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
 
 
187
 
char *intern_read_line(LINE_BUFFER *buffer,uint32_t *out_length)
188
 
{
189
 
  char *pos;
190
 
  size_t length;
191
 
 
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)
 
20
#include "client/linebuffer.h"
 
21
 
 
22
#include <vector>
 
23
 
 
24
using namespace std;
 
25
 
 
26
LineBuffer::LineBuffer(uint32_t my_max_size,FILE *my_file)
 
27
  :
 
28
    file(my_file),
 
29
    line(),
 
30
    max_size(my_max_size),
 
31
    eof(false)
 
32
{
 
33
  line.reserve(max_size);
 
34
}
 
35
 
 
36
void LineBuffer::addString(const string &str)
 
37
{
 
38
  buffer << str << endl;
 
39
}
 
40
 
 
41
char *LineBuffer::readline()
 
42
{
 
43
  uint32_t read_count;
 
44
 
 
45
  if (file && !eof)
 
46
  {
 
47
    if ((read_count= my_read(fileno(file),
 
48
                             (unsigned char *) (&line[0]),
 
49
                             max_size-1,MYF(MY_WME))))
200
50
    {
201
 
      if ((uint) (pos - buffer->start_of_line) < buffer->max_size)
202
 
      {
203
 
        if (!(length=fill_buffer(buffer)) || length == (size_t) -1)
204
 
          return(0);
205
 
        continue;
206
 
      }
207
 
      pos--;                                    /* break line here */
 
51
      line[read_count+1]= '\0';
 
52
      buffer << &line[0];
208
53
    }
209
 
    buffer->end_of_line=pos+1;
210
 
    *out_length=(uint32_t) (pos + 1 - buffer->eof - buffer->start_of_line);
211
 
    return(buffer->start_of_line);
 
54
    else
 
55
      eof= true;
212
56
  }
 
57
 
 
58
  buffer.getline(&line[0],max_size);
 
59
 
 
60
  if (buffer.eof())
 
61
    return 0;
 
62
  else
 
63
    return &line[0];
213
64
}
 
65