~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to libdrizzle/conn.c

  • Committer: Brian Aker
  • Date: 2010-12-08 22:35:56 UTC
  • mfrom: (1819.9.158 update-innobase)
  • Revision ID: brian@tangent.org-20101208223556-37mi4omqg7lkjzf3
Merge in Stewart's changes, 1.3 changes.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*
 
2
 * Drizzle Client & Protocol Library
 
3
 *
 
4
 * Copyright (C) 2008 Eric Day (eday@oddments.org)
 
5
 * All rights reserved.
 
6
 *
 
7
 * Redistribution and use in source and binary forms, with or without
 
8
 * modification, are permitted provided that the following conditions are
 
9
 * met:
 
10
 *
 
11
 *     * Redistributions of source code must retain the above copyright
 
12
 * notice, this list of conditions and the following disclaimer.
 
13
 *
 
14
 *     * Redistributions in binary form must reproduce the above
 
15
 * copyright notice, this list of conditions and the following disclaimer
 
16
 * in the documentation and/or other materials provided with the
 
17
 * distribution.
 
18
 *
 
19
 *     * The names of its contributors may not be used to endorse or
 
20
 * promote products derived from this software without specific prior
 
21
 * written permission.
 
22
 *
 
23
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 
24
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 
25
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 
26
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 
27
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 
28
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 
29
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 
30
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 
31
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 
32
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 
33
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 
34
 *
 
35
 */
 
36
 
 
37
 
 
38
/**
 
39
 * @file
 
40
 * @brief Connection Definitions
 
41
 */
 
42
 
 
43
#include "common.h"
 
44
 
 
45
/**
 
46
 * @addtogroup drizzle_con_static Static Connection Declarations
 
47
 * @ingroup drizzle_con
 
48
 * @{
 
49
 */
 
50
 
 
51
/**
 
52
 * Set socket options for a connection.
 
53
 *
 
54
 * @param[in] con Connection structure previously initialized with
 
55
 *  drizzle_con_create(), drizzle_con_clone(), or related functions.
 
56
 * @return Standard drizzle return value.
 
57
 */
 
58
static drizzle_return_t _con_setsockopt(drizzle_con_st *con);
 
59
 
 
60
/** @} */
 
61
 
 
62
/*
 
63
 * Common Definitions
 
64
 */
 
65
 
 
66
int drizzle_con_fd(const drizzle_con_st *con)
 
67
{
 
68
  return con->fd;
 
69
}
 
70
 
 
71
drizzle_return_t drizzle_con_set_fd(drizzle_con_st *con, int fd)
 
72
{
 
73
  drizzle_return_t ret;
 
74
 
 
75
  con->fd= fd;
 
76
 
 
77
  ret= _con_setsockopt(con);
 
78
  if (ret != DRIZZLE_RETURN_OK)
 
79
    con->drizzle->last_errno= errno;
 
80
 
 
81
  return ret;
 
82
}
 
83
 
 
84
void drizzle_con_close(drizzle_con_st *con)
 
85
{
 
86
  if (con->fd == -1)
 
87
    return;
 
88
 
 
89
  (void)close(con->fd);
 
90
  con->fd= -1;
 
91
 
 
92
  con->options&= (drizzle_con_options_t)~DRIZZLE_CON_READY;
 
93
  con->packet_number= 0;
 
94
  con->buffer_ptr= con->buffer;
 
95
  con->buffer_size= 0;
 
96
  con->events= 0;
 
97
  con->revents= 0;
 
98
 
 
99
  drizzle_state_reset(con);
 
100
}
 
101
 
 
102
drizzle_return_t drizzle_con_set_events(drizzle_con_st *con, short events)
 
103
{
 
104
  drizzle_return_t ret;
 
105
 
 
106
  if ((con->events | events) == con->events)
 
107
    return DRIZZLE_RETURN_OK;
 
108
 
 
109
  con->events|= events;
 
110
 
 
111
  if (con->drizzle->event_watch_fn != NULL)
 
112
  {
 
113
    ret= con->drizzle->event_watch_fn(con, con->events,
 
114
                                      con->drizzle->event_watch_context);
 
115
    if (ret != DRIZZLE_RETURN_OK)
 
116
    {
 
117
      drizzle_con_close(con);
 
118
      return ret;
 
119
    }
 
120
  }
 
121
 
 
122
  return DRIZZLE_RETURN_OK;
 
123
}
 
124
 
 
125
drizzle_return_t drizzle_con_set_revents(drizzle_con_st *con, short revents)
 
126
{
 
127
  drizzle_return_t ret;
 
128
 
 
129
  if (revents != 0)
 
130
    con->options|= DRIZZLE_CON_IO_READY;
 
131
 
 
132
  con->revents= revents;
 
133
 
 
134
  /* Remove external POLLOUT watch if we didn't ask for it. Otherwise we spin
 
135
     forever until another POLLIN state change. This is much more efficient
 
136
     than removing POLLOUT on every state change since some external polling
 
137
     mechanisms need to use a system call to change flags (like Linux epoll). */
 
138
  if (revents & POLLOUT && !(con->events & POLLOUT) &&
 
139
      con->drizzle->event_watch_fn != NULL)
 
140
  {
 
141
    ret= con->drizzle->event_watch_fn(con, con->events,
 
142
                                      con->drizzle->event_watch_context);
 
143
    if (ret != DRIZZLE_RETURN_OK)
 
144
    {
 
145
      drizzle_con_close(con);
 
146
      return ret;
 
147
    }
 
148
  }
 
149
 
 
150
  con->events&= (short)~revents;
 
151
 
 
152
  return DRIZZLE_RETURN_OK;
 
153
}
 
154
 
 
155
drizzle_st *drizzle_con_drizzle(const drizzle_con_st *con)
 
156
{
 
157
  return con->drizzle;
 
158
}
 
159
 
 
160
const char *drizzle_con_error(const drizzle_con_st *con)
 
161
{
 
162
  return drizzle_error(con->drizzle);
 
163
}
 
164
 
 
165
int drizzle_con_errno(const drizzle_con_st *con)
 
166
{
 
167
  return drizzle_errno(con->drizzle);
 
168
}
 
169
 
 
170
uint16_t drizzle_con_error_code(const drizzle_con_st *con)
 
171
{
 
172
  return drizzle_error_code(con->drizzle);
 
173
}
 
174
 
 
175
const char *drizzle_con_sqlstate(const drizzle_con_st *con)
 
176
{
 
177
  return drizzle_sqlstate(con->drizzle);
 
178
}
 
179
 
 
180
drizzle_con_options_t drizzle_con_options(const drizzle_con_st *con)
 
181
{
 
182
  return con->options;
 
183
}
 
184
 
 
185
void drizzle_con_set_options(drizzle_con_st *con,
 
186
                             drizzle_con_options_t options)
 
187
{
 
188
  con->options= options;
 
189
}
 
