[MEDIUM] stats: don't use s->ana_state anymore

The stats handler used to store internal states in s->ana_state. Now
we only rely on si->st0 in which we can store as many states as we
have possible outputs. This cleans up the stats code a lot and makes
it more maintainable. It has also reduced code size by a few hundred
bytes.
diff --git a/include/proto/dumpstats.h b/include/proto/dumpstats.h
index 0ba3ae0..79ae404 100644
--- a/include/proto/dumpstats.h
+++ b/include/proto/dumpstats.h
@@ -39,10 +39,18 @@
 #define STATS_TYPE_BE  1
 #define STATS_TYPE_SV  2
 
-#define STATS_ST_INIT  0
-#define STATS_ST_REQ   1
-#define STATS_ST_REP   2
-#define STATS_ST_CLOSE 3
+/* unix stats socket states */
+#define STAT_CLI_INIT   0   /* initial state */
+#define STAT_CLI_END    1   /* final state, let's close */
+#define STAT_CLI_GETREQ 2   /* wait for a request */
+#define STAT_CLI_OUTPUT 3   /* all states after this one are responses */
+#define STAT_CLI_PROMPT 3   /* display the prompt (first output, same code) */
+
+#define STAT_CLI_O_HELP 4   /* display help */
+#define STAT_CLI_O_INFO 5   /* dump info/stats */
+#define STAT_CLI_O_SESS 6   /* dump sessions */
+#define STAT_CLI_O_ERR  7   /* dump errors */
+
 
 int stats_sock_parse_request(struct stream_interface *si, char *line);
 void stats_io_handler(struct stream_interface *si);
