xref: /nrf52832-nimble/rt-thread/components/net/uip/apps/webclient/webclient.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /**
2*10465441SEvalZero  * \addtogroup apps
3*10465441SEvalZero  * @{
4*10465441SEvalZero  */
5*10465441SEvalZero 
6*10465441SEvalZero /**
7*10465441SEvalZero  * \defgroup webclient Web client
8*10465441SEvalZero  * @{
9*10465441SEvalZero  *
10*10465441SEvalZero  * This example shows a HTTP client that is able to download web pages
11*10465441SEvalZero  * and files from web servers. It requires a number of callback
12*10465441SEvalZero  * functions to be implemented by the module that utilizes the code:
13*10465441SEvalZero  * webclient_datahandler(), webclient_connected(),
14*10465441SEvalZero  * webclient_timedout(), webclient_aborted(), webclient_closed().
15*10465441SEvalZero  */
16*10465441SEvalZero 
17*10465441SEvalZero /**
18*10465441SEvalZero  * \file
19*10465441SEvalZero  * Implementation of the HTTP client.
20*10465441SEvalZero  * \author Adam Dunkels <adam@dunkels.com>
21*10465441SEvalZero  */
22*10465441SEvalZero 
23*10465441SEvalZero /*
24*10465441SEvalZero  * Copyright (c) 2002, Adam Dunkels.
25*10465441SEvalZero  * All rights reserved.
26*10465441SEvalZero  *
27*10465441SEvalZero  * Redistribution and use in source and binary forms, with or without
28*10465441SEvalZero  * modification, are permitted provided that the following conditions
29*10465441SEvalZero  * are met:
30*10465441SEvalZero  * 1. Redistributions of source code must retain the above copyright
31*10465441SEvalZero  *    notice, this list of conditions and the following disclaimer.
32*10465441SEvalZero  * 2. Redistributions in binary form must reproduce the above
33*10465441SEvalZero  *    copyright notice, this list of conditions and the following
34*10465441SEvalZero  *    disclaimer in the documentation and/or other materials provided
35*10465441SEvalZero  *    with the distribution.
36*10465441SEvalZero  * 3. The name of the author may not be used to endorse or promote
37*10465441SEvalZero  *    products derived from this software without specific prior
38*10465441SEvalZero  *    written permission.
39*10465441SEvalZero  *
40*10465441SEvalZero  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
41*10465441SEvalZero  * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
42*10465441SEvalZero  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
43*10465441SEvalZero  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
44*10465441SEvalZero  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
45*10465441SEvalZero  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
46*10465441SEvalZero  * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
47*10465441SEvalZero  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
48*10465441SEvalZero  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
49*10465441SEvalZero  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
50*10465441SEvalZero  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
51*10465441SEvalZero  *
52*10465441SEvalZero  * This file is part of the uIP TCP/IP stack.
53*10465441SEvalZero  *
54*10465441SEvalZero  * $Id: webclient.c,v 1.2 2006/06/11 21:46:37 adam Exp $
55*10465441SEvalZero  *
56*10465441SEvalZero  */
57*10465441SEvalZero 
58*10465441SEvalZero #include "uip.h"
59*10465441SEvalZero #include "uiplib.h"
60*10465441SEvalZero #include "webclient.h"
61*10465441SEvalZero #include "resolv.h"
62*10465441SEvalZero 
63*10465441SEvalZero #include <string.h>
64*10465441SEvalZero 
65*10465441SEvalZero #define WEBCLIENT_TIMEOUT 100
66*10465441SEvalZero 
67*10465441SEvalZero #define WEBCLIENT_STATE_STATUSLINE 0
68*10465441SEvalZero #define WEBCLIENT_STATE_HEADERS    1
69*10465441SEvalZero #define WEBCLIENT_STATE_DATA       2
70*10465441SEvalZero #define WEBCLIENT_STATE_CLOSE      3
71*10465441SEvalZero 
72*10465441SEvalZero #define HTTPFLAG_NONE   0
73*10465441SEvalZero #define HTTPFLAG_OK     1
74*10465441SEvalZero #define HTTPFLAG_MOVED  2
75*10465441SEvalZero #define HTTPFLAG_ERROR  3
76*10465441SEvalZero 
77*10465441SEvalZero 
78*10465441SEvalZero #define ISO_nl       0x0a
79*10465441SEvalZero #define ISO_cr       0x0d
80*10465441SEvalZero #define ISO_space    0x20
81*10465441SEvalZero 
82*10465441SEvalZero 
83*10465441SEvalZero static struct webclient_state s;
84*10465441SEvalZero 
85*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
86*10465441SEvalZero char *
webclient_mimetype(void)87*10465441SEvalZero webclient_mimetype(void)
88*10465441SEvalZero {
89*10465441SEvalZero   return s.mimetype;
90*10465441SEvalZero }
91*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
92*10465441SEvalZero char *
webclient_filename(void)93*10465441SEvalZero webclient_filename(void)
94*10465441SEvalZero {
95*10465441SEvalZero   return s.file;
96*10465441SEvalZero }
97*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
98*10465441SEvalZero char *
webclient_hostname(void)99*10465441SEvalZero webclient_hostname(void)
100*10465441SEvalZero {
101*10465441SEvalZero   return s.host;
102*10465441SEvalZero }
103*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
104*10465441SEvalZero unsigned short
webclient_port(void)105*10465441SEvalZero webclient_port(void)
106*10465441SEvalZero {
107*10465441SEvalZero   return s.port;
108*10465441SEvalZero }
109*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
110*10465441SEvalZero void
webclient_init(void)111*10465441SEvalZero webclient_init(void)
112*10465441SEvalZero {
113*10465441SEvalZero 
114*10465441SEvalZero }
115*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
116*10465441SEvalZero static void
init_connection(void)117*10465441SEvalZero init_connection(void)
118*10465441SEvalZero {
119*10465441SEvalZero   s.state = WEBCLIENT_STATE_STATUSLINE;
120*10465441SEvalZero 
121*10465441SEvalZero   s.getrequestleft = sizeof(http_get) - 1 + 1 +
122*10465441SEvalZero     sizeof(http_10) - 1 +
123*10465441SEvalZero     sizeof(http_crnl) - 1 +
124*10465441SEvalZero     sizeof(http_host) - 1 +
125*10465441SEvalZero     sizeof(http_crnl) - 1 +
126*10465441SEvalZero     strlen(http_user_agent_fields) +
127*10465441SEvalZero     strlen(s.file) + strlen(s.host);
128*10465441SEvalZero   s.getrequestptr = 0;
129*10465441SEvalZero 
130*10465441SEvalZero   s.httpheaderlineptr = 0;
131*10465441SEvalZero }
132*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
133*10465441SEvalZero void
webclient_close(void)134*10465441SEvalZero webclient_close(void)
135*10465441SEvalZero {
136*10465441SEvalZero   s.state = WEBCLIENT_STATE_CLOSE;
137*10465441SEvalZero }
138*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
139*10465441SEvalZero unsigned char
webclient_get(char * host,u16_t port,char * file)140*10465441SEvalZero webclient_get(char *host, u16_t port, char *file)
141*10465441SEvalZero {
142*10465441SEvalZero   struct uip_conn *conn;
143*10465441SEvalZero   uip_ipaddr_t *ipaddr;
144*10465441SEvalZero   static uip_ipaddr_t addr;
145*10465441SEvalZero 
146*10465441SEvalZero   /* First check if the host is an IP address. */
147*10465441SEvalZero   ipaddr = &addr;
148*10465441SEvalZero   if(uiplib_ipaddrconv(host, (unsigned char *)addr) == 0) {
149*10465441SEvalZero     ipaddr = (uip_ipaddr_t *)resolv_lookup(host);
150*10465441SEvalZero 
151*10465441SEvalZero     if(ipaddr == NULL) {
152*10465441SEvalZero       return 0;
153*10465441SEvalZero     }
154*10465441SEvalZero   }
155*10465441SEvalZero 
156*10465441SEvalZero   conn = uip_connect(ipaddr, htons(port));
157*10465441SEvalZero 
158*10465441SEvalZero   if(conn == NULL) {
159*10465441SEvalZero     return 0;
160*10465441SEvalZero   }
161*10465441SEvalZero 
162*10465441SEvalZero   s.port = port;
163*10465441SEvalZero   strncpy(s.file, file, sizeof(s.file));
164*10465441SEvalZero   strncpy(s.host, host, sizeof(s.host));
165*10465441SEvalZero 
166*10465441SEvalZero   init_connection();
167*10465441SEvalZero   return 1;
168*10465441SEvalZero }
169*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
170*10465441SEvalZero static unsigned char *
copy_string(unsigned char * dest,const unsigned char * src,unsigned char len)171*10465441SEvalZero copy_string(unsigned char *dest,
172*10465441SEvalZero 	    const unsigned char *src, unsigned char len)
173*10465441SEvalZero {
174*10465441SEvalZero   strncpy(dest, src, len);
175*10465441SEvalZero   return dest + len;
176*10465441SEvalZero }
177*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
178*10465441SEvalZero static void
senddata(void)179*10465441SEvalZero senddata(void)
180*10465441SEvalZero {
181*10465441SEvalZero   u16_t len;
182*10465441SEvalZero   char *getrequest;
183*10465441SEvalZero   char *cptr;
184*10465441SEvalZero 
185*10465441SEvalZero   if(s.getrequestleft > 0) {
186*10465441SEvalZero     cptr = getrequest = (char *)uip_appdata;
187*10465441SEvalZero 
188*10465441SEvalZero     cptr = copy_string(cptr, http_get, sizeof(http_get) - 1);
189*10465441SEvalZero     cptr = copy_string(cptr, s.file, strlen(s.file));
190*10465441SEvalZero     *cptr++ = ISO_space;
191*10465441SEvalZero     cptr = copy_string(cptr, http_10, sizeof(http_10) - 1);
192*10465441SEvalZero 
193*10465441SEvalZero     cptr = copy_string(cptr, http_crnl, sizeof(http_crnl) - 1);
194*10465441SEvalZero 
195*10465441SEvalZero     cptr = copy_string(cptr, http_host, sizeof(http_host) - 1);
196*10465441SEvalZero     cptr = copy_string(cptr, s.host, strlen(s.host));
197*10465441SEvalZero     cptr = copy_string(cptr, http_crnl, sizeof(http_crnl) - 1);
198*10465441SEvalZero 
199*10465441SEvalZero     cptr = copy_string(cptr, http_user_agent_fields,
200*10465441SEvalZero 		       strlen(http_user_agent_fields));
201*10465441SEvalZero 
202*10465441SEvalZero     len = s.getrequestleft > uip_mss()?
203*10465441SEvalZero       uip_mss():
204*10465441SEvalZero       s.getrequestleft;
205*10465441SEvalZero     uip_send(&(getrequest[s.getrequestptr]), len);
206*10465441SEvalZero   }
207*10465441SEvalZero }
208*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
209*10465441SEvalZero static void
acked(void)210*10465441SEvalZero acked(void)
211*10465441SEvalZero {
212*10465441SEvalZero   u16_t len;
213*10465441SEvalZero 
214*10465441SEvalZero   if(s.getrequestleft > 0) {
215*10465441SEvalZero     len = s.getrequestleft > uip_mss()?
216*10465441SEvalZero       uip_mss():
217*10465441SEvalZero       s.getrequestleft;
218*10465441SEvalZero     s.getrequestleft -= len;
219*10465441SEvalZero     s.getrequestptr += len;
220*10465441SEvalZero   }
221*10465441SEvalZero }
222*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
223*10465441SEvalZero static u16_t
parse_statusline(u16_t len)224*10465441SEvalZero parse_statusline(u16_t len)
225*10465441SEvalZero {
226*10465441SEvalZero   char *cptr;
227*10465441SEvalZero 
228*10465441SEvalZero   while(len > 0 && s.httpheaderlineptr < sizeof(s.httpheaderline)) {
229*10465441SEvalZero     s.httpheaderline[s.httpheaderlineptr] = *(char *)uip_appdata;
230*10465441SEvalZero     ++((char *)uip_appdata);
231*10465441SEvalZero     --len;
232*10465441SEvalZero     if(s.httpheaderline[s.httpheaderlineptr] == ISO_nl) {
233*10465441SEvalZero 
234*10465441SEvalZero       if((strncmp(s.httpheaderline, http_10,
235*10465441SEvalZero 		  sizeof(http_10) - 1) == 0) ||
236*10465441SEvalZero 	 (strncmp(s.httpheaderline, http_11,
237*10465441SEvalZero 		  sizeof(http_11) - 1) == 0)) {
238*10465441SEvalZero 	cptr = &(s.httpheaderline[9]);
239*10465441SEvalZero 	s.httpflag = HTTPFLAG_NONE;
240*10465441SEvalZero 	if(strncmp(cptr, http_200, sizeof(http_200) - 1) == 0) {
241*10465441SEvalZero 	  /* 200 OK */
242*10465441SEvalZero 	  s.httpflag = HTTPFLAG_OK;
243*10465441SEvalZero 	} else if(strncmp(cptr, http_301, sizeof(http_301) - 1) == 0 ||
244*10465441SEvalZero 		  strncmp(cptr, http_302, sizeof(http_302) - 1) == 0) {
245*10465441SEvalZero 	  /* 301 Moved permanently or 302 Found. Location: header line
246*10465441SEvalZero 	     will contain thw new location. */
247*10465441SEvalZero 	  s.httpflag = HTTPFLAG_MOVED;
248*10465441SEvalZero 	} else {
249*10465441SEvalZero 	  s.httpheaderline[s.httpheaderlineptr - 1] = 0;
250*10465441SEvalZero 	}
251*10465441SEvalZero       } else {
252*10465441SEvalZero 	uip_abort();
253*10465441SEvalZero 	webclient_aborted();
254*10465441SEvalZero 	return 0;
255*10465441SEvalZero       }
256*10465441SEvalZero 
257*10465441SEvalZero       /* We're done parsing the status line, so we reset the pointer
258*10465441SEvalZero 	 and start parsing the HTTP headers.*/
259*10465441SEvalZero       s.httpheaderlineptr = 0;
260*10465441SEvalZero       s.state = WEBCLIENT_STATE_HEADERS;
261*10465441SEvalZero       break;
262*10465441SEvalZero     } else {
263*10465441SEvalZero       ++s.httpheaderlineptr;
264*10465441SEvalZero     }
265*10465441SEvalZero   }
266*10465441SEvalZero   return len;
267*10465441SEvalZero }
268*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
269*10465441SEvalZero static char
casecmp(char * str1,const char * str2,char len)270*10465441SEvalZero casecmp(char *str1, const char *str2, char len)
271*10465441SEvalZero {
272*10465441SEvalZero   static char c;
273*10465441SEvalZero 
274*10465441SEvalZero   while(len > 0) {
275*10465441SEvalZero     c = *str1;
276*10465441SEvalZero     /* Force lower-case characters. */
277*10465441SEvalZero     if(c & 0x40) {
278*10465441SEvalZero       c |= 0x20;
279*10465441SEvalZero     }
280*10465441SEvalZero     if(*str2 != c) {
281*10465441SEvalZero       return 1;
282*10465441SEvalZero     }
283*10465441SEvalZero     ++str1;
284*10465441SEvalZero     ++str2;
285*10465441SEvalZero     --len;
286*10465441SEvalZero   }
287*10465441SEvalZero   return 0;
288*10465441SEvalZero }
289*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
290*10465441SEvalZero static u16_t
parse_headers(u16_t len)291*10465441SEvalZero parse_headers(u16_t len)
292*10465441SEvalZero {
293*10465441SEvalZero   char *cptr;
294*10465441SEvalZero   static unsigned char i;
295*10465441SEvalZero 
296*10465441SEvalZero   while(len > 0 && s.httpheaderlineptr < sizeof(s.httpheaderline)) {
297*10465441SEvalZero     s.httpheaderline[s.httpheaderlineptr] = *(char *)uip_appdata;
298*10465441SEvalZero     ++((char *)uip_appdata);
299*10465441SEvalZero     --len;
300*10465441SEvalZero     if(s.httpheaderline[s.httpheaderlineptr] == ISO_nl) {
301*10465441SEvalZero       /* We have an entire HTTP header line in s.httpheaderline, so
302*10465441SEvalZero 	 we parse it. */
303*10465441SEvalZero       if(s.httpheaderline[0] == ISO_cr) {
304*10465441SEvalZero 	/* This was the last header line (i.e., and empty "\r\n"), so
305*10465441SEvalZero 	   we are done with the headers and proceed with the actual
306*10465441SEvalZero 	   data. */
307*10465441SEvalZero 	s.state = WEBCLIENT_STATE_DATA;
308*10465441SEvalZero 	return len;
309*10465441SEvalZero       }
310*10465441SEvalZero 
311*10465441SEvalZero       s.httpheaderline[s.httpheaderlineptr - 1] = 0;
312*10465441SEvalZero       /* Check for specific HTTP header fields. */
313*10465441SEvalZero       if(casecmp(s.httpheaderline, http_content_type,
314*10465441SEvalZero 		     sizeof(http_content_type) - 1) == 0) {
315*10465441SEvalZero 	/* Found Content-type field. */
316*10465441SEvalZero 	cptr = strchr(s.httpheaderline, ';');
317*10465441SEvalZero 	if(cptr != NULL) {
318*10465441SEvalZero 	  *cptr = 0;
319*10465441SEvalZero 	}
320*10465441SEvalZero 	strncpy(s.mimetype, s.httpheaderline +
321*10465441SEvalZero 		sizeof(http_content_type) - 1, sizeof(s.mimetype));
322*10465441SEvalZero       } else if(casecmp(s.httpheaderline, http_location,
323*10465441SEvalZero 			    sizeof(http_location) - 1) == 0) {
324*10465441SEvalZero 	cptr = s.httpheaderline +
325*10465441SEvalZero 	  sizeof(http_location) - 1;
326*10465441SEvalZero 
327*10465441SEvalZero 	if(strncmp(cptr, http_http, 7) == 0) {
328*10465441SEvalZero 	  cptr += 7;
329*10465441SEvalZero 	  for(i = 0; i < s.httpheaderlineptr - 7; ++i) {
330*10465441SEvalZero 	    if(*cptr == 0 ||
331*10465441SEvalZero 	       *cptr == '/' ||
332*10465441SEvalZero 	       *cptr == ' ' ||
333*10465441SEvalZero 	       *cptr == ':') {
334*10465441SEvalZero 	      s.host[i] = 0;
335*10465441SEvalZero 	      break;
336*10465441SEvalZero 	    }
337*10465441SEvalZero 	    s.host[i] = *cptr;
338*10465441SEvalZero 	    ++cptr;
339*10465441SEvalZero 	  }
340*10465441SEvalZero 	}
341*10465441SEvalZero 	strncpy(s.file, cptr, sizeof(s.file));
342*10465441SEvalZero 	/*	s.file[s.httpheaderlineptr - i] = 0;*/
343*10465441SEvalZero       }
344*10465441SEvalZero 
345*10465441SEvalZero 
346*10465441SEvalZero       /* We're done parsing, so we reset the pointer and start the
347*10465441SEvalZero 	 next line. */
348*10465441SEvalZero       s.httpheaderlineptr = 0;
349*10465441SEvalZero     } else {
350*10465441SEvalZero       ++s.httpheaderlineptr;
351*10465441SEvalZero     }
352*10465441SEvalZero   }
353*10465441SEvalZero   return len;
354*10465441SEvalZero }
355*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
356*10465441SEvalZero static void
newdata(void)357*10465441SEvalZero newdata(void)
358*10465441SEvalZero {
359*10465441SEvalZero   u16_t len;
360*10465441SEvalZero 
361*10465441SEvalZero   len = uip_datalen();
362*10465441SEvalZero 
363*10465441SEvalZero   if(s.state == WEBCLIENT_STATE_STATUSLINE) {
364*10465441SEvalZero     len = parse_statusline(len);
365*10465441SEvalZero   }
366*10465441SEvalZero 
367*10465441SEvalZero   if(s.state == WEBCLIENT_STATE_HEADERS && len > 0) {
368*10465441SEvalZero     len = parse_headers(len);
369*10465441SEvalZero   }
370*10465441SEvalZero 
371*10465441SEvalZero   if(len > 0 && s.state == WEBCLIENT_STATE_DATA &&
372*10465441SEvalZero      s.httpflag != HTTPFLAG_MOVED) {
373*10465441SEvalZero     webclient_datahandler((char *)uip_appdata, len);
374*10465441SEvalZero   }
375*10465441SEvalZero }
376*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
377*10465441SEvalZero void
webclient_appcall(void)378*10465441SEvalZero webclient_appcall(void)
379*10465441SEvalZero {
380*10465441SEvalZero   if(uip_connected()) {
381*10465441SEvalZero     s.timer = 0;
382*10465441SEvalZero     s.state = WEBCLIENT_STATE_STATUSLINE;
383*10465441SEvalZero     senddata();
384*10465441SEvalZero     webclient_connected();
385*10465441SEvalZero     return;
386*10465441SEvalZero   }
387*10465441SEvalZero 
388*10465441SEvalZero   if(s.state == WEBCLIENT_STATE_CLOSE) {
389*10465441SEvalZero     webclient_closed();
390*10465441SEvalZero     uip_abort();
391*10465441SEvalZero     return;
392*10465441SEvalZero   }
393*10465441SEvalZero 
394*10465441SEvalZero   if(uip_aborted()) {
395*10465441SEvalZero     webclient_aborted();
396*10465441SEvalZero   }
397*10465441SEvalZero   if(uip_timedout()) {
398*10465441SEvalZero     webclient_timedout();
399*10465441SEvalZero   }
400*10465441SEvalZero 
401*10465441SEvalZero 
402*10465441SEvalZero   if(uip_acked()) {
403*10465441SEvalZero     s.timer = 0;
404*10465441SEvalZero     acked();
405*10465441SEvalZero   }
406*10465441SEvalZero   if(uip_newdata()) {
407*10465441SEvalZero     s.timer = 0;
408*10465441SEvalZero     newdata();
409*10465441SEvalZero   }
410*10465441SEvalZero   if(uip_rexmit() ||
411*10465441SEvalZero      uip_newdata() ||
412*10465441SEvalZero      uip_acked()) {
413*10465441SEvalZero     senddata();
414*10465441SEvalZero   } else if(uip_poll()) {
415*10465441SEvalZero     ++s.timer;
416*10465441SEvalZero     if(s.timer == WEBCLIENT_TIMEOUT) {
417*10465441SEvalZero       webclient_timedout();
418*10465441SEvalZero       uip_abort();
419*10465441SEvalZero       return;
420*10465441SEvalZero     }
421*10465441SEvalZero         /*    senddata();*/
422*10465441SEvalZero   }
423*10465441SEvalZero 
424*10465441SEvalZero   if(uip_closed()) {
425*10465441SEvalZero     if(s.httpflag != HTTPFLAG_MOVED) {
426*10465441SEvalZero       /* Send NULL data to signal EOF. */
427*10465441SEvalZero       webclient_datahandler(NULL, 0);
428*10465441SEvalZero     } else {
429*10465441SEvalZero       if(resolv_lookup(s.host) == NULL) {
430*10465441SEvalZero 	resolv_query(s.host);
431*10465441SEvalZero       }
432*10465441SEvalZero       webclient_get(s.host, s.port, s.file);
433*10465441SEvalZero     }
434*10465441SEvalZero   }
435*10465441SEvalZero }
436*10465441SEvalZero /*-----------------------------------------------------------------------------------*/
437*10465441SEvalZero 
438*10465441SEvalZero /** @} */
439*10465441SEvalZero /** @} */
440