190
 
 
191
void drizzle_con_add_options(drizzle_con_st *con,
 
192
                             drizzle_con_options_t options)
 
193
{
 
194
  con->options|= options;
 
195
 
 
196
  /* If asking for the experimental Drizzle protocol, clean the MySQL flag. */
 
197
  if (con->options & DRIZZLE_CON_EXPERIMENTAL)
 
198
    con->options&= (drizzle_con_options_t)~DRIZZLE_CON_MYSQL;
 
199
}
 
200
 
 
201
void drizzle_con_remove_options(drizzle_con_st *con,
 
202
                                drizzle_con_options_t options)
 
203
{
 
204
  con->options&= ~options;
 
205
}
 
206
 
 
207
const char *drizzle_con_host(const drizzle_con_st *con)
 
208
{
 
209
  if (con->socket_type == DRIZZLE_CON_SOCKET_TCP)
 
210
  {
 
211
    if (con->socket.tcp.host == NULL && !(con->options & DRIZZLE_CON_LISTEN))
 
212
      return DRIZZLE_DEFAULT_TCP_HOST;
 
213
 
 
214
    return con->socket.tcp.host;
 
215
  }
 
216
 
 
217
  return NULL;
 
218
}
 
219
 
 
220
in_port_t drizzle_con_port(const drizzle_con_st *con)
 
221
{
 
222
  if (con->socket_type == DRIZZLE_CON_SOCKET_TCP)
 
223
  {
 
224
    if (con->socket.tcp.port != 0)
 
225
      return con->socket.tcp.port;
 
226
 
 
227
    if (con->options & DRIZZLE_CON_MYSQL)
 
228
      return DRIZZLE_DEFAULT_TCP_PORT_MYSQL;
 
229
 
 
230
    return DRIZZLE_DEFAULT_TCP_PORT;
 
231
  }
 
232
 
 
233
  return 0;
 
234
}
 
235
 
 
236
void drizzle_con_set_tcp(drizzle_con_st *con, const char *host, in_port_t port)
 
237
{
 
238
  drizzle_con_reset_addrinfo(con);
 
239
 
 
240
  con->socket_type= DRIZZLE_CON_SOCKET_TCP;
 
241
 
 
242
  if (host == NULL)
 
243
    con->socket.tcp.host= NULL;
 
244
  else
 
245
  {
 
246
    con->socket.tcp.host= con->socket.tcp.host_buffer;
 
247
    strncpy(con->socket.tcp.host, host, NI_MAXHOST);
 
248
    con->socket.tcp.host[NI_MAXHOST - 1]= 0;
 
249
  }
 
250
 
 
251
  con->socket.tcp.port= port;
 
252
}
 
253
 
 
254
const char *drizzle_con_user(const drizzle_con_st *con)
 
255
{
 
256
  return con->user;
 
257
}
 
258
 
 
259
const char *drizzle_con_password(const drizzle_con_st *con)
 
260
{
 
261
  return con->password;
 
262
}
 
263
 
 
264
void drizzle_con_set_auth(drizzle_con_st *con, const char *user,
 
265
                          const char *password)
 
266
{
 
267
  if (user == NULL)
 
268
    con->user[0]= 0;
 
269
  else
 
270
  {
 
271
    strncpy(con->user, user, DRIZZLE_MAX_USER_SIZE);
 
272
    con->user[DRIZZLE_MAX_USER_SIZE - 1]= 0;
 
273
  }
 
274
 
 
275
  if (password == NULL)
 
276
    con->password[0]= 0;
 
277
  else
 
278
  {
 
279
    strncpy(con->password, password, DRIZZLE_MAX_PASSWORD_SIZE);
 
280
    con->password[DRIZZLE_MAX_PASSWORD_SIZE - 1]= 0;
 
281
  }
 
282
}
 
283
 
 
284
const char *drizzle_con_db(const drizzle_con_st *con)
 
285
{
 
286
  return con->db;
 
287
}
 
288
 
 
289
void drizzle_con_set_db(drizzle_con_st *con, const char *db)
 
290
{
 
291
  if (db == NULL)
 
292
    con->db[0]= 0;
 
293
  else
 
294
  {
 
295
    strncpy(con->db, db, DRIZZLE_MAX_DB_SIZE);
 
296
    con->db[DRIZZLE_MAX_DB_SIZE - 1]= 0;
 
297
  }
 
298
}
 
299
 
 
300
void *drizzle_con_context(const drizzle_con_st *con)
 
301
{
 
302
  return con->context;
 
303
}
 
304
 
 
305
void drizzle_con_set_context(drizzle_con_st *con, void *context)
 
306
{
 
307
  con->context= context;
 
308
}
 
309
 
 
310
void drizzle_con_set_context_free_fn(drizzle_con_st *con,
 
311
                                     drizzle_con_context_free_fn *function)
 
312
{
 
313
  con->context_free_fn= function;
 
314
}
 
315
 
 
316
uint8_t drizzle_con_protocol_version(const drizzle_con_st *con)
 
317
{
 
318
  return con->protocol_version;
 
319
}
 
320
 
 
321
const char *drizzle_con_server_version(const drizzle_con_st *con)
 
322
{
 
323
  return con->server_version;
 
324
}
 
325
 
 
326
uint32_t drizzle_con_server_version_number(const drizzle_con_st *con)
 
327
{
 
328
  uint32_t major;
 
329
  uint32_t minor;
 
330
  uint32_t version;
 
331
  const char *current;
 
332
  char *end;
 
333
 
 
334
  current= con->server_version;
 
335
 
 
336
  major= (uint32_t)strtoul(current, &end, 10);
 
337
  current= end + 1;
 
338
  minor= (uint32_t)strtoul(current, &end, 10);
 
339
  current= end + 1;
 
340
  version= (uint32_t)strtoul(current, &end, 10);
 
341
 
 
342
  return (major * 10000) + (minor * 100) + version;
 
343
}
 
344
 
 
345
uint32_t drizzle_con_thread_id(const drizzle_con_st *con)
 
346
{
 
347
  return con->thread_id;
 
348
}
 
349
 
 
350
const uint8_t *drizzle_con_scramble(const drizzle_con_st *con)
 
351
{
 
352
  return con->scramble;
 
353
}
 
354
 
 
355
drizzle_capabilities_t drizzle_con_capabilities(const drizzle_con_st *con)
 
356
{
 
357
  return con->capabilities;
 
358
}
 
359
 
 
360
drizzle_charset_t drizzle_con_charset(const drizzle_con_st *con)
 
361
{
 
362
  return con->charset;
 
363
}
 
364
 
 
365
drizzle_con_status_t drizzle_con_status(const drizzle_con_st *con)
 
366
{
 
367
  return con->status;
 
368
}
 
369
 
 
370
uint32_t drizzle_con_max_packet_size(const drizzle_con_st *con)
 
