[MEDIUM] secure the calling conditions of ->set_server_status_{up,down}

It's not always obvious for the callers of set_server_status_{up,down}
whether the new state really is up or down. Some flags as well as the
effective weight have to be considered. Let's ensure that those functions
perform the necessary check themselves so that if the state transition
cannot be performed, at least everything is updated as required.
diff --git a/src/backend.c b/src/backend.c
index cc12f7a..1c54474 100644
--- a/src/backend.c
+++ b/src/backend.c
@@ -128,16 +128,19 @@
 	    srv->eweight == srv->prev_eweight)
 		return;
 
+	if (srv_is_usable(srv->state, srv->eweight))
+		goto out_update_state;
+
 	/* FIXME: could be optimized since we know what changed */
 	recount_servers(p);
 	update_backend_weight(p);
+	p->lbprm.map.state |= PR_MAP_RECALC;
+ out_update_state:
 	srv->prev_state = srv->state;
 	srv->prev_eweight = srv->eweight;
-	p->lbprm.map.state |= PR_MAP_RECALC;
-
 }
 
-/* this function updates the map according to server <srv>'s new state */
+/* This function updates the map according to server <srv>'s new state */
 static void map_set_server_status_up(struct server *srv)
 {
 	struct proxy *p = srv->proxy;
@@ -146,12 +149,16 @@
 	    srv->eweight == srv->prev_eweight)
 		return;
 
+	if (!srv_is_usable(srv->state, srv->eweight))
+		goto out_update_state;
+
 	/* FIXME: could be optimized since we know what changed */
 	recount_servers(p);
 	update_backend_weight(p);
+	p->lbprm.map.state |= PR_MAP_RECALC;
+ out_update_state:
 	srv->prev_state = srv->state;
 	srv->prev_eweight = srv->eweight;
-	p->lbprm.map.state |= PR_MAP_RECALC;
 }
 
 /* This function recomputes the server map for proxy px. It relies on
@@ -285,8 +292,9 @@
 
 /* This function updates the server trees according to server <srv>'s new
  * state. It should be called when server <srv>'s status changes to down.
- * It is not important whether the server was already down or not. However,
- * it is mandatory that the new state be down.
+ * It is not important whether the server was already down or not. It is not
+ * important either that the new state is completely down (the caller may not
+ * know all the variables of a server's state).
  */
 static void fwrr_set_server_status_down(struct server *srv)
 {
@@ -297,6 +305,9 @@
 	    srv->eweight == srv->prev_eweight)
 		return;
 
+	if (srv_is_usable(srv->state, srv->eweight))
+		goto out_update_state;
+
 	if (!srv_is_usable(srv->prev_state, srv->prev_eweight))
 		/* server was already down */
 		goto out_update_backend;
@@ -331,15 +342,16 @@
 out_update_backend:
 	/* check/update tot_used, tot_weight */
 	update_backend_weight(p);
+ out_update_state:
 	srv->prev_state = srv->state;
 	srv->prev_eweight = srv->eweight;
-
 }
 
 /* This function updates the server trees according to server <srv>'s new
  * state. It should be called when server <srv>'s status changes to up.
- * It is not important whether the server was already down or not. However,
- * it is mandatory that the new state be up. This function will not change
+ * It is not important whether the server was already down or not. It is not
+ * important either that the new state is completely UP (the caller may not
+ * know all the variables of a server's state). This function will not change
  * the weight of a server which was already up.
  */
 static void fwrr_set_server_status_up(struct server *srv)
@@ -351,6 +363,9 @@
 	    srv->eweight == srv->prev_eweight)
 		return;
 
+	if (!srv_is_usable(srv->state, srv->eweight))
+		goto out_update_state;
+
 	if (srv_is_usable(srv->prev_state, srv->prev_eweight))
 		/* server was already up */
 		goto out_update_backend;
@@ -386,6 +401,7 @@
 out_update_backend:
 	/* check/update tot_used, tot_weight */
 	update_backend_weight(p);
+ out_update_state:
 	srv->prev_state = srv->state;
 	srv->prev_eweight = srv->eweight;
 }