switch(c->state) {
case CLOSED:
case LISTEN:
- case SYN_SENT:
- case SYN_RECEIVED:
debug("Error: send() called on unconnected connection %p\n", c);
errno = ENOTCONN;
return -1;
+ case SYN_SENT:
+ case SYN_RECEIVED:
case ESTABLISHED:
case CLOSE_WAIT:
break;
}
c->snd.last += len;
+
+ // Don't send anything yet if the connection has not fully established yet
+
+ if (c->state == SYN_SENT || c->state == SYN_RECEIVED)
+ return len;
+
ack(c, false);
if(!is_reliable(c)) {
c->rcv.irs = hdr.seq;
c->rcv.nxt = hdr.seq;
- set_state(c, ESTABLISHED);
+ if(c->shut_wr) {
+ c->snd.last++;
+ set_state(c, FIN_WAIT_1);
+ } else {
+ set_state(c, ESTABLISHED);
+ }
// TODO: notify application of this somehow.
break;
return 0;
}
+ // Only process shutting down writes once.
+ if (c->shut_wr)
+ return 0;
+
+ c->shut_wr = true;
+
switch(c->state) {
case CLOSED:
case LISTEN:
return -1;
case SYN_SENT:
- set_state(c, CLOSED);
return 0;
case SYN_RECEIVED:
}
size_t utcp_get_sndbuf_free(struct utcp_connection *c) {
- if(c && (c->state == ESTABLISHED || c->state == CLOSE_WAIT)) {
+ if (!c)
+ return 0;
+
+ switch(c->state) {
+ case SYN_SENT:
+ case SYN_RECEIVED:
+ case ESTABLISHED:
+ case CLOSE_WAIT:
return buffer_free(&c->sndbuf);
- } else {
+
+ default:
return 0;
}
}