371
{
 
372
  return con->max_packet_size;
 
373
}
 
374
 
 
375
/*
 
376
 * Client Definitions
 
377
 */
 
378
 
 
379
drizzle_return_t drizzle_con_connect(drizzle_con_st *con)
 
380
{
 
381
  if (con->options & DRIZZLE_CON_READY)
 
382
    return DRIZZLE_RETURN_OK;
 
383
 
 
384
  if (drizzle_state_none(con))
 
385
  {
 
386
    if (!(con->options & DRIZZLE_CON_RAW_PACKET))
 
387
    {
 
388
      drizzle_state_push(con, drizzle_state_handshake_server_read);
 
389
      drizzle_state_push(con, drizzle_state_packet_read);
 
390
    }
 
391
 
 
392
    drizzle_state_push(con, drizzle_state_connect);
 
393
    drizzle_state_push(con, drizzle_state_addrinfo);
 
394
  }
 
395
 
 
396
  return drizzle_state_loop(con);
 
397
}
 
398
 
 
399
drizzle_result_st *drizzle_con_quit(drizzle_con_st *con,
 
400
                                    drizzle_result_st *result,
 
401
                                    drizzle_return_t *ret_ptr)
 
402
{
 
403
  return drizzle_con_command_write(con, result, DRIZZLE_COMMAND_QUIT, NULL, 0,
 
404
                                   0, ret_ptr);
 
405
}
 
406
 
 
407
drizzle_result_st *drizzle_quit(drizzle_con_st *con,
 
408
                                drizzle_result_st *result,
 
409
                                drizzle_return_t *ret_ptr)
 
410
{
 
411
  return drizzle_con_quit(con, result, ret_ptr);
 
412
}
 
413
 
 
414
drizzle_result_st *drizzle_con_select_db(drizzle_con_st *con,
 
415
                                         drizzle_result_st *result,
 
416
                                         const char *db,
 
417
                                         drizzle_return_t *ret_ptr)
 
418
{
 
419
  drizzle_con_set_db(con, db);
 
420
  return drizzle_con_command_write(con, result, DRIZZLE_COMMAND_INIT_DB,
 
421
                                   db, strlen(db), strlen(db), ret_ptr);
 
422
}
 
423
 
 
424
drizzle_result_st *drizzle_select_db(drizzle_con_st *con,
 
425
                                     drizzle_result_st *result,
 
426
                                     const char *db,
 
427
                                     drizzle_return_t *ret_ptr)
 
428
{
 
429
  return drizzle_con_select_db(con, result, db, ret_ptr);
 
430
}
 
431
 
 
432
drizzle_result_st *drizzle_con_shutdown(drizzle_con_st *con,
 
433
                                        drizzle_result_st *result,
 
434
                                        drizzle_return_t *ret_ptr)
 
435
{
 
436
  if (con->options & DRIZZLE_CON_MYSQL)
 
437
  {
 
438
    return drizzle_con_command_write(con, result, DRIZZLE_COMMAND_SHUTDOWN,
 
439
                                     "0", 1, 1, ret_ptr);
 
440
  }
 
441
 
 
442
  return drizzle_con_command_write(con, result, DRIZZLE_COMMAND_SHUTDOWN, NULL,
 
443
                                   0, 0, ret_ptr);
 
444
}
 
445
 
 
446
drizzle_result_st *drizzle_shutdown(drizzle_con_st *con,
 
447
                                    drizzle_result_st *result, uint32_t level,
 
448
                                    drizzle_return_t *ret_ptr)
 
449
{
 
450
  (void) level;
 
451
  return drizzle_con_shutdown(con, result, ret_ptr);
 
452
}
 
453
 
 
454
drizzle_result_st *drizzle_con_ping(drizzle_con_st *con,
 
455
                                    drizzle_result_st *result,
 
456
                                    drizzle_return_t *ret_ptr)
 
457
{
 
458
  return drizzle_con_command_write(con, result, DRIZZLE_COMMAND_PING, NULL, 0,
 
459
                                   0, ret_ptr);
 
460
}
 
461
 
 
462
drizzle_result_st *drizzle_ping(drizzle_con_st *con,
 
463
                                drizzle_result_st *result,
 
464
                                drizzle_return_t *ret_ptr)
 
465
{
 
466
  return drizzle_con_ping(con, result, ret_ptr);
 
467
}
 
468
 
 
469
drizzle_result_st *drizzle_con_command_write(drizzle_con_st *con,
 
470
                                             drizzle_result_st *result,
 
471
                                             drizzle_command_t command,
 
472
                                             const void *data, size_t size,
 
473
                                             size_t total,
 
474
                                             drizzle_return_t *ret_ptr)
 
475
{
 
476
  if (!(con->options & DRIZZLE_CON_READY))
 
477
  {
 
478
    if (con->options & DRIZZLE_CON_RAW_PACKET)
 
479
    {
 
480
      drizzle_set_error(con->drizzle, "drizzle_command_write",
 
481
                        "connection not ready");
 
482
      *ret_ptr= DRIZZLE_RETURN_NOT_READY;
 
483
      return result;
 
484
    }
 
485
 
 
486
    *ret_ptr= drizzle_con_connect(con);
 
487
    if (*ret_ptr != DRIZZLE_RETURN_OK)
 
488
      return result;
 
489
  }
 
490
 
 
491
  if (drizzle_state_none(con))
 
492
  {
 
493
    if (con->options & (DRIZZLE_CON_RAW_PACKET | DRIZZLE_CON_NO_RESULT_READ))
 
494
      con->result= NULL;
 
495
    else
 
496
    {
 
497
      con->result= drizzle_result_create(con, result);
 
498
      if (con->result == NULL)
 
499
      {
 
500
        *ret_ptr= DRIZZLE_RETURN_MEMORY;
 
501
        return NULL;
 
502
      }
 
503
    }
 
504
 
 
505
    con->command= command;
 
506
    con->command_data= (uint8_t *)data;
 
507
    con->command_size= size;
 
508
    con->command_offset= 0;
 
509
    con->command_total= total;
 
510
 
 
511
    drizzle_state_push(con, drizzle_state_command_write);
 
512
  }
 
513
  else if (con->command_data == NULL)
 
514
  {
 
515
    con->command_data= (uint8_t *)data;
 
516
    con->command_size= size;
 
517
  }
 
518
 
 
519
  *ret_ptr= drizzle_state_loop(con);
 
520
  if (*ret_ptr == DRIZZLE_RETURN_PAUSE)
 
521
    *ret_ptr= DRIZZLE_RETURN_OK;
 
522
  else if (*ret_ptr != DRIZZLE_RETURN_OK &&
 
523
           *ret_ptr != DRIZZLE_RETURN_IO_WAIT &&
 
524
           *ret_ptr != DRIZZLE_RETURN_ERROR_CODE)
 
525
  {
 
526
    drizzle_result_free(con->result);
 
527
    con->result= result;
 
528
  }
 
529
 
 
530
  return con->result;
 
531
}
 
