client: optimized client_input_received()

Use memchr() instead of manually traversing the input buffer.  Update
the client's properties after all commands have been processed.  Check
for buffer overflow once.
This commit is contained in:
Max Kellermann 2008-10-15 18:21:45 +02:00
parent 0031eaaff5
commit 6eb62e4761

View File

@ -341,10 +341,9 @@ void client_new(int fd, const struct sockaddr *addr)
sockaddr_to_tmp_string(addr)); sockaddr_to_tmp_string(addr));
} }
static int client_process_line(struct client *client) static int client_process_line(struct client *client, char *line)
{ {
int ret = 1; int ret = 1;
char *line = client->buffer + client->bufferPos;
if (client->cmd_list_OK >= 0) { if (client->cmd_list_OK >= 0) {
if (strcmp(line, CLIENT_LIST_MODE_END) == 0) { if (strcmp(line, CLIENT_LIST_MODE_END) == 0) {
@ -414,32 +413,40 @@ static int client_process_line(struct client *client)
static int client_input_received(struct client *client, int bytesRead) static int client_input_received(struct client *client, int bytesRead)
{ {
char *start = client->buffer + client->bufferLength;
char *end = start + bytesRead;
char *newline, *next;
int ret; int ret;
char *buf_tail = &(client->buffer[client->bufferLength - 1]);
/* any input from the client makes it leave "idle" mode */ /* any input from the client makes it leave "idle" mode */
client->idle_waiting = false; client->idle_waiting = false;
while (bytesRead > 0) { client->bufferLength += bytesRead;
client->bufferLength++;
bytesRead--; /* process all lines */
buf_tail++; while ((newline = memchr(start, '\n', end - start)) != NULL) {
if (*buf_tail == '\n') { next = newline + 1;
*buf_tail = '\0';
if (client->bufferLength > client->bufferPos) { if (newline > start && newline[-1] == '\r')
if (*(buf_tail - 1) == '\r') --newline;
*(buf_tail - 1) = '\0'; *newline = 0;
}
ret = client_process_line(client); ret = client_process_line(client, start);
if (ret == COMMAND_RETURN_KILL || if (ret == COMMAND_RETURN_KILL ||
ret == COMMAND_RETURN_CLOSE) ret == COMMAND_RETURN_CLOSE)
return ret; return ret;
if (client_is_expired(client)) if (client_is_expired(client))
return COMMAND_RETURN_CLOSE; return COMMAND_RETURN_CLOSE;
client->bufferPos = client->bufferLength; start = next;
} }
if (client->bufferLength == CLIENT_MAX_BUFFER_LENGTH) {
/* mark consumed lines */
client->bufferPos = start - client->buffer;
/* if we're have reached the buffer's end, close the gab at
the beginning */
if (client->bufferLength == sizeof(client->buffer)) {
if (client->bufferPos == 0) { if (client->bufferPos == 0) {
ERROR("client %i: buffer overflow\n", ERROR("client %i: buffer overflow\n",
client->num); client->num);
@ -457,7 +464,6 @@ static int client_input_received(struct client *client, int bytesRead)
client->bufferLength); client->bufferLength);
client->bufferPos = 0; client->bufferPos = 0;
} }
}
return 0; return 0;
} }