commit - 4c2acd55c6ecb8d9a6a1c6c5c86295fdd859820c
commit + 35f1db5f28245579efbbb06eed6eaf1a3d6d84bc
blob - 4dfe62fbdb85935030dc3112032baee6668ba9ff
blob + 77c8cd8a04ae7c6dca8958663326b52a34053180
--- src/ngircd/conn.c
+++ src/ngircd/conn.c
#define SD_LISTEN_FDS_START 3 /** systemd(8) socket activation offset */
+#define THROTTLE_CMDS 1 /** Throttling: max commands reached */
+#define THROTTLE_BPS 2 /** Throttling: max bps reached */
+
static bool Handle_Write PARAMS(( CONN_ID Idx ));
static bool Conn_Write PARAMS(( CONN_ID Idx, char *Data, size_t Len ));
static int New_Connection PARAMS(( int Sock, bool IsSSL ));
static void Simple_Message PARAMS(( int Sock, const char *Msg ));
static int NewListener PARAMS(( const char *listen_addr, UINT16 Port ));
static void Account_Connection PARAMS((void));
+static void Throttle_Connection PARAMS((const CONN_ID Idx, CLIENT *Client,
+ const int Reason, unsigned int Value));
static array My_Listeners;
static array My_ConnArray;
Conn_Handler(void)
{
int i;
- size_t wdatalen, bytes_processed;
+ size_t wdatalen;
struct timeval tv;
time_t t;
if ((My_Connections[i].sock > NONE)
&& (array_bytes(&My_Connections[i].rbuf) > 0)) {
/* ... and try to handle the received data */
- bytes_processed = Handle_Buffer(i);
- /* if we processed data, and there might be
- * more commands in the input buffer, do not
- * try to read any more data now */
- if (bytes_processed &&
- array_bytes(&My_Connections[i].rbuf) > 2) {
- LogDebug
- ("Throttling connection %d: command limit reached!",
- i);
- Conn_SetPenalty(i, 1);
- }
+ Handle_Buffer(i);
}
}
{
/* Read buffer is full */
Log(LOG_ERR,
- "Receive buffer space exhausted (connection %d): %d bytes",
- Idx, array_bytes(&My_Connections[Idx].rbuf));
+ "Receive buffer space exhausted (connection %d): %d/%d bytes",
+ Idx, array_bytes(&My_Connections[Idx].rbuf), READBUFFER_LEN);
Conn_Close(Idx, "Receive buffer space exhausted", NULL, false);
return;
}
/* Update connection statistics */
My_Connections[Idx].bytes_in += len;
+
+ /* Handle read buffer */
My_Connections[Idx].bps += Handle_Buffer(Idx);
/* Make sure that there is still a valid client registered */
}
/* Look at the data in the (read-) buffer of this connection */
- if (Client_Type(c) != CLIENT_SERVER
- && Client_Type(c) != CLIENT_UNKNOWNSERVER
- && Client_Type(c) != CLIENT_SERVICE
- && My_Connections[Idx].bps >= maxbps) {
- LogDebug("Throttling connection %d: BPS exceeded! (%u >= %u)",
- Idx, My_Connections[Idx].bps, maxbps);
- Conn_SetPenalty(Idx, 1);
- }
+ if (My_Connections[Idx].bps >= maxbps)
+ Throttle_Connection(Idx, c, THROTTLE_BPS, maxbps);
} /* Read_Request */
/**
Idx, i, maxcmd, len_processed,
array_bytes(&My_Connections[Idx].rbuf));
#endif
+
+ /* If data has been processed but there is still data in the read
+ * buffer, the command limit triggered. Enforce the penalty time: */
+ if (len_processed && array_bytes(&My_Connections[Idx].rbuf) > 2)
+ Throttle_Connection(Idx, c, THROTTLE_CMDS, maxcmd);
return len_processed;
} /* Handle_Buffer */
}
return NONE;
} /* Conn_GetFromProc */
+
+/**
+ * Throttle a connection because of excessive usage.
+ *
+ * @param Reason The reason, see THROTTLE_xxx constants.
+ * @param Idx The connection index.
+ * @param Client The client of this connection.
+ * @param Seconds The time to delay this connection.
+ */
+static void
+Throttle_Connection(const CONN_ID Idx, CLIENT *Client, const int Reason,
+ unsigned int Value)
+{
+ assert(Idx > NONE);
+ assert(Client != NULL);
+ /* Never throttle servers or services, only interrupt processing */
+ if (Client_Type(Client) == CLIENT_SERVER
+ || Client_Type(Client) == CLIENT_UNKNOWNSERVER
+ || Client_Type(Client) == CLIENT_SERVICE)
+ return;
+ LogDebug("Throttling connection %d: code %d, value %d!", Idx,
+ Reason, Value);
+ Conn_SetPenalty(Idx, 1);
+}
+
#ifndef STRICT_RFC
GLOBAL long