532
 
 
533
/*
 
534
 * Server Definitions
 
535
 */
 
536
 
 
537
drizzle_return_t drizzle_con_listen(drizzle_con_st *con)
 
538
{
 
539
  if (con->options & DRIZZLE_CON_READY)
 
540
    return DRIZZLE_RETURN_OK;
 
541
 
 
542
  if (drizzle_state_none(con))
 
543
  {
 
544
    drizzle_state_push(con, drizzle_state_listen);
 
545
    drizzle_state_push(con, drizzle_state_addrinfo);
 
546
  }
 
547
 
 
548
  return drizzle_state_loop(con);
 
549
}
 
550
 
 
551
int drizzle_con_backlog(const drizzle_con_st *con)
 
552
{
 
553
  return con->backlog;
 
554
}
 
555
 
 
556
void drizzle_con_set_backlog(drizzle_con_st *con, int backlog)
 
557
{
 
558
  con->backlog= backlog;
 
559
}
 
560
 
 
561
void drizzle_con_set_protocol_version(drizzle_con_st *con,
 
562
                                      uint8_t protocol_version)
 
563
{
 
564
  con->protocol_version= protocol_version;
 
565
}
 
566
 
 
567
void drizzle_con_set_server_version(drizzle_con_st *con,
 
568
                                    const char *server_version)
 
569
{
 
570
  if (server_version == NULL)
 
571
    con->server_version[0]= 0;
 
572
  else
 
573
  {
 
574
    strncpy(con->server_version, server_version,
 
575
            DRIZZLE_MAX_SERVER_VERSION_SIZE);
 
576
    con->server_version[DRIZZLE_MAX_SERVER_VERSION_SIZE - 1]= 0;
 
577
  }
 
578
}
 
579
 
 
580
void drizzle_con_set_thread_id(drizzle_con_st *con, uint32_t thread_id)
 
581
{
 
582
  con->thread_id= thread_id;
 
583
}
 
584
 
 
585
void drizzle_con_set_scramble(drizzle_con_st *con, const uint8_t *scramble)
 
586
{
 
587
  if (scramble == NULL)
 
588
    con->scramble= NULL;
 
589
  else
 
590
  {
 
591
    con->scramble= con->scramble_buffer;
 
592
    memcpy(con->scramble, scramble, DRIZZLE_MAX_SCRAMBLE_SIZE);
 
593
  }
 
594
}
 
595
 
 
596
void drizzle_con_set_capabilities(drizzle_con_st *con,
 
597
                                  drizzle_capabilities_t capabilities)
 
598
{
 
599
  con->capabilities= capabilities;
 
600
}
 
601
 
 
602
void drizzle_con_set_charset(drizzle_con_st *con, drizzle_charset_t charset)
 
603
{
 
604
  con->charset= charset;
 
605
}
 
606
 
 
607
void drizzle_con_set_status(drizzle_con_st *con, drizzle_con_status_t status)
 
608
{
 
609
  con->status= status;
 
610
}
 
611
 
 
612
void drizzle_con_set_max_packet_size(drizzle_con_st *con,
 
613
                                     uint32_t max_packet_size)
 
614
{
 
615
  con->max_packet_size= max_packet_size;
 
616
}
 
617
 
 
618
void drizzle_con_copy_handshake(drizzle_con_st *con, drizzle_con_st *from)
 
619
{
 
620
  drizzle_con_set_auth(con, from->user, NULL);
 
621
  drizzle_con_set_scramble(con, from->scramble);
 
622
  drizzle_con_set_db(con, from->db);
 
623
  drizzle_con_set_protocol_version(con, from->protocol_version);
 
624
  drizzle_con_set_server_version(con, from->server_version);
 
625
  drizzle_con_set_thread_id(con, from->thread_id);
 
626
  drizzle_con_set_scramble(con, from->scramble);
 
627
  drizzle_con_set_capabilities(con, from->capabilities);
 
628
  drizzle_con_set_charset(con, from->charset);
 
629
  drizzle_con_set_status(con, from->status);
 
630
  drizzle_con_set_max_packet_size(con, from->max_packet_size);
 
631
}
 
632
 
 
633
void *drizzle_con_command_read(drizzle_con_st *con,
 
634
                               drizzle_command_t *command, size_t *offset,
 
635
                               size_t *size, size_t *total,
 
636
                               drizzle_return_t *ret_ptr)
 
637
{
 
638
  if (drizzle_state_none(con))
 
639
  {
 
640
    con->packet_number= 0;
 
641
    con->command_offset= 0;
 
642
    con->command_total= 0;
 
643
 
 
644
    drizzle_state_push(con, drizzle_state_command_read);
 
645
    drizzle_state_push(con, drizzle_state_packet_read);
 
646
  }
 
647
 
 
648
  *offset= con->command_offset;
 
649
 
 
650
  *ret_ptr= drizzle_state_loop(con);
 
651
  if (*ret_ptr == DRIZZLE_RETURN_PAUSE)
 
652
    *ret_ptr= DRIZZLE_RETURN_OK;
 
653
 
 
654
  *command= con->command;
 
655
  *size= con->command_size;
 
656
  *total= con->command_total;
 
657
 
 
658
  return con->command_data;
 
659
}
 
660
 
 
661
void *drizzle_con_command_buffer(drizzle_con_st *con,
 
662
                                 drizzle_command_t *command, size_t *total,
 
663
                                 drizzle_return_t *ret_ptr)
 
664
{
 
665
  uint8_t *command_data;
 
666
  size_t offset= 0;
 
667
  size_t size= 0;
 
668
 
 
669
  command_data= drizzle_con_command_read(con, command, &offset, &size, total,
 
670
                                         ret_ptr);
 
671
  if (*ret_ptr != DRIZZLE_RETURN_OK)
 
672
    return NULL;
 
673
 
 
674
  if (command_data == NULL)
 
675
  {
 
676
    *total= 0;
 
677
    return NULL;
 
678
  }
 
679
 
 
680
  if (con->command_buffer == NULL)
 
681
  {
 
682
    con->command_buffer= malloc((*total) + 1);
 
683
    if (con->command_buffer == NULL)
 
684
    {
 
685
      drizzle_set_error(con->drizzle, "drizzle_command_buffer", "malloc");
 
686
      *ret_ptr= DRIZZLE_RETURN_MEMORY;
 
687
      return NULL;
 
688
    }
 
689
  }
 
690
 
 
691
  memcpy(con->command_buffer + offset, command_data, size);
 
692
 
 
693
  while ((offset + size) != (*total))
 
694
  {
 
695
    command_data= drizzle_con_command_read(con, command, &offset, &size, total,
 
696
                                           ret_ptr);
 
697
    if (*ret_ptr != DRIZZLE_RETURN_OK)
 
698
      return NULL;
 
699
 
 
700
    memcpy(con->command_buffer + offset, command_data, size);
 
701
  }
 
702
 
 
703
  command_data= con->command_buffer;
 
704
  con->command_buffer= NULL;
 
705
  command_data[*total]= 0;
 
706
 
 
707
  return command_data;
 
708
}
 