diff --git a/src/dumpstats.c b/src/dumpstats.c
index 25a0bca..54f8bb8 100644
--- a/src/dumpstats.c
+++ b/src/dumpstats.c
@@ -232,10 +232,9 @@
 }
 
 /* Processes the stats interpreter on the statistics socket. This function is
- * called from an applet running in a stream interface. Right now we still
- * support older functions which used to emulate servers and to set
- * STATS_ST_CLOSE upon completion, but we also support a new interactive mode.
- * The function returns 1 if the request was understood, otherwise zero.
+ * called from an applet running in a stream interface. The function returns 1
+ * if the request was understood, otherwise zero. It sets si->st0 to a value
+ * designating the function which will have to process the request.
  */
 int stats_sock_parse_request(struct stream_interface *si, char *line)
 {
@@ -266,7 +265,6 @@
 	while (++arg <= MAX_STATS_ARGS)
 		args[arg] = line;
 
-	si->st0 = 0;
 	s->data_ctx.stats.flags = 0;
 	if (strcmp(args[0], "show") == 0) {
 		if (strcmp(args[1], "stat") == 0) {
@@ -280,20 +278,17 @@
 			s->data_ctx.stats.flags |= STAT_SHOW_STAT;
 			s->data_ctx.stats.flags |= STAT_FMT_CSV;
 			s->data_state = DATA_ST_INIT;
-			s->ana_state = STATS_ST_REP;
-			si->st0 = 3; // stats_dump_raw_to_buffer
+			si->st0 = STAT_CLI_O_INFO; // stats_dump_raw_to_buffer
 		}
 		else if (strcmp(args[1], "info") == 0) {
 			s->data_ctx.stats.flags |= STAT_SHOW_INFO;
 			s->data_ctx.stats.flags |= STAT_FMT_CSV;
 			s->data_state = DATA_ST_INIT;
-			s->ana_state = STATS_ST_REP;
-			si->st0 = 3; // stats_dump_raw_to_buffer
+			si->st0 = STAT_CLI_O_INFO; // stats_dump_raw_to_buffer
 		}
 		else if (strcmp(args[1], "sess") == 0) {
 			s->data_state = DATA_ST_INIT;
-			s->ana_state = STATS_ST_REP;
-			si->st0 = 4; // stats_dump_sess_to_buffer
+			si->st0 = STAT_CLI_O_SESS; // stats_dump_sess_to_buffer
 		}
 		else if (strcmp(args[1], "errors") == 0) {
 			if (*args[2])
@@ -302,8 +297,7 @@
 				s->data_ctx.errors.iid	= -1;
 			s->data_ctx.errors.px = NULL;
 			s->data_state = DATA_ST_INIT;
-			s->ana_state = STATS_ST_REP;
-			si->st0 = 5; // stats_dump_errors_to_buffer
+			si->st0 = STAT_CLI_O_ERR; // stats_dump_errors_to_buffer
 		}
 		else { /* neither "stat" nor "info" nor "sess" */
 			return 0;
@@ -316,15 +310,12 @@
 }
 
 /* This I/O handler runs as an applet embedded in a stream interface. It is
- * used to processes I/O from/to the stats unix socket. Right now we still
- * support older functions which used to emulate servers and to set
- * STATS_ST_CLOSE upon completion, but we also support a new interactive mode.
- * The system relies on a request/response flip-flop state machine. We read
- * a request, then we process it and send the response. Then we can read again.
- * This could be enhanced a lot but we're still bound to support older output
- * functions which were designed to work as hijackers.
- * At the moment, we use si->st0 as the output type, and si->st1 to indicate
- * whether we're in prompt mode or not.
+ * used to processes I/O from/to the stats unix socket. The system relies on a
+ * state machine handling requests and various responses. We read a request,
+ * then we process it and send the response, and we possibly display a prompt.
+ * Then we can read again. The state is stored in si->st0 and is one of the
+ * STAT_CLI_* constants. si->st1 is used to indicate whether prompt is enabled
+ * or not.
  */
 void stats_io_handler(struct stream_interface *si)
 {
@@ -338,18 +329,25 @@
 		goto out;
 
 	while (1) {
-		if (s->ana_state == STATS_ST_INIT) {
+		if (si->st0 == STAT_CLI_INIT) {
 			/* Stats output not initialized yet */
 			memset(&s->data_ctx.stats, 0, sizeof(s->data_ctx.stats));
 			s->data_source = DATA_SRC_STATS;
-			s->ana_state = STATS_ST_REQ;
+			si->st0 = STAT_CLI_GETREQ;
 		}
-		else if (s->ana_state == STATS_ST_REQ) {
+		else if (si->st0 == STAT_CLI_END) {
+			/* Let's close for real now. We just close the request
+			 * side, the conditions below will complete if needed.
+			 */
+			si->shutw(si);
+			break;
+		}
+		else if (si->st0 == STAT_CLI_GETREQ) {
 			reql = buffer_si_peekline(si->ob, trash, sizeof(trash));
 			if (reql <= 0) { /* closed or EOL not found */
 				if (reql == 0)
 					break;
-				s->ana_state = STATS_ST_CLOSE;
+				si->st0 = STAT_CLI_END;
 				continue;
 			}
 
@@ -369,7 +367,7 @@
 			 */
 			len = reql - 1;
 			if (trash[len] != '\n') {
-				s->ana_state = STATS_ST_CLOSE;
+				si->st0 = STAT_CLI_END;
 				continue;
 			}
 
@@ -378,77 +376,69 @@
 
 			trash[len] = '\0';
 
-			si->st0 = 1; // default to prompt
+			si->st0 = STAT_CLI_PROMPT;
 			if (len) {
 				if (strcmp(trash, "quit") == 0) {
-					s->ana_state = STATS_ST_CLOSE;
+					si->st0 = STAT_CLI_END;
 					continue;
 				}
 				else if (strcmp(trash, "prompt") == 0)
 					si->st1 = !si->st1;
 				else if (strcmp(trash, "help") == 0 ||
 					 !stats_sock_parse_request(si, trash))
-					si->st0 = 2; // help
+					si->st0 = STAT_CLI_O_HELP;
+				/* NB: stats_sock_parse_request() may have put
+				 * another STAT_CLI_O_* into si->st0.
+				 */
 			}
 			else if (!si->st1) {
 				/* if prompt is disabled, print help on empty lines,
 				 * so that the user at least knows how to enable
 				 * prompt and find help.
 				 */
-				si->st0 = 2;
+				si->st0 = STAT_CLI_O_HELP;
 			}
 
 			/* re-adjust req buffer */
 			buffer_skip(si->ob, reql);
-
-			s->ana_state = STATS_ST_REP;
 			req->flags |= BF_READ_DONTWAIT; /* we plan to read small requests */
 		}
-		else if (s->ana_state == STATS_ST_REP) {
+		else {	/* output functions: first check if the output buffer is closed then abort */
 			if (res->flags & (BF_SHUTR_NOW|BF_SHUTR)) {
-				s->ana_state = STATS_ST_CLOSE;
+				si->st0 = STAT_CLI_END;
 				continue;
 			}
 
 			switch (si->st0) {
-			case 2:	/* help */
+			case STAT_CLI_O_HELP:
 				if (buffer_feed(si->ib, stats_sock_usage.str, stats_sock_usage.len) < 0)
-					/* message sent or too large for buffer (!) */
-					si->st0 = 1; // send prompt
+					si->st0 = STAT_CLI_PROMPT;
 				break;
-			case 3:	/* stats/info dump, should be split later ? */
+			case STAT_CLI_O_INFO:
 				if (stats_dump_raw_to_buffer(s, res))
-					si->st0 = 1; // end of command, send prompt
+					si->st0 = STAT_CLI_PROMPT;
 				break;
-			case 4:	/* sessions dump */
+			case STAT_CLI_O_SESS:
 				if (stats_dump_sess_to_buffer(s, res))
-					si->st0 = 1; // end of command, send prompt
+					si->st0 = STAT_CLI_PROMPT;
 				break;
-			case 5:	/* errors dump */
+			case STAT_CLI_O_ERR:	/* errors dump */
 				if (stats_dump_errors_to_buffer(s, res))
-					si->st0 = 1; // end of command, send prompt
+					si->st0 = STAT_CLI_PROMPT;
 				break;
-			default: /* abnormal state or lack of space for prompt */
-				si->st0 = 1; // return to prompt
+			default: /* abnormal state */
+				si->st0 = STAT_CLI_PROMPT;
 				break;
 			}
 
-			if (si->st0 == 1) {	/* post-command prompt (LF or LF + '> ') */
-				if (!si->st1) {	/* non-interactive mode */
-					if (buffer_feed(si->ib, "\n", 1) < 0)
-						si->st0 = 0; // end of output
-				}
-				else {		/* interactive mode */
-					if (buffer_feed(si->ib, "\n> ", 3) < 0)
-						si->st0 = 0; // end of output
-				}
+			/* The post-command prompt is either LF alone or LF + '> ' in interactive mode */
+			if (si->st0 == STAT_CLI_PROMPT) {
+				if (buffer_feed(si->ib, si->st1 ? "\n> " : "\n", si->st1 ? 3 : 1) < 0)
+					si->st0 = STAT_CLI_GETREQ;
 			}
 
-			/* If the output functions are still there, it means
-			 * they require more room.
-			 */
-			if (si->st0 > 0) {
-				s->ana_state = STATS_ST_REP; /* some old applets still force CLOSE */
+			/* If the output functions are still there, it means they require more room. */
+			if (si->st0 >= STAT_CLI_OUTPUT) {
 				si->flags |= SI_FL_WAIT_ROOM;
 				break;
 			}
@@ -459,24 +449,16 @@
 			 * to be sent in non-interactive mode.
 			 */
 			if ((res->flags & (BF_SHUTW|BF_SHUTW_NOW)) || (!si->st1 && !req->send_max)) {
-				s->ana_state = STATS_ST_CLOSE;
+				si->st0 = STAT_CLI_END;
 				continue;
 			}
 
-			/* switch state back to ST_REQ to read next requests */
-			s->ana_state = STATS_ST_REQ;
-		}
-		else if (s->ana_state == STATS_ST_CLOSE) {
-			/* Let's close for real now. We just close the request
-			 * side, the conditions below will complete if needed.
-			 */
-			si->shutw(si);
-			s->ana_state = 0;
-			break;
+			/* switch state back to GETREQ to read next requests */
+			si->st0 = STAT_CLI_GETREQ;
 		}
 	}
 
-	if ((res->flags & BF_SHUTR) && (si->state == SI_ST_EST) && (s->ana_state != STATS_ST_REQ)) {
+	if ((res->flags & BF_SHUTR) && (si->state == SI_ST_EST) && (si->st0 != STAT_CLI_GETREQ)) {
 		DPRINTF(stderr, "%s@%d: si to buf closed. req=%08x, res=%08x, st=%d\n",
 			__FUNCTION__, __LINE__, req->flags, res->flags, si->state);
 		/* Other size has closed, let's abort if we have no more processing to do
@@ -487,7 +469,7 @@
 		si->shutw(si);
 	}
 
-	if ((req->flags & BF_SHUTW) && (si->state == SI_ST_EST) && (s->ana_state != STATS_ST_REP)) {
+	if ((req->flags & BF_SHUTW) && (si->state == SI_ST_EST) && (si->st0 < STAT_CLI_OUTPUT)) {
 		DPRINTF(stderr, "%s@%d: buf to si closed. req=%08x, res=%08x, st=%d\n",
 			__FUNCTION__, __LINE__, req->flags, res->flags, si->state);
 		/* We have no more processing to do, and nothing more to send, and
@@ -513,7 +495,6 @@
 	if (unlikely(si->state == SI_ST_DIS || si->state == SI_ST_CLO)) {
 		/* check that we have released everything then unregister */
 		stream_int_unregister_handler(si);
-		s->ana_state = 0;
 	}
 }
 
diff --git a/src/proto_uxst.c b/src/proto_uxst.c
index 5c4ac4d..41ae9c6 100644
--- a/src/proto_uxst.c
+++ b/src/proto_uxst.c
@@ -471,7 +471,8 @@
 
 		stream_int_register_handler(&s->si[1], stats_io_handler);
 		s->si[1].private = s;
-		s->si[1].st0 = s->si[1].st1 = 0;
+		s->si[1].st1 = 0;
+		s->si[1].st0 = STAT_CLI_INIT;
 
 		s->srv = s->prev_srv = s->srv_conn = NULL;
 		s->pend_pos = NULL;