W5200(WIZ820io) network interface

Committer:
va009039
Date:
Sat Apr 14 17:21:11 2012 +0000
Revision:
0:61831b843b44

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
va009039 0:61831b843b44 1 /*
va009039 0:61831b843b44 2 * mbed Tiny HTTP Client
va009039 0:61831b843b44 3 * Copyright (c) 2011 Hiroshi Suga
va009039 0:61831b843b44 4 * Released under the MIT License: http://mbed.org/license/mit
va009039 0:61831b843b44 5 */
va009039 0:61831b843b44 6
va009039 0:61831b843b44 7 /** @file
va009039 0:61831b843b44 8 * @brief Tiny HTTP Client
va009039 0:61831b843b44 9 */
va009039 0:61831b843b44 10
va009039 0:61831b843b44 11 #include "mbed.h"
va009039 0:61831b843b44 12 #include "EthernetNetIf.h"
va009039 0:61831b843b44 13 #include "TCPSocket.h"
va009039 0:61831b843b44 14 #include "DNSRequest.h"
va009039 0:61831b843b44 15 #include "TinyHTTP.h"
va009039 0:61831b843b44 16 #include <ctype.h>
va009039 0:61831b843b44 17
va009039 0:61831b843b44 18
va009039 0:61831b843b44 19 TCPSocket *http;
va009039 0:61831b843b44 20 volatile int tcp_ready, tcp_readable, tcp_writable;
va009039 0:61831b843b44 21 volatile int dns_status;
va009039 0:61831b843b44 22
va009039 0:61831b843b44 23 // Copyright (c) 2010 Donatien Garnier (donatiengar [at] gmail [dot] com)
va009039 0:61831b843b44 24 int base64enc(const char *input, unsigned int length, char *output, int len) {
va009039 0:61831b843b44 25 static const char base64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
va009039 0:61831b843b44 26 unsigned int c, c1, c2, c3;
va009039 0:61831b843b44 27
va009039 0:61831b843b44 28 if (len < ((((length-1)/3)+1)<<2)) return -1;
va009039 0:61831b843b44 29 for(unsigned int i = 0, j = 0; i<length; i+=3,j+=4) {
va009039 0:61831b843b44 30 c1 = ((((unsigned char)*((unsigned char *)&input[i]))));
va009039 0:61831b843b44 31 c2 = (length>i+1)?((((unsigned char)*((unsigned char *)&input[i+1])))):0;
va009039 0:61831b843b44 32 c3 = (length>i+2)?((((unsigned char)*((unsigned char *)&input[i+2])))):0;
va009039 0:61831b843b44 33
va009039 0:61831b843b44 34 c = ((c1 & 0xFC) >> 2);
va009039 0:61831b843b44 35 output[j+0] = base64[c];
va009039 0:61831b843b44 36 c = ((c1 & 0x03) << 4) | ((c2 & 0xF0) >> 4);
va009039 0:61831b843b44 37 output[j+1] = base64[c];
va009039 0:61831b843b44 38 c = ((c2 & 0x0F) << 2) | ((c3 & 0xC0) >> 6);
va009039 0:61831b843b44 39 output[j+2] = (length>i+1)?base64[c]:'=';
va009039 0:61831b843b44 40 c = (c3 & 0x3F);
va009039 0:61831b843b44 41 output[j+3] = (length>i+2)?base64[c]:'=';
va009039 0:61831b843b44 42 }
va009039 0:61831b843b44 43 output[(((length-1)/3)+1)<<2] = '\0';
va009039 0:61831b843b44 44 return 0;
va009039 0:61831b843b44 45 }
va009039 0:61831b843b44 46
va009039 0:61831b843b44 47 // Copyright (c) 2010 Donatien Garnier (donatiengar [at] gmail [dot] com)
va009039 0:61831b843b44 48 int urlencode(char *str, char *buf, int len) {
va009039 0:61831b843b44 49 static const char to_hex[] = "0123456789ABCDEF";
va009039 0:61831b843b44 50 // char *pstr = str, *buf = (char*)malloc(strlen(str) * 3 + 1), *pbuf = buf;
va009039 0:61831b843b44 51 char *pstr = str, *pbuf = buf;
va009039 0:61831b843b44 52
va009039 0:61831b843b44 53 if (len < (strlen(str) * 3 + 1)) return -1;
va009039 0:61831b843b44 54 while (*pstr) {
va009039 0:61831b843b44 55 if (isalnum(*pstr) || *pstr == '-' || *pstr == '_' || *pstr == '.' || *pstr == '~') {
va009039 0:61831b843b44 56 *pbuf++ = *pstr;
va009039 0:61831b843b44 57 } else if (*pstr == ' ') {
va009039 0:61831b843b44 58 *pbuf++ = '+';
va009039 0:61831b843b44 59 } else {
va009039 0:61831b843b44 60 *pbuf++ = '%';
va009039 0:61831b843b44 61 *pbuf++ = to_hex[(*pstr >> 4) & 0x0f];
va009039 0:61831b843b44 62 *pbuf++ = to_hex[*pstr & 0x0f];
va009039 0:61831b843b44 63 }
va009039 0:61831b843b44 64 pstr++;
va009039 0:61831b843b44 65 }
va009039 0:61831b843b44 66 *pbuf = '\0';
va009039 0:61831b843b44 67 return 0;
va009039 0:61831b843b44 68 }
va009039 0:61831b843b44 69
va009039 0:61831b843b44 70
va009039 0:61831b843b44 71 void isr_http (TCPSocketEvent e) {
va009039 0:61831b843b44 72
va009039 0:61831b843b44 73 #ifdef DEBUG
va009039 0:61831b843b44 74 printf("tcp(%d)\r\n", e);
va009039 0:61831b843b44 75 #endif
va009039 0:61831b843b44 76 switch(e) {
va009039 0:61831b843b44 77 case TCPSOCKET_CONNECTED:
va009039 0:61831b843b44 78 tcp_ready = 1;
va009039 0:61831b843b44 79 break;
va009039 0:61831b843b44 80
va009039 0:61831b843b44 81 case TCPSOCKET_READABLE: //Incoming data
va009039 0:61831b843b44 82 tcp_readable = 1;
va009039 0:61831b843b44 83 break;
va009039 0:61831b843b44 84
va009039 0:61831b843b44 85 case TCPSOCKET_WRITEABLE: //We can send data
va009039 0:61831b843b44 86 tcp_writable = 1;
va009039 0:61831b843b44 87 break;
va009039 0:61831b843b44 88
va009039 0:61831b843b44 89 case TCPSOCKET_CONTIMEOUT:
va009039 0:61831b843b44 90 case TCPSOCKET_CONRST:
va009039 0:61831b843b44 91 case TCPSOCKET_CONABRT:
va009039 0:61831b843b44 92 case TCPSOCKET_ERROR:
va009039 0:61831b843b44 93 case TCPSOCKET_DISCONNECTED:
va009039 0:61831b843b44 94 tcp_ready = 0;
va009039 0:61831b843b44 95 break;
va009039 0:61831b843b44 96 }
va009039 0:61831b843b44 97 }
va009039 0:61831b843b44 98
va009039 0:61831b843b44 99 void createauth (char *user, char *pwd, char *buf, int len) {
va009039 0:61831b843b44 100 char tmp[80];
va009039 0:61831b843b44 101
va009039 0:61831b843b44 102 strncpy(buf, "Authorization: Basic ", len);
va009039 0:61831b843b44 103 snprintf(tmp, sizeof(tmp), "%s:%s", user, pwd);
va009039 0:61831b843b44 104 base64enc(tmp, strlen(tmp), &buf[strlen(buf)], len - strlen(buf));
va009039 0:61831b843b44 105 strncat(buf, "\r\n", len - strlen(buf));
va009039 0:61831b843b44 106 }
va009039 0:61831b843b44 107
va009039 0:61831b843b44 108 void isr_dns (DNSReply r) {
va009039 0:61831b843b44 109
va009039 0:61831b843b44 110 #ifdef DEBUG
va009039 0:61831b843b44 111 printf("dns(%d)\r\n", r);
va009039 0:61831b843b44 112 #endif
va009039 0:61831b843b44 113 if (r == DNS_FOUND) {
va009039 0:61831b843b44 114 dns_status = 1;
va009039 0:61831b843b44 115 } else {
va009039 0:61831b843b44 116 dns_status = -1;
va009039 0:61831b843b44 117 }
va009039 0:61831b843b44 118 }
va009039 0:61831b843b44 119
va009039 0:61831b843b44 120 int httpRequest (int method, Host *host, char *uri, char *head, char *body) {
va009039 0:61831b843b44 121 TCPSocketErr err;
va009039 0:61831b843b44 122 Timer timeout;
va009039 0:61831b843b44 123 char buf[1500];
va009039 0:61831b843b44 124 int i, ret = -1;
va009039 0:61831b843b44 125
va009039 0:61831b843b44 126 http = new TCPSocket;
va009039 0:61831b843b44 127 tcp_ready = 0;
va009039 0:61831b843b44 128 tcp_readable = 0;
va009039 0:61831b843b44 129 tcp_writable = 0;
va009039 0:61831b843b44 130
va009039 0:61831b843b44 131 http->setOnEvent(isr_http);
va009039 0:61831b843b44 132
va009039 0:61831b843b44 133 // connect
va009039 0:61831b843b44 134 if (host->getIp().isNull()) {
va009039 0:61831b843b44 135 // resolv
va009039 0:61831b843b44 136 DNSRequest dns;
va009039 0:61831b843b44 137 dns_status = 0;
va009039 0:61831b843b44 138 dns.setOnReply(isr_dns);
va009039 0:61831b843b44 139 if (dns.resolve(host) != DNS_OK) goto exit;
va009039 0:61831b843b44 140 timeout.reset();
va009039 0:61831b843b44 141 timeout.start();
va009039 0:61831b843b44 142 while (timeout.read_ms() < HTTP_TIMEOUT) {
va009039 0:61831b843b44 143 if (dns_status) break;
va009039 0:61831b843b44 144 Net::poll();
va009039 0:61831b843b44 145 }
va009039 0:61831b843b44 146 timeout.stop();
va009039 0:61831b843b44 147 if (dns_status <= 0) goto exit;
va009039 0:61831b843b44 148 #ifdef DEBUG
va009039 0:61831b843b44 149 printf("%s [%d.%d.%d.%d]\r\n", host->getName(), (unsigned char)host->getIp()[0], (unsigned char)host->getIp()[1], (unsigned char)host->getIp()[2], (unsigned char)host->getIp()[3]);
va009039 0:61831b843b44 150 #endif
va009039 0:61831b843b44 151 }
va009039 0:61831b843b44 152 if (! host->getPort()) {
va009039 0:61831b843b44 153 host->setPort(HTTP_PORT);
va009039 0:61831b843b44 154 }
va009039 0:61831b843b44 155 err = http->connect(*host);
va009039 0:61831b843b44 156 if (err != TCPSOCKET_OK) goto exit;
va009039 0:61831b843b44 157
va009039 0:61831b843b44 158 // wait connect
va009039 0:61831b843b44 159 timeout.reset();
va009039 0:61831b843b44 160 timeout.start();
va009039 0:61831b843b44 161 while (timeout.read_ms() < HTTP_TIMEOUT) {
va009039 0:61831b843b44 162 if (tcp_ready) break;
va009039 0:61831b843b44 163 Net::poll();
va009039 0:61831b843b44 164 }
va009039 0:61831b843b44 165 timeout.stop();
va009039 0:61831b843b44 166 if (! tcp_ready) goto exit;
va009039 0:61831b843b44 167
va009039 0:61831b843b44 168 // send request
va009039 0:61831b843b44 169 if (method == METHOD_POST) {
va009039 0:61831b843b44 170 http->send("POST ", 5);
va009039 0:61831b843b44 171 } else {
va009039 0:61831b843b44 172 http->send("GET ", 4);
va009039 0:61831b843b44 173 }
va009039 0:61831b843b44 174 http->send(uri, strlen(uri));
va009039 0:61831b843b44 175 http->send(" HTTP/1.1\r\nHost: ", 17);
va009039 0:61831b843b44 176 http->send(host->getName(), strlen(host->getName()));
va009039 0:61831b843b44 177 http->send("\r\n", 2);
va009039 0:61831b843b44 178 http->send("Connection: close\r\n", 19);
va009039 0:61831b843b44 179 if (head) {
va009039 0:61831b843b44 180 http->send(head, strlen(head));
va009039 0:61831b843b44 181 }
va009039 0:61831b843b44 182 if (method == METHOD_POST) {
va009039 0:61831b843b44 183 sprintf(buf, "Content-Length: %d\r\n", strlen(body));
va009039 0:61831b843b44 184 http->send(buf, strlen(buf));
va009039 0:61831b843b44 185 }
va009039 0:61831b843b44 186 http->send("\r\n", 2);
va009039 0:61831b843b44 187
va009039 0:61831b843b44 188 // post method
va009039 0:61831b843b44 189 if (method == METHOD_POST && body) {
va009039 0:61831b843b44 190 http->send(body, strlen(body));
va009039 0:61831b843b44 191 }
va009039 0:61831b843b44 192
va009039 0:61831b843b44 193 // wait responce
va009039 0:61831b843b44 194 timeout.reset();
va009039 0:61831b843b44 195 timeout.start();
va009039 0:61831b843b44 196 while (timeout.read_ms() < HTTP_TIMEOUT) {
va009039 0:61831b843b44 197 if (tcp_readable) break;
va009039 0:61831b843b44 198 Net::poll();
va009039 0:61831b843b44 199 }
va009039 0:61831b843b44 200 timeout.stop();
va009039 0:61831b843b44 201 if (! tcp_readable) goto exit;
va009039 0:61831b843b44 202
va009039 0:61831b843b44 203 // recv responce
va009039 0:61831b843b44 204 i = http->recv(buf, sizeof(buf) - 1);
va009039 0:61831b843b44 205 buf[i] = 0;
va009039 0:61831b843b44 206 if (i < sizeof(buf) - 1) tcp_readable = 0;
va009039 0:61831b843b44 207 if (strncmp(buf, "HTTP/", 5) == 0) {
va009039 0:61831b843b44 208 ret = atoi(&buf[9]);
va009039 0:61831b843b44 209 }
va009039 0:61831b843b44 210 #ifdef DEBUG
va009039 0:61831b843b44 211 printf(buf);
va009039 0:61831b843b44 212 #endif
va009039 0:61831b843b44 213
va009039 0:61831b843b44 214 // recv dummy
va009039 0:61831b843b44 215 timeout.reset();
va009039 0:61831b843b44 216 timeout.start();
va009039 0:61831b843b44 217 while (timeout.read_ms() < HTTP_TIMEOUT) {
va009039 0:61831b843b44 218 if (tcp_readable) {
va009039 0:61831b843b44 219 i = http->recv(buf, sizeof(buf) - 1);
va009039 0:61831b843b44 220 buf[i] = 0;
va009039 0:61831b843b44 221 if (i < sizeof(buf) - 1) tcp_readable = 0;
va009039 0:61831b843b44 222 #ifdef DEBUG
va009039 0:61831b843b44 223 printf(buf);
va009039 0:61831b843b44 224 #endif
va009039 0:61831b843b44 225 timeout.reset();
va009039 0:61831b843b44 226 } else
va009039 0:61831b843b44 227 if (! tcp_ready) {
va009039 0:61831b843b44 228 break;
va009039 0:61831b843b44 229 }
va009039 0:61831b843b44 230 Net::poll();
va009039 0:61831b843b44 231 }
va009039 0:61831b843b44 232 timeout.stop();
va009039 0:61831b843b44 233
va009039 0:61831b843b44 234 exit:
va009039 0:61831b843b44 235 http->resetOnEvent();
va009039 0:61831b843b44 236 http->close();
va009039 0:61831b843b44 237 delete http;
va009039 0:61831b843b44 238
va009039 0:61831b843b44 239 return ret;
va009039 0:61831b843b44 240 }