709
 
 
710
/*
 
711
 * Local Definitions
 
712
 */
 
713
 
 
714
void drizzle_con_reset_addrinfo(drizzle_con_st *con)
 
715
{
 
716
  switch (con->socket_type)
 
717
  {
 
718
  case DRIZZLE_CON_SOCKET_TCP:
 
719
    if (con->socket.tcp.addrinfo != NULL)
 
720
    {
 
721
      freeaddrinfo(con->socket.tcp.addrinfo);
 
722
      con->socket.tcp.addrinfo= NULL;
 
723
    }
 
724
    break;
 
725
 
 
726
  case DRIZZLE_CON_SOCKET_UDS:
 
727
    con->socket.uds.addrinfo.ai_addr= NULL;
 
728
    break;
 
729
 
 
730
  default:
 
731
    break;
 
732
  }
 
733
 
 
734
  con->addrinfo_next= NULL;
 
735
}
 
736
 
 
737
/*
 
738
 * State Definitions
 
739
 */
 
740
 
 
741
drizzle_return_t drizzle_state_addrinfo(drizzle_con_st *con)
 
742
{
 
743
  drizzle_con_tcp_st *tcp;
 
744
  const char *host;
 
745
  char port[NI_MAXSERV];
 
746
  struct addrinfo ai;
 
747
  int ret;
 
748
 
 
749
  drizzle_log_debug(con->drizzle, "drizzle_state_addrinfo");
 
750
 
 
751
  switch (con->socket_type)
 
752
  {
 
753
  case DRIZZLE_CON_SOCKET_TCP:
 
754
    tcp= &(con->socket.tcp);
 
755
 
 
756
    if (tcp->addrinfo != NULL)
 
757
    {
 
758
      freeaddrinfo(tcp->addrinfo);
 
759
      tcp->addrinfo= NULL;
 
760
    }
 
761
 
 
762
    if (tcp->port != 0)
 
763
      snprintf(port, NI_MAXSERV, "%u", tcp->port);
 
764
    else if (con->options & DRIZZLE_CON_MYSQL)
 
765
      snprintf(port, NI_MAXSERV, "%u", DRIZZLE_DEFAULT_TCP_PORT_MYSQL);
 
766
    else
 
767
      snprintf(port, NI_MAXSERV, "%u", DRIZZLE_DEFAULT_TCP_PORT);
 
768
 
 
769
    memset(&ai, 0, sizeof(struct addrinfo));
 
770
    ai.ai_socktype= SOCK_STREAM;
 
771
    ai.ai_protocol= IPPROTO_TCP;
 
772
 
 
773
    if (con->options & DRIZZLE_CON_LISTEN)
 
774
    {
 
775
      ai.ai_flags = AI_PASSIVE;
 
776
      ai.ai_family = AF_UNSPEC;
 
777
      host= tcp->host;
 
778
    }
 
779
    else
 
780
    {
 
781
      if (tcp->host == NULL)
 
782
        host= DRIZZLE_DEFAULT_TCP_HOST;
 
783
      else
 
784
        host= tcp->host;
 
785
    }
 
786
 
 
787
    ret= getaddrinfo(host, port, &ai, &(tcp->addrinfo));
 
788
    if (ret != 0)
 
789
    {
 
790
      drizzle_set_error(con->drizzle, "drizzle_state_addrinfo",
 
791
                        "getaddrinfo:%s", gai_strerror(ret));
 
792
      return DRIZZLE_RETURN_GETADDRINFO;
 
793
    }
 
794
 
 
795
    con->addrinfo_next= tcp->addrinfo;
 
796
 
 
797
    break;
 
798
 
 
799
  case DRIZZLE_CON_SOCKET_UDS:
 
800
    con->addrinfo_next= &(con->socket.uds.addrinfo);
 
801
    break;
 
802
 
 
803
  default:
 
804
    break;
 
805
  }
 
806
 
 
807
  drizzle_state_pop(con);
 
808
  return DRIZZLE_RETURN_OK;
 
809
}
 
810
 
 
811
drizzle_return_t drizzle_state_connect(drizzle_con_st *con)
 
812
{
 
813
  int ret;
 
814
  drizzle_return_t dret;
 
815
 
 
816
  drizzle_log_debug(con->drizzle, "drizzle_state_connect");
 
817
 
 
818
  if (con->fd != -1)
 
819
  {
 
820
    (void)close(con->fd);
 
821
    con->fd= -1;
 
822
  }
 
823
 
 
824
  if (con->addrinfo_next == NULL)
 
825
  {
 
826
    drizzle_set_error(con->drizzle, "drizzle_state_connect",
 
827
                      "could not connect");
 
828
    drizzle_state_reset(con);
 
829
    return DRIZZLE_RETURN_COULD_NOT_CONNECT;
 
830
  }
 
831
 
 
832
  con->fd= socket(con->addrinfo_next->ai_family,
 
833
                  con->addrinfo_next->ai_socktype,
 
834
                  con->addrinfo_next->ai_protocol);
 
835
  if (con->fd == -1)
 
836
  {
 
837
    drizzle_set_error(con->drizzle, "drizzle_state_connect", "socket:%d",
 
838
                      errno);
 
839
    con->drizzle->last_errno= errno;
 
840
    return DRIZZLE_RETURN_ERRNO;
 
841
  }
 
842
 
 
843
  dret= _con_setsockopt(con);
 
844
  if (dret != DRIZZLE_RETURN_OK)
 
845
  {
 
846
    con->drizzle->last_errno= errno;
 
847
    return dret;
 
848
  }
 
849
 
 
850
  while (1)
 
851
  {
 
852
    ret= connect(con->fd, con->addrinfo_next->ai_addr,
 
853
                 con->addrinfo_next->ai_addrlen);
 
854
 
 
855
#ifdef _WIN32
 
856
    /*Mapping windows specific error codes to Posix*/
 
857
    errno = WSAGetLastError();
 
858
    switch(errno) {
 
859
    case WSAEINVAL:
 
860
    case WSAEALREADY:
 
861
    case WSAEWOULDBLOCK:
 
862
      errno = EINPROGRESS;
 
863
      break;
 
864
    default:
 
865
      break;
 
866
    }
 
867
#endif /* _WIN32 */
 
868
        
 
869
    drizzle_log_crazy(con->drizzle, "connect return=%d errno=%d", ret, errno);
 
870
 
 
871
    if (ret == 0)
 
872
    {
 
873
      con->addrinfo_next= NULL;
 
874
      break;
 
875
    }
 
876
 
 
877
    if (errno == EAGAIN || errno == EINTR)
 
878
      continue;
 
879
 
 
880
    if (errno == EINPROGRESS)
 
881
    {
 
882
      drizzle_state_pop(con);
 
883
      drizzle_state_push(con, drizzle_state_connecting);
 
884
      return DRIZZLE_RETURN_OK;
 
885
    }
 
886
 
 
887
    if (errno == ECONNREFUSED || errno == ENETUNREACH || errno == ETIMEDOUT)
 
888
    {
 
889
      con->addrinfo_next= con->addrinfo_next->ai_next;
 
890
      return DRIZZLE_RETURN_OK;
 
891
    }
 
892
 
 
893
    drizzle_set_error(con->drizzle, "drizzle_state_connect", "connect:%d",
 
894
                      errno);
 
895
    con->drizzle->last_errno= errno;
 
896
    return DRIZZLE_RETURN_ERRNO;
 
897
  }
 
898
 
 
899
  drizzle_state_pop(con);
 
900
  return DRIZZLE_RETURN_OK;
 
901
}
 
