Improve sendq to always send big packets for better network performance

This commit is contained in:
mpc
2004-07-28 04:48:35 +00:00
committed by zzz
parent 5eaec4c841
commit a1ff325b7b
2 changed files with 81 additions and 46 deletions

View File

@@ -34,18 +34,28 @@
extern "C" {
#endif
#include <stdbool.h>
#include <stdint.h>
/*
* Lengths
*/
#define SAM_CMD_LEN 128 /* the maximum length a SAM command can be */
#define SAM_DGRAM_PAYLOAD_MAX ((31 * 1024) - 30) /* max size of a single datagram packet (-30 temporary bug fix for SAM) */
#define SAM_ERRMSG_LEN 23 /* the longest message returned from sam_strerror */
#define SAM_LOGMSG_LEN 256 /* the longest log message */
#define SAM_NAME_LEN 256 /* the longest `name' arg for naming lookup callback*/
#define SAM_STREAM_PAYLOAD_MAX 32768 /* max size of a single stream packet */
#define SAM_PKCMD_LEN (SAM_PUBKEY_LEN + SAM_CMD_LEN)/*a public key SAM command*/
#define SAM_PUBKEY_LEN 517 /* it's actually 516, but +1 for '\0' */
#define SAM_REPLY_LEN 1024 /* the maximum length a SAM non-data reply can be */
/* The maximum length a SAM command can be */
#define SAM_CMD_LEN 128
/*The maximum size of a single datagram packet (-30 temporary bug fix for SAM)*/
#define SAM_DGRAM_PAYLOAD_MAX ((31 * 1024) - 30)
/* The longest log message */
#define SAM_LOGMSG_LEN 256
/* The longest `name' arg for the naming lookup callback */
#define SAM_NAME_LEN 256
/* The max size of a single stream packet */
#define SAM_STREAM_PAYLOAD_MAX (32 * 1024)
/* The length of a base 64 public key - it's actually 516, but +1 for '\0' */
#define SAM_PUBKEY_LEN 517
/* A public key SAM command's length */
#define SAM_PKCMD_LEN (SAM_PUBKEY_LEN + SAM_CMD_LEN)
/* The maximum length a SAM non-data reply can be */
#define SAM_REPLY_LEN 1024
/*
* Shorten some standard variable types
@@ -81,14 +91,14 @@ typedef enum { /* see sam_strerror() for detailed descriptions of these */
*/
/* SAM controls */
extern bool sam_close(void);
extern bool sam_close();
extern samerr_t sam_connect(const char *samhost, uint16_t samport,
const char *destname, sam_conn_t style, uint_t tunneldepth);
extern void sam_naming_lookup(const char *name);
extern bool sam_read_buffer(void);
extern bool sam_read_buffer();
extern const char *sam_strerror(samerr_t code);
/* SAM controls - callbacks */
extern void (*sam_diedback)(void);
extern void (*sam_diedback)();
extern void (*sam_logback)(char *str);
extern void (*sam_namingback)(char *name, sam_pubkey_t pubkey,
samerr_t result);
@@ -105,10 +115,9 @@ extern void (*sam_databack)(sam_sid_t stream_id, void *data, size_t size);
extern void (*sam_statusback)(sam_sid_t stream_id, samerr_t result);
/* Stream send queue */
extern samerr_t sam_sendq_add(sam_sendq_t *sendq, const void *data,
size_t dsize);
extern sam_sendq_t *sam_sendq_create(void);
extern void sam_sendq_send(sam_sendq_t *sendq, sam_sid_t stream_id);
extern void sam_sendq_add(sam_sid_t stream_id, sam_sendq_t **sendq,
const void *data, size_t dsize);
extern void sam_sendq_flush(sam_sid_t stream_id, sam_sendq_t **sendq);
/* Datagram commands */
extern samerr_t sam_dgram_send(const sam_pubkey_t dest, const void *data,

View File

@@ -31,19 +31,20 @@
#include "platform.h"
#include "sam.h"
static bool sam_hello(void);
static bool sam_hello();
static void sam_log(const char *format, ...);
static void sam_parse(char *s);
static ssize_t sam_read1(char *buf, size_t n);
static ssize_t sam_read2(void *buf, size_t n);
static bool sam_readable(void);
static bool sam_readable();
static sam_sendq_t *sam_sendq_create();
static samerr_t sam_session_create(const char *destname, sam_conn_t style,
uint_t tunneldepth);
static bool sam_socket_connect(const char *host, uint16_t port);
static bool sam_socket_resolve(const char *hostname, char *ipaddr);
#ifdef WINSOCK
static samerr_t sam_winsock_cleanup(void);
static samerr_t sam_winsock_startup(void);
static samerr_t sam_winsock_cleanup();
static samerr_t sam_winsock_startup();
static const char *sam_winsock_strerror(int code);
#endif
static ssize_t sam_write(const void *buf, size_t n);
@@ -61,7 +62,7 @@ void (*sam_databack)(sam_sid_t stream_id, void *data, size_t size) = NULL;
/* a peer sent some datagram data (`data' MUST be freed) */
void (*sam_dgramback)(sam_pubkey_t dest, void *data, size_t size) = NULL;
/* we lost the connection to the SAM host */
void (*sam_diedback)(void) = NULL;
void (*sam_diedback)() = NULL;
/* logging callback */
void (*sam_logback)(char *str) = NULL;
/* naming lookup reply - `pubkey' will be NULL if `result' isn't SAM_OK */
@@ -79,7 +80,7 @@ static bool samd_connected = false; /* Whether we're connected with SAM */
*
* Returns: true on success, false on failure
*/
bool sam_close(void)
bool sam_close()
{
if (!samd_connected)
return true;
@@ -212,7 +213,7 @@ samerr_t sam_dgram_send(const sam_pubkey_t dest, const void *data, size_t size)
*
* Returns: true on success, false on reply failure
*/
static bool sam_hello(void)
static bool sam_hello()
{
#define SAM_HELLO_CMD "HELLO VERSION MIN=1.0 MAX=1.0\n"
#define SAM_HELLO_REPLY "HELLO REPLY RESULT=OK VERSION=1.0"
@@ -491,7 +492,7 @@ static void sam_parse(char *s)
*
* Returns: true if we read anything, or false if nothing was there
*/
bool sam_read_buffer(void)
bool sam_read_buffer()
{
bool read_something = false;
char reply[SAM_REPLY_LEN];
@@ -639,7 +640,7 @@ static ssize_t sam_read2(void *buf, size_t n)
*
* Returns: true if data is waiting, false otherwise
*/
static bool sam_readable(void)
static bool sam_readable()
{
fd_set rset; /* set of readable descriptors */
struct timeval tv;
@@ -679,36 +680,58 @@ static bool sam_readable(void)
*
* Returns: true on success, false on error
*/
samerr_t sam_sendq_add(sam_sendq_t *sendq, const void *data, size_t dsize)
void sam_sendq_add(sam_sid_t stream_id, sam_sendq_t **sendq, const void *data,
size_t dsize)
{
assert(dsize >= 0);
if (dsize == 0) {
SAMLOGS("dsize is 0 - adding nothing");
return SAM_OK;
} else if (sendq->size + dsize > SAM_STREAM_PAYLOAD_MAX) {
SAMLOGS("The queue size would exceed the maximum SAM payload size -" \
" will not add to the queue");
return SAM_TOO_BIG;
SAMLOGS("dsize is 0 - doing nothing");
return;
}
sendq->data = realloc(sendq->data, sendq->size + dsize);
memcpy(sendq->data + sendq->size, data, dsize);
sendq->size += dsize;
/* if the sendq pointer is set to NULL, create a sendq */
if (*sendq == NULL)
*sendq = sam_sendq_create();
return SAM_OK;
/* the added data doesn't fill the queue - add but don't send */
if ((*sendq)->size + dsize < SAM_STREAM_PAYLOAD_MAX) {
memcpy((*sendq)->data + (*sendq)->size, data, dsize);
(*sendq)->size += dsize;
return;
}
/* what luck! - an exact fit - send the packet */
if ((*sendq)->size + dsize == SAM_STREAM_PAYLOAD_MAX) {
memcpy((*sendq)->data + (*sendq)->size, data, dsize);
(*sendq)->size = SAM_STREAM_PAYLOAD_MAX;
sam_sendq_flush(stream_id, sendq);
return;
}
/* they have more data than the queue can hold, so we'll have to send some*/
size_t s = SAM_STREAM_PAYLOAD_MAX - (*sendq)->size; // space free in packet
memcpy((*sendq)->data + (*sendq)->size, data, s); //append as much as we can
dsize -= s; /* update dsize to the size of whatever data hasn't been sent*/
(*sendq)->size = SAM_STREAM_PAYLOAD_MAX; /* it's a full packet */
sam_sendq_flush(stream_id, sendq); /* send the queued data */
sam_sendq_add(stream_id, sendq, data + s, dsize); /* recurse the rest */
return;
}
/*
* Creates a data queue for use with sam_stream_send
* Creates a data queue for use with sam_sendq_add()
*
* Returns: pointer to the newly created send queue
*/
sam_sendq_t *sam_sendq_create(void)
static sam_sendq_t *sam_sendq_create()
{
sam_sendq_t *sendq;
sendq = malloc(sizeof(sam_sendq_t));
sendq->data = NULL;
sendq->data = malloc(SAM_STREAM_PAYLOAD_MAX);
/* ^^ a waste of memory perhaps, but more efficient than realloc'ing every
* time data is added the to queue */
sendq->size = 0;
return sendq;
@@ -720,10 +743,13 @@ sam_sendq_t *sam_sendq_create(void)
* sendq - the send queue
* stream_id - stream number to send to
*/
void sam_sendq_send(sam_sendq_t *sendq, sam_sid_t stream_id)
void sam_sendq_flush(sam_sid_t stream_id, sam_sendq_t **sendq)
{
sam_stream_send(stream_id, sendq->data, sendq->size);
free(sendq);
sam_stream_send(stream_id, (*sendq)->data, (*sendq)->size);
/* we now free it in case they aren't going to use it anymore */
free((*sendq)->data);
free(*sendq);
*sendq = NULL;
return;
}
@@ -1034,7 +1060,7 @@ const char *sam_strerror(samerr_t code)
*
* Returns: SAM error code
*/
samerr_t sam_winsock_cleanup(void)
samerr_t sam_winsock_cleanup()
{
if (WSACleanup() == SOCKET_ERROR) {
SAMLOG("WSACleanup() failed: %s",
@@ -1050,7 +1076,7 @@ samerr_t sam_winsock_cleanup(void)
*
* Returns: SAM error code
*/
samerr_t sam_winsock_startup(void)
samerr_t sam_winsock_startup()
{
/*
* Is Windows retarded or what?