From eec39e9d6f991c90c8555859b5f10eb30c0370c1 Mon Sep 17 00:00:00 2001
From: Christian Heller <c.heller@plomlompom.de>
Date: Fri, 14 Nov 2014 07:02:36 +0100
Subject: [PATCH] Simplify queue reading/writing.

---
 src/client/io.c        |   5 +--
 src/client/world.h     |   1 -
 src/common/readwrite.c | 100 +++++++++++++++--------------------------
 src/common/readwrite.h |  17 +++----
 src/server/io.c        |   8 ++--
 src/server/world.h     |   1 -
 6 files changed, 48 insertions(+), 84 deletions(-)

diff --git a/src/client/io.c b/src/client/io.c
index 299df7f..69a0865 100644
--- a/src/client/io.c
+++ b/src/client/io.c
@@ -231,8 +231,7 @@ static void ping_pong_test(time_t last_server_answer_time)
 
 static void try_growing_queue(time_t * last_server_answer_time)
 {
-    if (read_file_into_queue(world.file_server_out, &world.queue,
-                             &world.queue_size))
+    if (read_file_into_queue(world.file_server_out, &world.queue))
     {
         * last_server_answer_time = time(0);
     }
@@ -244,7 +243,7 @@ static uint8_t read_outfile()
 {
     uint8_t ret = 0;
     char * msg;
-    while (NULL != (msg=get_message_from_queue(&world.queue,&world.queue_size)))
+    while (NULL != (msg = get_message_from_queue(&world.queue)))
     {
         char * log_prefix = "LOG ";
         if (!strncmp(msg, log_prefix, strlen(log_prefix)))
diff --git a/src/client/world.h b/src/client/world.h
index 047a13d..745100a 100644
--- a/src/client/world.h
+++ b/src/client/world.h
@@ -38,7 +38,6 @@ struct World
     char * player_inventory; /* one-item-per-line string list of owned items */
     char * mem_map; /* map cells of player's map memory */
     char * queue; /* Stores un-processed messages read from the input file. */
-    uint32_t queue_size;/* Length of .queue sequence of \0-terminated strings.*/
     struct yx_uint8 player_pos; /* coordinates of player on map */
     uint16_t turn; /* world/game turn */
     uint8_t halfdelay; /* how long to wait for getch() input in io_loop() */
diff --git a/src/common/readwrite.c b/src/common/readwrite.c
index 77d83d9..94150f1 100644
--- a/src/common/readwrite.c
+++ b/src/common/readwrite.c
@@ -12,7 +12,7 @@
                     * fputc(), fwrite(), fclose(), fopen(), clearerr()
                     */
 #include <stdlib.h> /* free() */
-#include <string.h> /* strlen(), memcpy() */
+#include <string.h> /* strlen(), memcpy(), strchr() */
 #include <unistd.h> /* access(), unlink() */
 #include "rexit.h" /* exit_err(), exit_trouble() */
 #include "try_malloc.h" /* try_malloc() */
@@ -183,80 +183,54 @@ extern uint32_t textfile_width(FILE * file)
 
 
 
-extern uint8_t read_file_into_queue(FILE * file, char ** queue,
-                                    uint32_t * queue_size)
+extern uint8_t read_file_into_queue(FILE * file, char ** queue)
 {
-    int test = try_fgetc(file, __func__);
-    if (EOF != test)
+    uint8_t ret = 0;
+    int test;
+    while (EOF != (test = try_fgetc(file, __func__)))
     {
-        char * err_size = "Queue growing too large.";
-        do
+        ret = 1;
+        if ('\0' != test)
         {
-            char c = (char) test;
-            if ('\n' == c)
+            if (*queue)
             {
-                c = '\0';
+                char * new_queue = try_malloc(strlen(*queue) + 1 + 1, __func__);
+                memcpy(new_queue, *queue, strlen(*queue));
+                new_queue[strlen(*queue)] = (char) test;
+                new_queue[strlen(*queue) + 1] = '\0';
+                free(*queue);
+                *queue = new_queue;
+            }
+            else
+            {
+                *queue = try_malloc(1 + 1, __func__);
+                (*queue)[0] = (char) test;
+                (*queue)[1] = '\0';
             }
-            char * new_queue = try_malloc(*queue_size + 1, __func__);
-            memcpy(new_queue, *queue, *queue_size);
-            char * new_pos = new_queue + *queue_size;
-            * new_pos = c;
-            exit_err(*queue_size == UINT32_MAX, err_size);
-            *queue_size = *queue_size + 1;
-            free(*queue);
-            *queue = new_queue;
-        }
-        while (EOF != (test = try_fgetc(file, __func__)));
-        if (*queue_size && '\0' != (*queue)[*queue_size - 1])
-        {
-            char * new_queue = try_malloc(*queue_size + 1, __func__);
-            memcpy(new_queue, *queue, *queue_size);
-            new_queue[*queue_size] = '\0';
-            exit_err(*queue_size == UINT32_MAX, err_size);
-            *queue_size = *queue_size + 1;
-            free(*queue);
-            *queue = new_queue;
         }
-        return 1;
     }
-    return 0;
+    return ret;
 }
 
 
 
-extern char * get_message_from_queue(char ** queue, uint32_t * queue_size)
+extern char * get_message_from_queue(char ** queue)
 {
-    char * message = NULL;
-    if (*queue_size)
+    if (!(*queue))
     {
-        size_t cutout_len = strlen(*queue);
-        uint8_t is_nullbyte_chunk = !cutout_len;
-        if (0 < cutout_len)
-        {
-            cutout_len++;
-            message = try_malloc(cutout_len, __func__);
-            memcpy(message, *queue, cutout_len);
-        }
-        for (;
-             cutout_len != *queue_size && '\0' == (*queue)[cutout_len];
-             cutout_len++);
-        *queue_size = *queue_size - cutout_len;
-        if (0 == *queue_size)
-        {
-            free(*queue);            /* NULL so read_file_into_queue() and    */
-            *queue = NULL;           /* cleanup() may free() this every time, */
-        }                            /* even when it's un-allocated.          */
-        else
-        {
-            char * new_queue = try_malloc(*queue_size, __func__);
-            memcpy(new_queue, &((*queue)[cutout_len]), *queue_size);
-            free(*queue);
-            *queue = new_queue;
-            if (is_nullbyte_chunk)
-            {
-                return get_message_from_queue(queue, queue_size);
-            }
-        }
+        return NULL;
+    }
+    char * first_nl = strchr(*queue, '\n');
+    if (!first_nl)
+    {
+        return NULL;
     }
-    return message;
+    char * msg = try_malloc(first_nl - (*queue) + 1, __func__);
+    memcpy(msg, *queue, first_nl - (*queue));
+    msg[first_nl - (*queue)] = '\0';
+    char * new_queue = try_malloc(strlen(first_nl + 1) + 1, __func__);
+    memcpy(new_queue, first_nl + 1, strlen(first_nl + 1) + 1);
+    free(*queue);
+    *queue = new_queue;
+    return msg;
 }
diff --git a/src/common/readwrite.h b/src/common/readwrite.h
index c151f13..ce7b749 100644
--- a/src/common/readwrite.h
+++ b/src/common/readwrite.h
@@ -53,20 +53,13 @@ extern void detect_atomic_leftover(char * path);
 /* Return largest line length from "file" (including  newline chars). */
 extern uint32_t textfile_width(FILE * file);
 
-/* Read "file" for load of bytes to put onto "queue" of "queue_size" (growing
- * that size). May put many \0-terminated strings on the queue at once. \n chars
- * are replaced with \0 chars. If the queue doesn't end in \0, a \o byte is
- * appended to it. Returns 1 if reading succeeds, 0 if nothing is read.
- */
-extern uint8_t read_file_into_queue(FILE * file, char ** queue,
-                                    uint32_t * queue_size);
+/* Read "file" for load of non-zero bytes to put onto "queue" string. */
+extern uint8_t read_file_into_queue(FILE * file, char ** queue);
 
-/* Cut out and return first \0-terminated string from "queue" and appropriately
- * reduce "queue_size". Return NULL if queue is empty. Superfluous \0 bytes
- * after the string are also cut out. Should the queue start with \0 bytes,
- * those are cut out before returning anything after them.
+/* Return message from "queue" (identified as \n-delimited string, with \n as
+ * \0 in the returned message; return nothing if no \n delimiter found).
  */
-extern char * get_message_from_queue(char ** queue, uint32_t * queue_size);
+extern char * get_message_from_queue(char ** queue);
 
 
 
diff --git a/src/server/io.c b/src/server/io.c
index 0cf2c05..d4c9faf 100644
--- a/src/server/io.c
+++ b/src/server/io.c
@@ -190,7 +190,7 @@ static void try_growing_queue()
     dur.tv_nsec = 33333333;
     while (1)
     {
-        if (read_file_into_queue(world.file_in, &world.queue,&world.queue_size))
+        if (read_file_into_queue(world.file_in, &world.queue))
         {
             return;
         }
@@ -349,9 +349,9 @@ static void write_map(struct Thing * player, FILE * file)
 
 extern char * io_round()
 {
-    if (0 < world.queue_size)
+    if (world.queue && strlen(world.queue))
     {
-        return get_message_from_queue(&world.queue, &world.queue_size);
+        return get_message_from_queue(&world.queue);
     }
     if (world.do_update)
     {
@@ -359,7 +359,7 @@ extern char * io_round()
         world.do_update = 0;
     }
     try_growing_queue();
-    return get_message_from_queue(&world.queue, &world.queue_size);
+    return get_message_from_queue(&world.queue);
 }
 
 
diff --git a/src/server/world.h b/src/server/world.h
index eedb2e3..f6deb10 100644
--- a/src/server/world.h
+++ b/src/server/world.h
@@ -29,7 +29,6 @@ struct World
     struct Thing * things; /* All physical things of the game world. */
     char * server_test; /* String uniquely identifying server process. */
     char * queue; /* Stores un-processed messages read from the input file. */
-    uint32_t queue_size;/* Length of .queue sequence of \0-terminated strings.*/
     uint32_t seed; /* Randomness seed. */
     uint32_t seed_map; /* Map seed. */
     uint16_t replay; /* Turn up to which to replay game. No replay if zero. */
-- 
2.30.2