902
 
 
903
drizzle_return_t drizzle_state_connecting(drizzle_con_st *con)
 
904
{
 
905
  drizzle_return_t ret;
 
906
 
 
907
  drizzle_log_debug(con->drizzle, "drizzle_state_connecting");
 
908
 
 
909
  while (1)
 
910
  {
 
911
    if (con->revents & POLLOUT)
 
912
    {
 
913
      drizzle_state_pop(con);
 
914
      return DRIZZLE_RETURN_OK;
 
915
    }
 
916
    else if (con->revents & (POLLERR | POLLHUP | POLLNVAL))
 
917
    {
 
918
      con->revents= 0;
 
919
      drizzle_state_pop(con);
 
920
      drizzle_state_push(con, drizzle_state_connect);
 
921
      con->addrinfo_next= con->addrinfo_next->ai_next;
 
922
      return DRIZZLE_RETURN_OK;
 
923
    }
 
924
 
 
925
    ret= drizzle_con_set_events(con, POLLOUT);
 
926
    if (ret != DRIZZLE_RETURN_OK)
 
927
      return ret;
 
928
 
 
929
    if (con->drizzle->options & DRIZZLE_NON_BLOCKING)
 
930
      return DRIZZLE_RETURN_IO_WAIT;
 
931
 
 
932
    ret= drizzle_con_wait(con->drizzle);
 
933
    if (ret != DRIZZLE_RETURN_OK)
 
934
      return ret;
 
935
  }
 
936
}
 
937
 
 
938
drizzle_return_t drizzle_state_read(drizzle_con_st *con)
 
939
{
 
940
  drizzle_return_t ret;
 
941
  ssize_t read_size;
 
942
 
 
943
  drizzle_log_debug(con->drizzle, "drizzle_state_read");
 
944
 
 
945
  if (con->buffer_size == 0)
 
946
    con->buffer_ptr= con->buffer;
 
947
  else if ((con->buffer_ptr - con->buffer) > (DRIZZLE_MAX_BUFFER_SIZE / 2))
 
948
  {
 
949
    memmove(con->buffer, con->buffer_ptr, con->buffer_size);
 
950
    con->buffer_ptr= con->buffer;
 
951
  }
 
952
 
 
953
  while (1)
 
954
  {
 
955
    read_size = recv(con->fd, (char *)con->buffer_ptr + con->buffer_size,
 
956
                     (size_t)DRIZZLE_MAX_BUFFER_SIZE -
 
957
                     ((size_t)(con->buffer_ptr - con->buffer) +
 
958
                      con->buffer_size),0); 
 
959
#ifdef _WIN32
 
960
    /*Get windows error codes and map it to Posix*/
 
961
    errno = WSAGetLastError();
 
962
    switch(errno) {
 
963
    case WSAEWOULDBLOCK:
 
964
    case 10057:
 
965
      errno = EAGAIN;
 
966
      break;
 
967
    default:
 
968
      break;
 
969
    }
 
970
#endif /* _WIN32 */     
 
971
    drizzle_log_crazy(con->drizzle, "read fd=%d return=%zd errno=%d", con->fd,
 
972
                      read_size, errno);
 
973
 
 
974
    if (read_size == 0)
 
975
    {
 
976
      drizzle_set_error(con->drizzle, "drizzle_state_read",
 
977
                        "lost connection to server (EOF)");
 
978
      return DRIZZLE_RETURN_LOST_CONNECTION;
 
979
    }
 
980
    else if (read_size == -1)
 
981
    {
 
982
      if (errno == EAGAIN)
 
983
      {
 
984
        ret= drizzle_con_set_events(con, POLLIN);
 
985
        if (ret != DRIZZLE_RETURN_OK)
 
986
          return 0;
 
987
 
 
988
        if (con->drizzle->options & DRIZZLE_NON_BLOCKING)
 
989
          return DRIZZLE_RETURN_IO_WAIT;
 
990
 
 
991
        ret= drizzle_con_wait(con->drizzle);
 
992
        if (ret != DRIZZLE_RETURN_OK)
 
993
          return ret;
 
994
 
 
995
        continue;
 
996
      }
 
997
      else if (errno == ECONNREFUSED)
 
998
      {
 
999
        con->revents= 0;
 
1000
        drizzle_state_pop(con);
 
1001
        drizzle_state_push(con, drizzle_state_connect);
 
1002
        con->addrinfo_next= con->addrinfo_next->ai_next;
 
1003
        return DRIZZLE_RETURN_OK;
 
1004
      }
 
1005
      else if (errno == EINTR)
 
1006
        continue;
 
1007
      else if (errno == EPIPE || errno == ECONNRESET)
 
1008
      {
 
1009
        drizzle_set_error(con->drizzle, "drizzle_state_read",
 
1010
                          "lost connection to server (%d)", errno);
 
1011
        return DRIZZLE_RETURN_LOST_CONNECTION;
 
1012
      }
 
1013
 
 
1014
      drizzle_set_error(con->drizzle, "drizzle_state_read", "read:%d", errno);
 
1015
      con->drizzle->last_errno= errno;
 
1016
      return DRIZZLE_RETURN_ERRNO;
 
1017
    }
 
1018
 
 
1019
    con->buffer_size+= (size_t)read_size;
 
1020
    break;
 
1021
  }
 
1022
 
 
1023
  drizzle_state_pop(con);;
 
1024
  return DRIZZLE_RETURN_OK;
 
1025
}
 
1026
 
 
1027
drizzle_return_t drizzle_state_write(drizzle_con_st *con)
 
