ZG2100 Network interface source

Committer:
donatien
Date:
Fri Jul 09 15:37:23 2010 +0000
Revision:
0:b802fc31f1db
Child:
1:3a7c15057192

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
donatien 0:b802fc31f1db 1
donatien 0:b802fc31f1db 2 /*
donatien 0:b802fc31f1db 3 Copyright (c) 2010 Donatien Garnier (donatiengar [at] gmail [dot] com)
donatien 0:b802fc31f1db 4
donatien 0:b802fc31f1db 5 Permission is hereby granted, free of charge, to any person obtaining a copy
donatien 0:b802fc31f1db 6 of this software and associated documentation files (the "Software"), to deal
donatien 0:b802fc31f1db 7 in the Software without restriction, including without limitation the rights
donatien 0:b802fc31f1db 8 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
donatien 0:b802fc31f1db 9 copies of the Software, and to permit persons to whom the Software is
donatien 0:b802fc31f1db 10 furnished to do so, subject to the following conditions:
donatien 0:b802fc31f1db 11
donatien 0:b802fc31f1db 12 The above copyright notice and this permission notice shall be included in
donatien 0:b802fc31f1db 13 all copies or substantial portions of the Software.
donatien 0:b802fc31f1db 14
donatien 0:b802fc31f1db 15 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
donatien 0:b802fc31f1db 16 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
donatien 0:b802fc31f1db 17 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
donatien 0:b802fc31f1db 18 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
donatien 0:b802fc31f1db 19 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
donatien 0:b802fc31f1db 20 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
donatien 0:b802fc31f1db 21 THE SOFTWARE.
donatien 0:b802fc31f1db 22 */
donatien 0:b802fc31f1db 23
donatien 0:b802fc31f1db 24 #include "lwipNetTcpSocket.h"
donatien 0:b802fc31f1db 25 #include "lwip/tcp.h"
donatien 0:b802fc31f1db 26
donatien 0:b802fc31f1db 27 //#define __DEBUG
donatien 0:b802fc31f1db 28 #include "dbg/dbg.h"
donatien 0:b802fc31f1db 29
donatien 0:b802fc31f1db 30 #include "netCfg.h"
donatien 0:b802fc31f1db 31 #if NET_LWIP_STACK
donatien 0:b802fc31f1db 32
donatien 0:b802fc31f1db 33 LwipNetTcpSocket::LwipNetTcpSocket(tcp_pcb* pPcb /*= NULL*/) : NetTcpSocket(), m_pPcb(pPcb), m_lpInNetTcpSocket(), //Passes a pcb if already created (by an accept req for instance), in that case transfers ownership
donatien 0:b802fc31f1db 34 m_pReadPbuf(NULL)
donatien 0:b802fc31f1db 35 {
donatien 0:b802fc31f1db 36 DBG("\r\nNew NetTcpSocket %p\r\n", (void*)this);
donatien 0:b802fc31f1db 37 if(!m_pPcb)
donatien 0:b802fc31f1db 38 m_pPcb = tcp_new();
donatien 0:b802fc31f1db 39 if(m_pPcb)
donatien 0:b802fc31f1db 40 {
donatien 0:b802fc31f1db 41 //Setup callbacks
donatien 0:b802fc31f1db 42 tcp_arg( (tcp_pcb*) m_pPcb, (void*) this ); //this will be passed to each static callback
donatien 0:b802fc31f1db 43
donatien 0:b802fc31f1db 44 tcp_recv( (tcp_pcb*) m_pPcb, LwipNetTcpSocket::sRecvCb );
donatien 0:b802fc31f1db 45 tcp_sent((tcp_pcb*) m_pPcb, LwipNetTcpSocket::sSentCb );
donatien 0:b802fc31f1db 46 tcp_err( (tcp_pcb*) m_pPcb, LwipNetTcpSocket::sErrCb );
donatien 0:b802fc31f1db 47 //Connected callback is defined in connect()
donatien 0:b802fc31f1db 48 //Accept callback is defined in listen()
donatien 0:b802fc31f1db 49 DBG("\r\nNetTcpSocket created.\r\n");
donatien 0:b802fc31f1db 50 }
donatien 0:b802fc31f1db 51 }
donatien 0:b802fc31f1db 52
donatien 0:b802fc31f1db 53 LwipNetTcpSocket::~LwipNetTcpSocket()
donatien 0:b802fc31f1db 54 {
donatien 0:b802fc31f1db 55 /* if(m_pPcb)
donatien 0:b802fc31f1db 56 tcp_close( (tcp_pcb*) m_pPcb); //Disconnect & free pcb*/
donatien 0:b802fc31f1db 57 close();
donatien 0:b802fc31f1db 58 }
donatien 0:b802fc31f1db 59
donatien 0:b802fc31f1db 60 NetTcpSocketErr LwipNetTcpSocket::bind(const Host& me)
donatien 0:b802fc31f1db 61 {
donatien 0:b802fc31f1db 62 if(!m_pPcb)
donatien 0:b802fc31f1db 63 return NETTCPSOCKET_MEM; //NetTcpSocket was not properly initialised, should destroy it & retry
donatien 0:b802fc31f1db 64
donatien 0:b802fc31f1db 65 err_t err = tcp_bind( (tcp_pcb*) m_pPcb, IP_ADDR_ANY, me.getPort()); //IP_ADDR_ANY : Bind the connection to all local addresses
donatien 0:b802fc31f1db 66 if(err)
donatien 0:b802fc31f1db 67 return NETTCPSOCKET_INUSE;
donatien 0:b802fc31f1db 68
donatien 0:b802fc31f1db 69 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 70 }
donatien 0:b802fc31f1db 71
donatien 0:b802fc31f1db 72 NetTcpSocketErr LwipNetTcpSocket::listen()
donatien 0:b802fc31f1db 73 {
donatien 0:b802fc31f1db 74 if(!m_pPcb)
donatien 0:b802fc31f1db 75 return NETTCPSOCKET_MEM; //NetTcpSocket was not properly initialised, should destroy it & retry
donatien 0:b802fc31f1db 76 /*
donatien 0:b802fc31f1db 77 From doc/rawapi.txt :
donatien 0:b802fc31f1db 78
donatien 0:b802fc31f1db 79 The tcp_listen() function returns a new connection identifier, and
donatien 0:b802fc31f1db 80 the one passed as an argument to the function will be
donatien 0:b802fc31f1db 81 deallocated. The reason for this behavior is that less memory is
donatien 0:b802fc31f1db 82 needed for a connection that is listening, so tcp_listen() will
donatien 0:b802fc31f1db 83 reclaim the memory needed for the original connection and allocate a
donatien 0:b802fc31f1db 84 new smaller memory block for the listening connection.
donatien 0:b802fc31f1db 85 */
donatien 0:b802fc31f1db 86
donatien 0:b802fc31f1db 87 // tcp_pcb* pNewPcb = tcp_listen(m_pPcb);
donatien 0:b802fc31f1db 88 tcp_pcb* pNewPcb = tcp_listen_with_backlog((tcp_pcb*)m_pPcb, 5);
donatien 0:b802fc31f1db 89 if( !pNewPcb ) //Not enough memory to create the listening pcb
donatien 0:b802fc31f1db 90 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 91
donatien 0:b802fc31f1db 92 m_pPcb = pNewPcb;
donatien 0:b802fc31f1db 93
donatien 0:b802fc31f1db 94 tcp_accept( (tcp_pcb*) m_pPcb, LwipNetTcpSocket::sAcceptCb );
donatien 0:b802fc31f1db 95
donatien 0:b802fc31f1db 96 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 97 }
donatien 0:b802fc31f1db 98
donatien 0:b802fc31f1db 99 NetTcpSocketErr LwipNetTcpSocket::connect(const Host& host)
donatien 0:b802fc31f1db 100 {
donatien 0:b802fc31f1db 101 if(!m_pPcb)
donatien 0:b802fc31f1db 102 return NETTCPSOCKET_MEM; //NetTcpSocket was not properly initialised, should destroy it & retry
donatien 0:b802fc31f1db 103
donatien 0:b802fc31f1db 104 ip_addr_t ip = host.getIp().getStruct();
donatien 0:b802fc31f1db 105 err_t err = tcp_connect( (tcp_pcb*) m_pPcb, &ip, host.getPort(), LwipNetTcpSocket::sConnectedCb );
donatien 0:b802fc31f1db 106
donatien 0:b802fc31f1db 107 if(!err)
donatien 0:b802fc31f1db 108 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 109
donatien 0:b802fc31f1db 110 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 111 }
donatien 0:b802fc31f1db 112
donatien 0:b802fc31f1db 113 NetTcpSocketErr LwipNetTcpSocket::accept(Host* pClient, NetTcpSocket** ppNewNetTcpSocket)
donatien 0:b802fc31f1db 114 {
donatien 0:b802fc31f1db 115 if( !m_pPcb ) //Pcb doesn't exist (anymore)
donatien 0:b802fc31f1db 116 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 117 //Dequeue a connection
donatien 0:b802fc31f1db 118 //if( m_lpInPcb.empty() )
donatien 0:b802fc31f1db 119 if( m_lpInNetTcpSocket.empty() )
donatien 0:b802fc31f1db 120 return NETTCPSOCKET_EMPTY;
donatien 0:b802fc31f1db 121
donatien 0:b802fc31f1db 122 tcp_accepted( ((tcp_pcb*) m_pPcb) ); //Should fire proper events //WARN: m_pPcb is the GOOD param here (and not pInPcb)
donatien 0:b802fc31f1db 123
donatien 0:b802fc31f1db 124 /* tcp_pcb* pInPcb = m_lpInPcb.front();
donatien 0:b802fc31f1db 125 m_lpInPcb.pop();*/
donatien 0:b802fc31f1db 126
donatien 0:b802fc31f1db 127 if( (m_lpInNetTcpSocket.front()) == NULL )
donatien 0:b802fc31f1db 128 {
donatien 0:b802fc31f1db 129 m_lpInNetTcpSocket.pop();
donatien 0:b802fc31f1db 130 return NETTCPSOCKET_RST;
donatien 0:b802fc31f1db 131 }
donatien 0:b802fc31f1db 132
donatien 0:b802fc31f1db 133 if( (m_lpInNetTcpSocket.front())->m_closed )
donatien 0:b802fc31f1db 134 {
donatien 0:b802fc31f1db 135 Net::releaseTcpSocket(m_lpInNetTcpSocket.front());
donatien 0:b802fc31f1db 136 m_lpInNetTcpSocket.pop();
donatien 0:b802fc31f1db 137 return NETTCPSOCKET_RST;
donatien 0:b802fc31f1db 138 }
donatien 0:b802fc31f1db 139
donatien 0:b802fc31f1db 140 ip_addr_t* ip = (ip_addr_t*) &( (m_lpInNetTcpSocket.front()->m_pPcb)->remote_ip);
donatien 0:b802fc31f1db 141
donatien 0:b802fc31f1db 142 *ppNewNetTcpSocket = m_lpInNetTcpSocket.front();
donatien 0:b802fc31f1db 143 *pClient = Host(
donatien 0:b802fc31f1db 144 IpAddr(
donatien 0:b802fc31f1db 145 ip
donatien 0:b802fc31f1db 146 ),
donatien 0:b802fc31f1db 147 m_lpInNetTcpSocket.front()->m_pPcb->remote_port
donatien 0:b802fc31f1db 148 );
donatien 0:b802fc31f1db 149 m_lpInNetTcpSocket.pop();
donatien 0:b802fc31f1db 150 // *pClient = Host( IpAddr(pInPcb->remote_ip), pInPcb->remote_port );
donatien 0:b802fc31f1db 151
donatien 0:b802fc31f1db 152 //Return a new socket
donatien 0:b802fc31f1db 153 // *ppNewNetTcpSocket = (NetTcpSocket*) new LwipNetTcpSocket(pInPcb);
donatien 0:b802fc31f1db 154
donatien 0:b802fc31f1db 155 //tcp_accepted( ((tcp_pcb*) m_pPcb) ); //Should fire proper events //WARN: m_pPcb is the GOOD param here (and not pInPcb)
donatien 0:b802fc31f1db 156
donatien 0:b802fc31f1db 157 /* if(*ppNewNetTcpSocket == NULL)
donatien 0:b802fc31f1db 158 {
donatien 0:b802fc31f1db 159 DBG("\r\nNot enough mem, socket dropped in LwipNetTcpSocket::accept.\r\n");
donatien 0:b802fc31f1db 160 tcp_abort(pInPcb);
donatien 0:b802fc31f1db 161 }*/
donatien 0:b802fc31f1db 162
donatien 0:b802fc31f1db 163 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 164 }
donatien 0:b802fc31f1db 165
donatien 0:b802fc31f1db 166 #define MAX(a,b) (((a)>(b))?(a):(b))
donatien 0:b802fc31f1db 167 #define MIN(a,b) (((a)<(b))?(a):(b))
donatien 0:b802fc31f1db 168
donatien 0:b802fc31f1db 169 int /*if < 0 : NetTcpSocketErr*/ LwipNetTcpSocket::send(const char* buf, int len)
donatien 0:b802fc31f1db 170 {
donatien 0:b802fc31f1db 171 if( !m_pPcb ) //Pcb doesn't exist (anymore)
donatien 0:b802fc31f1db 172 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 173 int outLen = MIN( len, tcp_sndbuf( (tcp_pcb*) m_pPcb) );
donatien 0:b802fc31f1db 174 //tcp_sndbuf() returns the number of bytes available in the output queue, so never go above it
donatien 0:b802fc31f1db 175 err_t err = tcp_write( (tcp_pcb*) m_pPcb, (void*) buf, outLen, TCP_WRITE_FLAG_COPY );
donatien 0:b802fc31f1db 176 //Flags are TCP_WRITE_FLAG_COPY & TCP_WRITE_FLAG_MORE (see tcp_out.c) :
donatien 0:b802fc31f1db 177 //If TCP_WRITE_FLAG_MORE is not set ask client to push buffered data to app
donatien 0:b802fc31f1db 178 if(err)
donatien 0:b802fc31f1db 179 {
donatien 0:b802fc31f1db 180 switch( err )
donatien 0:b802fc31f1db 181 {
donatien 0:b802fc31f1db 182 case ERR_CONN:
donatien 0:b802fc31f1db 183 return (int) NETTCPSOCKET_SETUP; //Not connected properly
donatien 0:b802fc31f1db 184 case ERR_ARG:
donatien 0:b802fc31f1db 185 return (int) NETTCPSOCKET_SETUP; //Wrong args ! (like buf pointing to NULL)
donatien 0:b802fc31f1db 186 case ERR_MEM:
donatien 0:b802fc31f1db 187 default:
donatien 0:b802fc31f1db 188 return (int) NETTCPSOCKET_MEM; //Not enough memory
donatien 0:b802fc31f1db 189 }
donatien 0:b802fc31f1db 190 }
donatien 0:b802fc31f1db 191 return outLen;
donatien 0:b802fc31f1db 192 }
donatien 0:b802fc31f1db 193
donatien 0:b802fc31f1db 194 int /*if < 0 : NetTcpSocketErr*/ LwipNetTcpSocket::recv(char* buf, int len)
donatien 0:b802fc31f1db 195 {
donatien 0:b802fc31f1db 196 if( !m_pPcb ) //Pcb doesn't exist (anymore)
donatien 0:b802fc31f1db 197 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 198 int inLen = 0;
donatien 0:b802fc31f1db 199 int cpyLen = 0;
donatien 0:b802fc31f1db 200
donatien 0:b802fc31f1db 201 static int rmgLen = 0;
donatien 0:b802fc31f1db 202 //Contains the remaining len in this pbuf
donatien 0:b802fc31f1db 203
donatien 0:b802fc31f1db 204 if( !m_pReadPbuf )
donatien 0:b802fc31f1db 205 {
donatien 0:b802fc31f1db 206 rmgLen = 0;
donatien 0:b802fc31f1db 207 return 0;
donatien 0:b802fc31f1db 208 }
donatien 0:b802fc31f1db 209
donatien 0:b802fc31f1db 210 if ( !rmgLen ) //We did not know m_pReadPbuf->len last time we called this fn
donatien 0:b802fc31f1db 211 {
donatien 0:b802fc31f1db 212 rmgLen = m_pReadPbuf->len;
donatien 0:b802fc31f1db 213 }
donatien 0:b802fc31f1db 214
donatien 0:b802fc31f1db 215 while ( inLen < len )
donatien 0:b802fc31f1db 216 {
donatien 0:b802fc31f1db 217 cpyLen = MIN( (len - inLen), rmgLen ); //Remaining len to copy, remaining len in THIS pbuf
donatien 0:b802fc31f1db 218 memcpy((void*)buf, (void*)((char*)(m_pReadPbuf->payload) + (m_pReadPbuf->len - rmgLen)), cpyLen);
donatien 0:b802fc31f1db 219 inLen += cpyLen;
donatien 0:b802fc31f1db 220 buf += cpyLen;
donatien 0:b802fc31f1db 221
donatien 0:b802fc31f1db 222 rmgLen = rmgLen - cpyLen; //Update rmgLen
donatien 0:b802fc31f1db 223
donatien 0:b802fc31f1db 224 if( rmgLen > 0 )
donatien 0:b802fc31f1db 225 {
donatien 0:b802fc31f1db 226 //We did not read this pbuf completely, so let's save it's pos & return
donatien 0:b802fc31f1db 227 break;
donatien 0:b802fc31f1db 228 }
donatien 0:b802fc31f1db 229
donatien 0:b802fc31f1db 230 if(m_pReadPbuf->next)
donatien 0:b802fc31f1db 231 {
donatien 0:b802fc31f1db 232 pbuf* pNextPBuf = m_pReadPbuf->next;
donatien 0:b802fc31f1db 233 m_pReadPbuf->next = NULL; //So that it is not freed as well
donatien 0:b802fc31f1db 234 //We get the reference to pNextPBuf from m_pReadPbuf
donatien 0:b802fc31f1db 235 pbuf_free((pbuf*)m_pReadPbuf);
donatien 0:b802fc31f1db 236 m_pReadPbuf = pNextPBuf;
donatien 0:b802fc31f1db 237 rmgLen = m_pReadPbuf->len;
donatien 0:b802fc31f1db 238 }
donatien 0:b802fc31f1db 239 else
donatien 0:b802fc31f1db 240 {
donatien 0:b802fc31f1db 241 pbuf_free((pbuf*)m_pReadPbuf);
donatien 0:b802fc31f1db 242 m_pReadPbuf = NULL;
donatien 0:b802fc31f1db 243 rmgLen = 0;
donatien 0:b802fc31f1db 244 break; //No more data to read
donatien 0:b802fc31f1db 245 }
donatien 0:b802fc31f1db 246
donatien 0:b802fc31f1db 247 }
donatien 0:b802fc31f1db 248
donatien 0:b802fc31f1db 249 //tcp_recved(m_pPcb, inLen); //Acknowledge the reception
donatien 0:b802fc31f1db 250
donatien 0:b802fc31f1db 251 return inLen;
donatien 0:b802fc31f1db 252 }
donatien 0:b802fc31f1db 253
donatien 0:b802fc31f1db 254 NetTcpSocketErr LwipNetTcpSocket::close()
donatien 0:b802fc31f1db 255 {
donatien 0:b802fc31f1db 256 //DBG("\r\nLwipNetTcpSocket::close() : Closing...\r\n");
donatien 0:b802fc31f1db 257
donatien 0:b802fc31f1db 258 if(m_closed)
donatien 0:b802fc31f1db 259 return NETTCPSOCKET_OK; //Already being closed
donatien 0:b802fc31f1db 260 m_closed = true;
donatien 0:b802fc31f1db 261
donatien 0:b802fc31f1db 262 if( !m_pPcb ) //Pcb doesn't exist (anymore)
donatien 0:b802fc31f1db 263 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 264
donatien 0:b802fc31f1db 265 //Cleanup incoming data
donatien 0:b802fc31f1db 266 cleanUp();
donatien 0:b802fc31f1db 267
donatien 0:b802fc31f1db 268 if( !!tcp_close( (tcp_pcb*) m_pPcb) )
donatien 0:b802fc31f1db 269 {
donatien 0:b802fc31f1db 270 DBG("\r\nLwipNetTcpSocket::close() could not close properly, abort.\r\n");
donatien 0:b802fc31f1db 271 tcp_abort( (tcp_pcb*) m_pPcb);
donatien 0:b802fc31f1db 272 m_pPcb = NULL;
donatien 0:b802fc31f1db 273 return NETTCPSOCKET_MEM;
donatien 0:b802fc31f1db 274 }
donatien 0:b802fc31f1db 275
donatien 0:b802fc31f1db 276 DBG("\r\nLwipNetTcpSocket::close() : connection closed successfully.\r\n");
donatien 0:b802fc31f1db 277
donatien 0:b802fc31f1db 278 m_pPcb = NULL;
donatien 0:b802fc31f1db 279 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 280 }
donatien 0:b802fc31f1db 281
donatien 0:b802fc31f1db 282 NetTcpSocketErr LwipNetTcpSocket::poll()
donatien 0:b802fc31f1db 283 {
donatien 0:b802fc31f1db 284 NetTcpSocket::flushEvents();
donatien 0:b802fc31f1db 285 return NETTCPSOCKET_OK;
donatien 0:b802fc31f1db 286 }
donatien 0:b802fc31f1db 287
donatien 0:b802fc31f1db 288 // Callbacks events
donatien 0:b802fc31f1db 289
donatien 0:b802fc31f1db 290 err_t LwipNetTcpSocket::acceptCb(struct tcp_pcb *newpcb, err_t err)
donatien 0:b802fc31f1db 291 {
donatien 0:b802fc31f1db 292 if(err)
donatien 0:b802fc31f1db 293 {
donatien 0:b802fc31f1db 294 DBG("\r\nError %d in LwipNetTcpSocket::acceptCb.\r\n", err);
donatien 0:b802fc31f1db 295 return err;
donatien 0:b802fc31f1db 296 }
donatien 0:b802fc31f1db 297 //FIXME: MEM Errs
donatien 0:b802fc31f1db 298 //m_lpInPcb.push(newpcb); //Add connection to the queue
donatien 0:b802fc31f1db 299 LwipNetTcpSocket* pNewNetTcpSocket = new LwipNetTcpSocket(newpcb);
donatien 0:b802fc31f1db 300
donatien 0:b802fc31f1db 301 if(pNewNetTcpSocket == NULL)
donatien 0:b802fc31f1db 302 {
donatien 0:b802fc31f1db 303 DBG("\r\nNot enough mem, socket dropped in LwipNetTcpSocket::acceptCb.\r\n");
donatien 0:b802fc31f1db 304 tcp_abort(newpcb);
donatien 0:b802fc31f1db 305 return ERR_ABRT;
donatien 0:b802fc31f1db 306 }
donatien 0:b802fc31f1db 307
donatien 0:b802fc31f1db 308 pNewNetTcpSocket->m_refs++;
donatien 0:b802fc31f1db 309 m_lpInNetTcpSocket.push( pNewNetTcpSocket );
donatien 0:b802fc31f1db 310
donatien 0:b802fc31f1db 311 // tcp_accepted(newpcb);
donatien 0:b802fc31f1db 312 // tcp_accepted( m_pPcb ); //Should fire proper events //WARN: m_pPcb is the GOOD param here (and not pInPcb)
donatien 0:b802fc31f1db 313 queueEvent(NETTCPSOCKET_ACCEPT);
donatien 0:b802fc31f1db 314 return ERR_OK;
donatien 0:b802fc31f1db 315 }
donatien 0:b802fc31f1db 316
donatien 0:b802fc31f1db 317 err_t LwipNetTcpSocket::connectedCb(struct tcp_pcb *tpcb, err_t err)
donatien 0:b802fc31f1db 318 {
donatien 0:b802fc31f1db 319 queueEvent(NETTCPSOCKET_CONNECTED);
donatien 0:b802fc31f1db 320 return ERR_OK;
donatien 0:b802fc31f1db 321 }
donatien 0:b802fc31f1db 322
donatien 0:b802fc31f1db 323 void LwipNetTcpSocket::errCb(err_t err)
donatien 0:b802fc31f1db 324 {
donatien 0:b802fc31f1db 325 DBG("\r\nNetTcpSocket %p - Error %d in LwipNetTcpSocket::errCb.\r\n", (void*)this, err);
donatien 0:b802fc31f1db 326 //WARN: At this point, m_pPcb has been freed by lwIP
donatien 0:b802fc31f1db 327 m_pPcb = NULL;
donatien 0:b802fc31f1db 328 //These errors are fatal, discard all events queued before so that the errors are handled first
donatien 0:b802fc31f1db 329 discardEvents();
donatien 0:b802fc31f1db 330 m_closed = true;
donatien 0:b802fc31f1db 331 cleanUp();
donatien 0:b802fc31f1db 332 if( err == ERR_ABRT)
donatien 0:b802fc31f1db 333 queueEvent(NETTCPSOCKET_CONABRT);
donatien 0:b802fc31f1db 334 else //if( err == ERR_RST)
donatien 0:b802fc31f1db 335 queueEvent(NETTCPSOCKET_CONRST);
donatien 0:b802fc31f1db 336 }
donatien 0:b802fc31f1db 337
donatien 0:b802fc31f1db 338 err_t LwipNetTcpSocket::sentCb(tcp_pcb* tpcb, u16_t len)
donatien 0:b802fc31f1db 339 {
donatien 0:b802fc31f1db 340 // DBG("\r\n%d bytes ACKed by host.\r\n", len);
donatien 0:b802fc31f1db 341 queueEvent(NETTCPSOCKET_WRITEABLE);
donatien 0:b802fc31f1db 342 return ERR_OK;
donatien 0:b802fc31f1db 343 }
donatien 0:b802fc31f1db 344
donatien 0:b802fc31f1db 345 err_t LwipNetTcpSocket::recvCb(tcp_pcb* tpcb, pbuf *p, err_t err)
donatien 0:b802fc31f1db 346 {
donatien 0:b802fc31f1db 347 //Store pbuf ptr
donatien 0:b802fc31f1db 348 // DBG("\r\nReceive CB with err = %d & len = %d.\r\n", err, p->tot_len);
donatien 0:b802fc31f1db 349 // tcp_recved( (tcp_pcb*) m_pPcb, p->tot_len); //Acknowledge the reception
donatien 0:b802fc31f1db 350
donatien 0:b802fc31f1db 351 if(err)
donatien 0:b802fc31f1db 352 {
donatien 0:b802fc31f1db 353 queueEvent(NETTCPSOCKET_ERROR);
donatien 0:b802fc31f1db 354 return ERR_OK; //FIXME: More robust error handling there
donatien 0:b802fc31f1db 355 }
donatien 0:b802fc31f1db 356 else if(!p)
donatien 0:b802fc31f1db 357 {
donatien 0:b802fc31f1db 358 DBG("\r\nNetTcpSocket %p - Connection closed by remote host (LwipNetTcpSocket::recvCb).\r\n", (void*)this);
donatien 0:b802fc31f1db 359 //Buf is NULL, that means that the connection has been closed by remote host
donatien 0:b802fc31f1db 360
donatien 0:b802fc31f1db 361 //FIX: 27/05/2010: We do not want to deallocate the socket while some data might still be readable
donatien 0:b802fc31f1db 362 //REMOVED: close();
donatien 0:b802fc31f1db 363
donatien 0:b802fc31f1db 364 //However we do not want to close the socket yet
donatien 0:b802fc31f1db 365
donatien 0:b802fc31f1db 366 queueEvent(NETTCPSOCKET_DISCONNECTED);
donatien 0:b802fc31f1db 367 return ERR_OK;
donatien 0:b802fc31f1db 368 }
donatien 0:b802fc31f1db 369
donatien 0:b802fc31f1db 370 //We asserted that p is a valid pointer
donatien 0:b802fc31f1db 371
donatien 0:b802fc31f1db 372 //New data processing
donatien 0:b802fc31f1db 373 tcp_recved( tpcb, p->tot_len); //Acknowledge the reception
donatien 0:b802fc31f1db 374 if(!m_pReadPbuf)
donatien 0:b802fc31f1db 375 {
donatien 0:b802fc31f1db 376 m_pReadPbuf = p;
donatien 0:b802fc31f1db 377 queueEvent(NETTCPSOCKET_READABLE);
donatien 0:b802fc31f1db 378 }
donatien 0:b802fc31f1db 379 else
donatien 0:b802fc31f1db 380 {
donatien 0:b802fc31f1db 381 pbuf_cat((pbuf*)m_pReadPbuf, p); //m_pReadPbuf is not empty, tail p to it and drop our ref
donatien 0:b802fc31f1db 382 //No need to queue an event in that case since the read buf has not been processed yet
donatien 0:b802fc31f1db 383 }
donatien 0:b802fc31f1db 384 return ERR_OK;
donatien 0:b802fc31f1db 385 }
donatien 0:b802fc31f1db 386
donatien 0:b802fc31f1db 387
donatien 0:b802fc31f1db 388 void LwipNetTcpSocket::cleanUp() //Flush input buffer
donatien 0:b802fc31f1db 389 {
donatien 0:b802fc31f1db 390 //Ensure that further error won't be followed to this inst (which can be destroyed)
donatien 0:b802fc31f1db 391 if( m_pPcb )
donatien 0:b802fc31f1db 392 {
donatien 0:b802fc31f1db 393 tcp_arg( (tcp_pcb*) m_pPcb, (void*) NULL );
donatien 0:b802fc31f1db 394 tcp_recv( (tcp_pcb*) m_pPcb, NULL );
donatien 0:b802fc31f1db 395 tcp_sent((tcp_pcb*) m_pPcb, NULL );
donatien 0:b802fc31f1db 396 tcp_err( (tcp_pcb*) m_pPcb, NULL );
donatien 0:b802fc31f1db 397 }
donatien 0:b802fc31f1db 398
donatien 0:b802fc31f1db 399 if( m_pReadPbuf )
donatien 0:b802fc31f1db 400 {
donatien 0:b802fc31f1db 401 DBG("\r\nDeallocating unread data.\r\n");
donatien 0:b802fc31f1db 402 pbuf_free((pbuf*)m_pReadPbuf); //Free all unread data
donatien 0:b802fc31f1db 403 m_pReadPbuf = NULL;
donatien 0:b802fc31f1db 404 recv(NULL,0); //Update recv ptr position
donatien 0:b802fc31f1db 405 }
donatien 0:b802fc31f1db 406 }
donatien 0:b802fc31f1db 407
donatien 0:b802fc31f1db 408 // Static callbacks from LwIp
donatien 0:b802fc31f1db 409
donatien 0:b802fc31f1db 410 err_t LwipNetTcpSocket::sAcceptCb(void *arg, struct tcp_pcb *newpcb, err_t err)
donatien 0:b802fc31f1db 411 {
donatien 0:b802fc31f1db 412 LwipNetTcpSocket* pMe = (LwipNetTcpSocket*) arg;
donatien 0:b802fc31f1db 413 return pMe->acceptCb( newpcb, err );
donatien 0:b802fc31f1db 414 }
donatien 0:b802fc31f1db 415
donatien 0:b802fc31f1db 416 err_t LwipNetTcpSocket::sConnectedCb(void *arg, struct tcp_pcb *tpcb, err_t err)
donatien 0:b802fc31f1db 417 {
donatien 0:b802fc31f1db 418 LwipNetTcpSocket* pMe = (LwipNetTcpSocket*) arg;
donatien 0:b802fc31f1db 419 return pMe->connectedCb( tpcb, err );
donatien 0:b802fc31f1db 420 }
donatien 0:b802fc31f1db 421
donatien 0:b802fc31f1db 422 void LwipNetTcpSocket::sErrCb(void *arg, err_t err)
donatien 0:b802fc31f1db 423 {
donatien 0:b802fc31f1db 424 if( !arg )
donatien 0:b802fc31f1db 425 {
donatien 0:b802fc31f1db 426 DBG("\r\nNetTcpSocket - Error %d in LwipNetTcpSocket::sErrCb.\r\n", err);
donatien 0:b802fc31f1db 427 return; //The socket has been destroyed, discard error
donatien 0:b802fc31f1db 428 }
donatien 0:b802fc31f1db 429 LwipNetTcpSocket* pMe = (LwipNetTcpSocket*) arg;
donatien 0:b802fc31f1db 430 return pMe->errCb( err );
donatien 0:b802fc31f1db 431 }
donatien 0:b802fc31f1db 432
donatien 0:b802fc31f1db 433 err_t LwipNetTcpSocket::sSentCb(void *arg, struct tcp_pcb *tpcb, u16_t len)
donatien 0:b802fc31f1db 434 {
donatien 0:b802fc31f1db 435 LwipNetTcpSocket* pMe = (LwipNetTcpSocket*) arg;
donatien 0:b802fc31f1db 436 return pMe->sentCb( tpcb, len );
donatien 0:b802fc31f1db 437 }
donatien 0:b802fc31f1db 438
donatien 0:b802fc31f1db 439 err_t LwipNetTcpSocket::sRecvCb(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
donatien 0:b802fc31f1db 440 {
donatien 0:b802fc31f1db 441 if( tpcb->flags & TF_RXCLOSED )
donatien 0:b802fc31f1db 442 {
donatien 0:b802fc31f1db 443 //The Pcb is in a closing state
donatien 0:b802fc31f1db 444 //Discard that data here since we might have destroyed the corresponding socket object
donatien 0:b802fc31f1db 445 tcp_recved( tpcb, p->tot_len);
donatien 0:b802fc31f1db 446 pbuf_free( p );
donatien 0:b802fc31f1db 447 return ERR_OK;
donatien 0:b802fc31f1db 448 }
donatien 0:b802fc31f1db 449 LwipNetTcpSocket* pMe = (LwipNetTcpSocket*) arg;
donatien 0:b802fc31f1db 450 return pMe->recvCb( tpcb, p, err );
donatien 0:b802fc31f1db 451 }
donatien 0:b802fc31f1db 452
donatien 0:b802fc31f1db 453 #endif