Line data Source code
1 : /* TDSPool - Connection pooling for TDS based databases
2 : * Copyright (C) 2001 Brian Bruns
3 : *
4 : * This program is free software; you can redistribute it and/or modify
5 : * it under the terms of the GNU General Public License as published by
6 : * the Free Software Foundation; either version 2 of the License, or
7 : * (at your option) any later version.
8 : *
9 : * This program is distributed in the hope that it will be useful,
10 : * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 : * GNU General Public License for more details.
13 : *
14 : * You should have received a copy of the GNU General Public License
15 : * along with this program; if not, write to the Free Software
16 : * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 : *
18 : */
19 :
20 : #include <config.h>
21 :
22 : #include <stdarg.h>
23 : #include <stdio.h>
24 : #include <assert.h>
25 :
26 : #if HAVE_STDLIB_H
27 : #include <stdlib.h>
28 : #endif /* HAVE_STDLIB_H */
29 :
30 : #if HAVE_UNISTD_H
31 : #include <unistd.h>
32 : #endif /* HAVE_UNISTD_H */
33 :
34 : #if HAVE_ERRNO_H
35 : #include <errno.h>
36 : #endif /* HAVE_ERRNO_H */
37 :
38 : #include <ctype.h>
39 :
40 : #include "pool.h"
41 : #include <freetds/utils/string.h>
42 : #include <freetds/checks.h>
43 : #include <freetds/bytes.h>
44 :
45 : void
46 714 : dump_login(TDSLOGIN * login)
47 : {
48 1428 : fprintf(stderr, "host %s\n", tds_dstr_cstr(&login->client_host_name));
49 1428 : fprintf(stderr, "user %s\n", tds_dstr_cstr(&login->user_name));
50 1428 : fprintf(stderr, "pass %s\n", tds_dstr_cstr(&login->password));
51 1428 : fprintf(stderr, "app %s\n", tds_dstr_cstr(&login->app_name));
52 1428 : fprintf(stderr, "srvr %s\n", tds_dstr_cstr(&login->server_name));
53 714 : fprintf(stderr, "vers %d.%d\n", TDS_MAJOR(login), TDS_MINOR(login));
54 1428 : fprintf(stderr, "lib %s\n", tds_dstr_cstr(&login->library));
55 1428 : fprintf(stderr, "lang %s\n", tds_dstr_cstr(&login->language));
56 1428 : fprintf(stderr, "char %s\n", tds_dstr_cstr(&login->server_charset));
57 714 : fprintf(stderr, "bsiz %d\n", login->block_size);
58 714 : }
59 :
60 : /**
61 : * Read part of packet. Function does not block.
62 : * @return true if packet is not complete and we must call again,
63 : * false on full packet or error.
64 : */
65 : bool
66 66633 : pool_packet_read(TDSSOCKET *tds)
67 : {
68 : unsigned int packet_len;
69 : int readed;
70 :
71 66633 : tdsdump_log(TDS_DBG_INFO1, "tds in_len %d in_pos %d\n", tds->in_len, tds->in_pos);
72 :
73 : // TODO MARS
74 :
75 : /* determine packet size */
76 66633 : packet_len = tds->in_len >= 4 ? TDS_GET_A2BE(&tds->in_buf[2]) : 8;
77 66633 : if (TDS_UNLIKELY(packet_len < 8)) {
78 0 : tds->in_len = 0;
79 0 : return false;
80 : }
81 :
82 : /* get another packet */
83 66633 : if (tds->in_len >= packet_len) {
84 : /* packet was not fully forwarded */
85 33240 : if (tds->in_pos < tds->in_len)
86 : return false;
87 33240 : tds->in_pos = 0;
88 33240 : tds->in_len = 0;
89 : }
90 :
91 : for (;;) {
92 : /* determine packet size */
93 133114 : packet_len = 8;
94 133114 : if (tds->in_len >= 4) {
95 68031 : packet_len = TDS_GET_A2BE(&tds->in_buf[2]);
96 68031 : if (packet_len < 8)
97 : break;
98 68031 : tdsdump_log(TDS_DBG_INFO1, "packet_len %u in_len %d\n", packet_len, tds->in_len);
99 : /* resize packet if not enough */
100 68031 : if (packet_len > tds->recv_packet->capacity) {
101 : TDSPACKET *packet;
102 :
103 276 : packet = tds_realloc_packet(tds->recv_packet, packet_len);
104 276 : if (!packet)
105 : break;
106 276 : tds->in_buf = packet->buf;
107 276 : tds->recv_packet = packet;
108 : }
109 68031 : CHECK_TDS_EXTRA(tds);
110 68031 : if (tds->in_len >= packet_len)
111 : return false;
112 : }
113 :
114 100586 : assert(packet_len > tds->in_len);
115 100586 : assert(packet_len <= tds->recv_packet->capacity);
116 100586 : assert(tds->in_len < tds->recv_packet->capacity);
117 :
118 100586 : readed = READSOCKET(tds_get_s(tds), &tds->in_buf[tds->in_len], packet_len - tds->in_len);
119 100586 : tdsdump_log(TDS_DBG_INFO1, "readed %d\n", readed);
120 :
121 : /* socket closed */
122 100586 : if (readed == 0)
123 : break;
124 :
125 : /* error */
126 99879 : if (readed < 0) {
127 33398 : int err = sock_errno;
128 33398 : if (err == EINTR)
129 0 : continue;
130 33398 : if (TDSSOCK_WOULDBLOCK(err))
131 : return true;
132 : break;
133 : }
134 :
135 : /* got some data */
136 66481 : tds->in_len += readed;
137 : }
138 :
139 : /* failure */
140 712 : tds->in_len = 0;
141 712 : return false;
142 : }
143 :
144 : int
145 32529 : pool_write(TDS_SYS_SOCKET sock, const void *buf, size_t len)
146 : {
147 : int ret;
148 32529 : const unsigned char *p = (const unsigned char *) buf;
149 :
150 97587 : while (len) {
151 32530 : ret = WRITESOCKET(sock, p, len);
152 32530 : if (ret <= 0) {
153 1 : int err = errno;
154 1 : if (TDSSOCK_WOULDBLOCK(err) || err == EINTR)
155 : break;
156 : return -1;
157 : }
158 32529 : p += ret;
159 32529 : len -= ret;
160 : }
161 32529 : return p - (const unsigned char *) buf;
162 : }
163 :
164 : void
165 1426 : pool_event_add(TDS_POOL *pool, TDS_POOL_EVENT *ev, TDS_POOL_EXECUTE execute)
166 : {
167 1426 : tds_mutex_lock(&pool->events_mtx);
168 1426 : ev->execute = execute;
169 1426 : ev->next = pool->events;
170 1426 : pool->events = ev;
171 1426 : tds_mutex_unlock(&pool->events_mtx);
172 1426 : WRITESOCKET(pool->event_fd, "x", 1);
173 1426 : }
174 :
175 : bool
176 32529 : pool_write_data(TDS_POOL_SOCKET *from, TDS_POOL_SOCKET *to)
177 : {
178 : int ret;
179 : TDSSOCKET *tds;
180 :
181 32529 : tdsdump_log(TDS_DBG_INFO1, "trying to send\n");
182 :
183 32529 : tds = from->tds;
184 32529 : tdsdump_log(TDS_DBG_INFO1, "sending %d bytes\n", tds->in_len);
185 : /* cf. net.c for better technique. */
186 32529 : ret = pool_write(tds_get_s(to->tds), tds->in_buf + tds->in_pos, tds->in_len - tds->in_pos);
187 : /* write failed, cleanup member */
188 32529 : if (ret < 0)
189 : return false;
190 :
191 32529 : tds->in_pos += ret;
192 32529 : if (tds->in_pos < tds->in_len) {
193 : /* partial write, schedule a future write */
194 1 : to->poll_send = true;
195 1 : from->poll_recv = false;
196 : } else {
197 32528 : to->poll_send = false;
198 32528 : from->poll_recv = true;
199 : }
200 : return true;
201 : }
|