1028
{
 
1029
  drizzle_return_t ret;
 
1030
  ssize_t write_size;
 
1031
 
 
1032
  drizzle_log_debug(con->drizzle, "drizzle_state_write");
 
1033
 
 
1034
  while (con->buffer_size != 0)
 
1035
  {
 
1036
  
 
1037
    write_size = send(con->fd,(char *) con->buffer_ptr, con->buffer_size, 0);
 
1038
 
 
1039
    drizzle_log_crazy(con->drizzle, "write fd=%d return=%zd errno=%d", con->fd,
 
1040
                      write_size, errno);
 
1041
 
 
1042
    if (write_size == 0)
 
1043
    {
 
1044
      drizzle_set_error(con->drizzle, "drizzle_state_write",
 
1045
                        "lost connection to server (EOF)");
 
1046
      return DRIZZLE_RETURN_LOST_CONNECTION;
 
1047
    }
 
1048
    else if (write_size == -1)
 
1049
    {
 
1050
      if (errno == EAGAIN)
 
1051
      {
 
1052
        ret= drizzle_con_set_events(con, POLLOUT);
 
1053
        if (ret != DRIZZLE_RETURN_OK)
 
1054
          return ret;
 
1055
 
 
1056
        if (con->drizzle->options & DRIZZLE_NON_BLOCKING)
 
1057
          return DRIZZLE_RETURN_IO_WAIT;
 
1058
 
 
1059
        ret= drizzle_con_wait(con->drizzle);
 
1060
        if (ret != DRIZZLE_RETURN_OK)
 
1061
          return ret;
 
1062
 
 
1063
        continue;
 
1064
      }
 
1065
      else if (errno == EINTR)
 
1066
        continue;
 
1067
      else if (errno == EPIPE || errno == ECONNRESET)
 
1068
      {
 
1069
        drizzle_set_error(con->drizzle, "drizzle_state_write",
 
1070
                          "lost connection to server (%d)", errno);
 
1071
        return DRIZZLE_RETURN_LOST_CONNECTION;
 
1072
      }
 
1073
 
 
1074
      drizzle_set_error(con->drizzle, "drizzle_state_write", "write:%d", errno);
 
1075
      con->drizzle->last_errno= errno;
 
1076
      return DRIZZLE_RETURN_ERRNO;
 
1077
    }
 
1078
 
 
1079
    con->buffer_ptr+= write_size;
 
1080
    con->buffer_size-= (size_t)write_size;
 
1081
    if (con->buffer_size == 0)
 
1082
      break;
 
1083
  }
 
1084
 
 
1085
  con->buffer_ptr= con->buffer;
 
1086
 
 
1087
  drizzle_state_pop(con);
 
1088
  return DRIZZLE_RETURN_OK;
 
1089
}
 
1090
 
 
1091
drizzle_return_t drizzle_state_listen(drizzle_con_st *con)
 
1092
{
 
1093
  char host[NI_MAXHOST];
 
1094
  char port[NI_MAXSERV];
 
1095
  int ret;
 
1096
  int fd;
 
1097
  int opt;
 
1098
  drizzle_con_st *new_con;
 
1099
 
 
1100
  for (; con->addrinfo_next != NULL;
 
1101
       con->addrinfo_next= con->addrinfo_next->ai_next)
 
1102
  {
 
1103
    ret= getnameinfo(con->addrinfo_next->ai_addr,
 
1104
                     con->addrinfo_next->ai_addrlen, host, NI_MAXHOST, port,
 
1105
                     NI_MAXSERV, NI_NUMERICHOST | NI_NUMERICSERV);
 
1106
    if (ret != 0)
 
1107
    {
 
1108
      drizzle_set_error(con->drizzle, "drizzle_state_listen", "getnameinfo:%s",
 
1109
                        gai_strerror(ret));
 
1110
      return DRIZZLE_RETURN_GETADDRINFO;
 
1111
    }
 
1112
 
 
1113
    /* Call to socket() can fail for some getaddrinfo results, try another. */
 
1114
    fd= socket(con->addrinfo_next->ai_family, con->addrinfo_next->ai_socktype,
 
1115
               con->addrinfo_next->ai_protocol);
 
1116
    if (fd == -1)
 
1117
    {
 
1118
      drizzle_log_info(con->drizzle, "could not listen on %s:%s", host, port);
 
1119
      drizzle_set_error(con->drizzle, "drizzle_state_listen", "socket:%d",
 
1120
                        errno);
 
1121
      continue;
 
1122
    }
 
1123
        
 
1124
        opt= 1;
 
1125
#ifdef _WIN32
 
1126
        ret= setsockopt(fd, SOL_SOCKET, SO_REUSEADDR,(const char*) &opt, sizeof(opt));
 
1127
#else
 
1128
        ret= setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
 
1129
#endif /* _WIN32 */
 
1130
    if (ret == -1)
 
1131
    {
 
1132
      close(fd);
 
1133
      drizzle_set_error(con->drizzle, "drizzle_state_listen", "setsockopt:%d",
 
1134
                        errno);
 
1135
      return DRIZZLE_RETURN_ERRNO;
 
1136
    }
 
1137
 
 
1138
    ret= bind(fd, con->addrinfo_next->ai_addr, con->addrinfo_next->ai_addrlen);
 
1139
    if (ret == -1)
 
1140
    {
 
1141
      close(fd);
 
1142
      drizzle_set_error(con->drizzle, "drizzle_state_listen", "bind:%d", errno);
 
1143
      if (errno == EADDRINUSE)
 
1144
      {
 
1145
        if (con->fd == -1)
 
1146
        {
 
1147
          drizzle_log_info(con->drizzle, "could not listen on %s:%s", host,
 
1148
                           port);
 
1149
        }
 
1150
 
 
1151
        continue;
 
1152
      }
 
1153
 
 
1154
      return DRIZZLE_RETURN_ERRNO;
 
1155
    }
 
1156
 
 
1157
    if (listen(fd, con->backlog) == -1)
 
1158
    {
 
1159
      close(fd);
 
1160
      drizzle_set_error(con->drizzle, "drizzle_state_listen", "listen:%d",
 
1161
                        errno);
 
1162
      return DRIZZLE_RETURN_ERRNO;
 
1163
    }
 
1164
 
 
1165
    if (con->fd == -1)
 
1166
    {
 
1167
      con->fd= fd;
 
1168
      new_con= con;
 
1169
    }
 
1170
    else
 
1171
    {
 
1172
      new_con= drizzle_con_clone(con->drizzle, NULL, con);
 
1173
      if (new_con == NULL)
 
1174
      {
 
1175
        close(fd);
 
1176
        return DRIZZLE_RETURN_MEMORY;
 
1177
      }
 
1178
 
 
1179
      new_con->fd= fd;
 
1180
    }
 
1181
 
 
1182
    /* Wait for read events on the listening socket. */
 
1183
    ret= drizzle_con_set_events(new_con, POLLIN);
 
1184
    if (ret != DRIZZLE_RETURN_OK)
 
1185
    {
 
1186
      drizzle_con_free(new_con);
 
1187
      return ret;
 
1188
    }
 
1189
 
 
1190
    drizzle_log_info(con->drizzle, "listening on %s:%s", host, port);
 
1191
  }
 
1192
 
 
1193
  /* Report last socket() error if we couldn't find an address to bind. */
 
1194
  if (con->fd == -1)
 
1195
    return DRIZZLE_RETURN_ERRNO;
 
1196
 
 
1197
  drizzle_state_pop(con);
 
1198
  return DRIZZLE_RETURN_OK;
 
1199
}
 
