3 * Copyright (c) 2004 Scott Ullrich <GeekGod@GeekGod.com>
4 * Portions Copyright (c) 2004 Chris Pressey <cpressey@catseye.mine.nu>
6 * Copyright (c) 2004 The DragonFly Project.
9 * This code is derived from software contributed to The DragonFly Project
10 * by Scott Ullrich and Chris Pressey (see above for e-mail addresses).
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following conditions
16 * 1. Redistributions of source code must retain the above copyright
17 * notice, this list of conditions and the following disclaimer.
19 * 2. Redistributions in binary form must reproduce the above copyright
20 * notice, this list of conditions and the following disclaimer in
21 * the documentation and/or other materials provided with the
24 * 3. Neither the name of The DragonFly Project nor the names of its
25 * contributors may be used to endorse or promote products derived
26 * from this software without specific, prior written permission.
28 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
31 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
32 * COPYRIGHT HOLDERS, CONTRIBUTORS OR VOICES IN THE AUTHOR'S HEAD
33 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY
34 * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
35 * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
36 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
37 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
38 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
39 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
41 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
42 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
43 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
44 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
45 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
46 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
52 * $Id: conn_tcp.c,v 1.16 2005/02/06 19:53:19 cpressey Exp $
58 #include <sys/types.h>
61 #include <sys/errno.h>
62 #include <sys/socket.h>
63 #include <netinet/in.h>
64 #include <arpa/inet.h>
74 #include <libaura/buffer.h>
76 #define NEEDS_DFUI_STRUCTURE_DEFINITIONS
78 #undef NEEDS_DFUI_STRUCTURE_DEFINITIONS
83 /***** BACKEND ******/
88 * Connect to the frontend.
91 dfui_tcp_be_start(struct dfui_connection
*c
)
93 struct sockaddr_in servaddr
;
97 server_port
= atoi(c
->rendezvous
);
100 * Create the tcp socket
103 if ((T_TCP(c
)->listen_sd
= socket(AF_INET
, SOCK_STREAM
, 0)) == -1)
104 return(DFUI_FAILURE
);
105 dfui_debug("LISTEN_SOCKET<<%d>>\n", T_TCP(c
)->listen_sd
);
107 if (setsockopt(T_TCP(c
)->listen_sd
, SOL_SOCKET
, SO_REUSEADDR
,
108 &tru
, sizeof(tru
)) == -1) {
109 return(DFUI_FAILURE
);
112 bzero(&servaddr
, sizeof(servaddr
));
113 servaddr
.sin_family
= AF_INET
;
114 servaddr
.sin_port
= htons(server_port
);
115 switch(inet_pton(AF_INET
, "127.0.0.1", &servaddr
.sin_addr
)) {
117 warnx("inet_pton(): address not parseable");
118 return(DFUI_FAILURE
);
123 return(DFUI_FAILURE
);
126 if (bind(T_TCP(c
)->listen_sd
, (struct sockaddr
*)&servaddr
, sizeof(servaddr
)) == -1) {
128 return(DFUI_FAILURE
);
130 dfui_debug("BOUND_ON<<%d>>\n", T_TCP(c
)->listen_sd
);
131 if (listen(T_TCP(c
)->listen_sd
, 0) == -1)
132 return(DFUI_FAILURE
);
133 dfui_debug("LISTENING_ON<<%d>>\n", T_TCP(c
)->listen_sd
);
134 /* at this point we should be listening on the rendezvous port */
135 return(DFUI_SUCCESS
);
139 * Tell the frontend that we're done and disconnect from it.
142 dfui_tcp_be_stop(struct dfui_connection
*c
)
144 if (dfui_tcp_be_ll_exchange(c
, DFUI_BE_MSG_STOP
, "")) {
145 close(T_TCP(c
)->listen_sd
);
146 close(T_TCP(c
)->connected_sd
);
147 fclose(T_TCP(c
)->stream
);
148 return(DFUI_SUCCESS
);
150 return(DFUI_FAILURE
);
156 * Exchange a message with the frontend. This involves two receive()/reply()
157 * cycles: one to provide our message, one to get a reply from the frontend.
159 * Note that this does not immediately send the message to the frontend -
160 * it can't, because we're a service and it's a client. What it does is
161 * keep the message handy and wait for a frontend request to come in. It
162 * then replies to that request with our message.
164 * The protocol looks something like the following, using the PRESENT and
165 * SUBMIT exchange as an example:
167 * frontend (client) | backend (service)
168 * ------------------+------------------
171 * READY --> ll_receive()
172 * <-- PRESENT(form) ll_reply()
175 * SUBMIT(form) --> ll_receive()
176 * <-- READY ll_reply()
178 * Each of those exchanges is a pair of calls, on our end, to
179 * dfui_tcp_be_ll_receive() and dfui_npipe_be_ll_reply().
181 * The set of messages that the client can pass us is determined by
182 * the conversation state:
184 * o In stage 1, only READY and ABORT are meaningful.
185 * o After a PRESENT, the messages SUBMIT and ABORT are meaningul
187 * o During a PROG_*, the messages CONTINUE, CANCEL, and ABORT
188 * are meaningful in stage 2.
190 * If the frontend sends us with READY in stage 2, we assume it has
191 * fallen out of sync, so we send the same initial reply again, going
192 * back to stage 1 as it were.
194 * After this call, the message is available in c->ebuf.
197 dfui_tcp_be_ll_exchange(struct dfui_connection
*c
, char msgtype
, const char *msg
)
202 * Construct our message to send.
205 fmsg
= malloc(strlen(msg
) + 2);
207 strcpy(fmsg
+ 1, msg
);
210 * Get the frontend's message.
213 dfui_tcp_be_ll_receive(c
);
216 * Frontend message should have been either READY or ABORT.
217 * If ABORT, we get out of here pronto.
220 if (aura_buffer_buf(c
->ebuf
)[0] == DFUI_FE_MSG_ABORT
) {
222 return(DFUI_FAILURE
);
225 /* XXX if (!READY) ??? */
228 dfui_tcp_be_ll_reply(c
, fmsg
);
231 * Here, the frontend has picked up our request and is
232 * processing it. We have to wait for the response.
235 dfui_tcp_be_ll_receive(c
);
238 * Did we get READY from this?
242 } while (aura_buffer_buf(c
->ebuf
)[0] == DFUI_FE_MSG_READY
);
244 fmsg
[0] = DFUI_BE_MSG_READY
;
246 dfui_tcp_be_ll_reply(c
, fmsg
);
249 return(DFUI_SUCCESS
);
253 * Receive a message from the frontend.
254 * This call is synchronous.
255 * After this call, the NUL-terminated message is available in
259 dfui_tcp_be_ll_receive(struct dfui_connection
*c
)
266 if (!T_TCP(c
)->is_connected
) {
267 dfui_debug("NOT_CONNECTED,ACCEPTING_ON<<%d>>\n", T_TCP(c
)->listen_sd
);
268 T_TCP(c
)->connected_sd
= accept(T_TCP(c
)->listen_sd
, NULL
, NULL
);
269 dfui_debug("ACCEPTED<<%d>>\n", T_TCP(c
)->connected_sd
);
270 T_TCP(c
)->stream
= fdopen(T_TCP(c
)->connected_sd
, "r+");
271 T_TCP(c
)->is_connected
= 1;
273 dfui_debug("ALREADY_CONNECTED<<>>\n");
276 dfui_debug("WAITING<<>>\n");
278 if (read_data(T_TCP(c
)->stream
, (char *)&length
, sizeof(length
)) == -1) {
279 dfui_debug("LOST_THEM<<>>\n");
280 fclose(T_TCP(c
)->stream
);
281 T_TCP(c
)->is_connected
= 0;
285 buf
= malloc(length
+ 1);
286 if (read_data(T_TCP(c
)->stream
, buf
, length
) == -1) {
287 dfui_debug("LOST_THEM<<>>\n");
288 fclose(T_TCP(c
)->stream
);
289 T_TCP(c
)->is_connected
= 0;
293 aura_buffer_set(c
->ebuf
, buf
, length
);
296 dfui_debug("RECEIVED<<%s>>\n", aura_buffer_buf(c
->ebuf
));
298 return(DFUI_SUCCESS
);
302 * Send a NUL-terminated reply to the frontend.
305 dfui_tcp_be_ll_reply(struct dfui_connection
*c
, const char *fmsg
)
309 dfui_debug("SEND<<%s>>\n", fmsg
);
310 length
= strlen(fmsg
);
311 write_data(T_TCP(c
)->stream
, (char *)&length
, sizeof(length
));
312 write_data(T_TCP(c
)->stream
, fmsg
, length
);
314 return(DFUI_SUCCESS
);
317 /******** FRONTEND ********/
322 dfui_tcp_fe_connect(struct dfui_connection
*c
)
324 struct sockaddr_in servaddr
;
328 server_port
= atoi(c
->rendezvous
);
331 * Create the tcp socket
335 if ((T_TCP(c
)->connected_sd
= socket(AF_INET
, SOCK_STREAM
, 0)) == -1) {
336 return(DFUI_FAILURE
);
339 dfui_debug("CLIENT_SOCKET<<%d>>\n", T_TCP(c
)->connected_sd
);
340 bzero(&servaddr
, sizeof(servaddr
));
341 servaddr
.sin_family
= AF_INET
;
342 servaddr
.sin_port
= htons(server_port
);
343 inet_pton(AF_INET
, "127.0.0.1", &servaddr
.sin_addr
);
345 if (connect(T_TCP(c
)->connected_sd
, (struct sockaddr
*)&servaddr
,
346 sizeof(servaddr
)) == 0) {
347 dfui_debug("CONNECTED<<>>\n");
350 dfui_debug("NO_CONNECT<<>>\n");
351 close(T_TCP(c
)->connected_sd
);
356 /* at this point we should be connected */
358 T_TCP(c
)->stream
= fdopen(T_TCP(c
)->connected_sd
, "r+");
360 return(DFUI_SUCCESS
);
364 dfui_tcp_fe_disconnect(struct dfui_connection
*c
)
366 close(T_TCP(c
)->connected_sd
);
367 return(DFUI_SUCCESS
);
373 * Ask for, and subsequently receieve, a message from the backend.
374 * msgtype should be one of the DFUI_FE_MSG_* constants.
375 * This call is synchronous.
376 * After this call, the null-terminated, encoded message is
377 * available in T_TCP(c)->buf.
380 dfui_tcp_fe_ll_request(struct dfui_connection
*c
, char msgtype
, const char *msg
)
386 * First, assert that the connection is open.
389 if (c
== NULL
|| T_TCP(c
)->connected_sd
== -1)
390 return(DFUI_FAILURE
);
393 * Construct a message.
396 fmsg
= malloc(strlen(msg
) + 2);
398 strcpy(fmsg
+ 1, msg
);
399 dfui_debug("SEND<<%s>>\n", fmsg
);
402 * Send a NUL-terminated message to the backend.
405 length
= strlen(fmsg
);
406 result
= write_data(T_TCP(c
)->stream
, (char *)&length
, sizeof(length
));
407 dfui_debug("result<<%d>>\n", result
);
408 result
= write_data(T_TCP(c
)->stream
, (char *)fmsg
, length
);
409 dfui_debug("result<<%d>>\n", result
);
412 * Receive a reply from the backend.
413 * If our message was a READY, this should be a message like PRESENT.
414 * Otherwise it should simply be a READY.
417 dfui_debug("WAITING<<>>\n");
418 result
= read_data(T_TCP(c
)->stream
, (char *)&length
, sizeof(length
));
419 dfui_debug("result<<%d>>\n", result
);
420 buf
= malloc(length
+ 1);
421 result
= read_data(T_TCP(c
)->stream
, buf
, length
);
422 dfui_debug("result<<%d>>\n", result
);
423 aura_buffer_set(c
->ebuf
, buf
, length
);
426 dfui_debug("RECV<<%s>>\n", aura_buffer_buf(c
->ebuf
));
430 return(DFUI_SUCCESS
);
434 read_data(FILE *f
, char *buf
, int n
)
436 int bcount
; /* counts bytes read */
437 int br
; /* bytes read this pass */
442 if ((br
= fread(buf
, 1, n
- bcount
, f
)) > 0) {
443 dfui_debug("READ_BYTES<<%d>>\n", br
);
446 } else if (br
<= 0) {
447 dfui_debug("read_data_error<<%d>>\n", br
);
455 write_data(FILE *f
, const char *buf
, int n
)
457 int bcount
; /* counts bytes written */
458 int bw
; /* bytes written this pass */
463 if ((bw
= fwrite(buf
, 1, n
- bcount
, f
)) > 0) {
464 dfui_debug("WROTE_BYTES<<%d>>\n", bw
);
467 } else if (bw
<= 0) {
468 dfui_debug("write_data_error<<%d>>\n", bw
);