X-Git-Url: http://andersk.mit.edu/gitweb/libfaim.git/blobdiff_plain/c78446b536c54db4f1bbe83effad0270d9e0fbfc..313a06b7b4be217e90a17413862dc04552f693c3:/aim_rxqueue.c diff --git a/aim_rxqueue.c b/aim_rxqueue.c index 9d8dd7f..a26648a 100644 --- a/aim_rxqueue.c +++ b/aim_rxqueue.c @@ -8,11 +8,48 @@ #include +#ifndef _WIN32 +#include +#endif + +/* + * Since not all implementations support MSG_WAITALL, define + * an alternate guarenteed read function... + * + * We keep recv() for systems that can do it because it means + * a single system call for the entire packet, where read may + * take more for a badly fragmented packet. + * + */ +faim_internal int aim_recv(int fd, void *buf, size_t count) +{ +#ifdef MSG_WAITALL + return recv(fd, buf, count, MSG_WAITALL); +#else + int left, ret, cur = 0; + + left = count; + + while (left) { + ret = recv(fd, ((unsigned char *)buf)+cur, left, 0); + if (ret == -1) + return -1; + if (ret == 0) + return cur; + + cur += ret; + left -= ret; + } + + return cur; +#endif +} + /* * Grab a single command sequence off the socket, and enqueue * it in the incoming event queue in a seperate struct. */ -int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) +faim_export int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) { unsigned char generic[6]; struct command_rx_struct *newrx = NULL; @@ -23,6 +60,9 @@ int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) if (conn->fd < 3) /* can happen when people abuse the interface */ return 0; + if (conn->status & AIM_CONN_STATUS_INPROGRESS) + return aim_conn_completeconnect(sess, conn); + /* * Rendezvous (client-client) connections do not speak * FLAP, so this function will break on them. @@ -41,8 +81,7 @@ int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) * 4 short -- Number of data bytes that follow. */ faim_mutex_lock(&conn->active); - if (recv(conn->fd, generic, 6, MSG_WAITALL) < 6){ - printf("faim: flap: read underflow (header)\n"); + if (aim_recv(conn->fd, generic, 6) < 6){ aim_conn_close(conn); faim_mutex_unlock(&conn->active); return -1; @@ -90,8 +129,7 @@ int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) } /* read the data portion of the packet */ - if (recv(conn->fd, newrx->data, newrx->commandlen, MSG_WAITALL) < newrx->commandlen){ - printf("faim: flap: read underflow (payload)\n"); + if (aim_recv(conn->fd, newrx->data, newrx->commandlen) < newrx->commandlen){ free(newrx->data); free(newrx); aim_conn_close(conn); @@ -138,7 +176,7 @@ int aim_get_command(struct aim_session_t *sess, struct aim_conn_t *conn) * does not keep a pointer, it's lost forever. * */ -void aim_purge_rxqueue(struct aim_session_t *sess) +faim_export void aim_purge_rxqueue(struct aim_session_t *sess) { struct command_rx_struct *cur = NULL; struct command_rx_struct *tmp; @@ -196,7 +234,7 @@ void aim_purge_rxqueue(struct aim_session_t *sess) * XXX: this is something that was handled better in the old connection * handling method, but eh. */ -void aim_rxqueue_cleanbyconn(struct aim_session_t *sess, struct aim_conn_t *conn) +faim_internal void aim_rxqueue_cleanbyconn(struct aim_session_t *sess, struct aim_conn_t *conn) { struct command_rx_struct *currx;