1200
 
 
1201
/*
 
1202
 * Static Definitions
 
1203
 */
 
1204
 
 
1205
static drizzle_return_t _con_setsockopt(drizzle_con_st *con)
 
1206
{
 
1207
  int ret;
 
1208
  struct linger linger;
 
1209
  struct timeval waittime;
 
1210
 
 
1211
  ret= 1;
 
1212
 
 
1213
#ifdef _WIN32
 
1214
  ret= setsockopt(con->fd, IPPROTO_TCP, TCP_NODELAY, (const char*)&ret,
 
1215
                  (socklen_t)sizeof(int));
 
1216
#else
 
1217
  ret= setsockopt(con->fd, IPPROTO_TCP, TCP_NODELAY, &ret,
 
1218
                  (socklen_t)sizeof(int));
 
1219
#endif /* _WIN32 */
 
1220
 
 
1221
  if (ret == -1 && errno != EOPNOTSUPP)
 
1222
  {
 
1223
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1224
                      "setsockopt:TCP_NODELAY:%d", errno);
 
1225
    return DRIZZLE_RETURN_ERRNO;
 
1226
  }
 
1227
 
 
1228
  linger.l_onoff= 1;
 
1229
  linger.l_linger= DRIZZLE_DEFAULT_SOCKET_TIMEOUT;
 
1230
 
 
1231
#ifdef _WIN32
 
1232
  ret= setsockopt(con->fd, SOL_SOCKET, SO_LINGER, (const char*)&linger,
 
1233
                  (socklen_t)sizeof(struct linger));
 
1234
#else
 
1235
  ret= setsockopt(con->fd, SOL_SOCKET, SO_LINGER, &linger,
 
1236
                  (socklen_t)sizeof(struct linger));
 
1237
#endif /* _WIN32 */
 
1238
 
 
1239
  if (ret == -1)
 
1240
  {
 
1241
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1242
                      "setsockopt:SO_LINGER:%d", errno);
 
1243
    return DRIZZLE_RETURN_ERRNO;
 
1244
  }
 
1245
 
 
1246
  waittime.tv_sec= DRIZZLE_DEFAULT_SOCKET_TIMEOUT;
 
1247
  waittime.tv_usec= 0;
 
1248
 
 
1249
#ifdef _WIN32
 
1250
  ret= setsockopt(con->fd, SOL_SOCKET, SO_SNDTIMEO, (const char*)&waittime,
 
1251
                  (socklen_t)sizeof(struct timeval));
 
1252
#else
 
1253
  ret= setsockopt(con->fd, SOL_SOCKET, SO_SNDTIMEO, &waittime,
 
1254
                  (socklen_t)sizeof(struct timeval));
 
1255
#endif /* _WIN32 */
 
1256
 
 
1257
  if (ret == -1 && errno != ENOPROTOOPT)
 
1258
  {
 
1259
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1260
                      "setsockopt:SO_SNDTIMEO:%d", errno);
 
1261
    return DRIZZLE_RETURN_ERRNO;
 
1262
  }
 
1263
 
 
1264
#ifdef _WIN32
 
1265
  ret= setsockopt(con->fd, SOL_SOCKET, SO_RCVTIMEO, (const char*)&waittime,
 
1266
                  (socklen_t)sizeof(struct timeval));
 
1267
#else
 
1268
  ret= setsockopt(con->fd, SOL_SOCKET, SO_RCVTIMEO, &waittime,
 
1269
                  (socklen_t)sizeof(struct timeval));
 
1270
#endif /* _WIN32 */
 
1271
 
 
1272
  if (ret == -1 && errno != ENOPROTOOPT)
 
1273
  {
 
1274
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1275
                      "setsockopt:SO_RCVTIMEO:%d", errno);
 
1276
    return DRIZZLE_RETURN_ERRNO;
 
1277
  }
 
1278
 
 
1279
  ret= DRIZZLE_DEFAULT_SOCKET_SEND_SIZE;
 
1280
#ifdef _WIN32
 
1281
  ret= setsockopt(con->fd, SOL_SOCKET, SO_SNDBUF, (const char*)&ret, (socklen_t)sizeof(int));
 
1282
#else
 
1283
  ret= setsockopt(con->fd, SOL_SOCKET, SO_SNDBUF, &ret, (socklen_t)sizeof(int));
 
1284
#endif /* _WIN32 */
 
1285
  if (ret == -1)
 
1286
  {
 
1287
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1288
                      "setsockopt:SO_SNDBUF:%d", errno);
 
1289
    return DRIZZLE_RETURN_ERRNO;
 
1290
  }
 
1291
 
 
1292
  ret= DRIZZLE_DEFAULT_SOCKET_RECV_SIZE;
 
1293
#ifdef _WIN32
 
1294
  ret= setsockopt(con->fd, SOL_SOCKET, SO_RCVBUF, (const char*)&ret, (socklen_t)sizeof(int));
 
1295
#else
 
1296
  ret= setsockopt(con->fd, SOL_SOCKET, SO_RCVBUF, &ret, (socklen_t)sizeof(int));
 
1297
#endif /* _WIN32 */
 
1298
  if (ret == -1)
 
1299
  {
 
1300
    drizzle_set_error(con->drizzle, "_con_setsockopt",
 
1301
                      "setsockopt:SO_RCVBUF:%d", errno);
 
1302
    return DRIZZLE_RETURN_ERRNO;
 
1303
  }
 
1304
 
 
1305
#if defined (_WIN32)
 
1306
  unsigned long asyncmode = 1;
 
1307
  ioctlsocket(con->fd, FIONBIO, &asyncmode);
 
1308
#else
 
1309
  ret= fcntl(con->fd, F_GETFL, 0);
 
1310
  if (ret == -1)
 
1311
  {
 
1312
    drizzle_set_error(con->drizzle, "_con_setsockopt", "fcntl:F_GETFL:%d",
 
1313
                      errno);
 
1314
    return DRIZZLE_RETURN_ERRNO;
 
1315
  }
 
1316
 
 
1317
  ret= fcntl(con->fd, F_SETFL, ret | O_NONBLOCK);
 
1318
  if (ret == -1)
 
1319
  {
 
1320
    drizzle_set_error(con->drizzle, "_con_setsockopt", "fcntl:F_SETFL:%d",
 
1321
                      errno);
 
1322
    return DRIZZLE_RETURN_ERRNO;
 
1323
  }
 
1324
#endif
 
1325
 
 
1326
  return DRIZZLE_RETURN_OK;
 
1327
}