summaryrefslogtreecommitdiff
path: root/src/or
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2005-03-02 20:22:10 +0000
committerNick Mathewson <nickm@torproject.org>2005-03-02 20:22:10 +0000
commitb494c2223df3a53e47045cf8044bbaeb2377ebf1 (patch)
treeaad36c5019bc43d1e5639db095bd20bd7cc85e8d /src/or
parent65230fd39f0c333ce8616f5a578f4941ea9f0b94 (diff)
downloadtor-b494c2223df3a53e47045cf8044bbaeb2377ebf1.tar.gz
tor-b494c2223df3a53e47045cf8044bbaeb2377ebf1.zip
Specify and implement fragmented control messages to allow for (among other things) long GETINFO replies. Otherwise we could hit the 64K barrier on questions like "please dump your client-side DNS cache."
svn:r3726
Diffstat (limited to 'src/or')
-rw-r--r--src/or/buffers.c93
-rw-r--r--src/or/control.c116
-rw-r--r--src/or/or.h2
3 files changed, 152 insertions, 59 deletions
diff --git a/src/or/buffers.c b/src/or/buffers.c
index a6e765eace..c6a8d698be 100644
--- a/src/or/buffers.c
+++ b/src/or/buffers.c
@@ -645,20 +645,24 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) {
}
}
+
+#define CONTROL_CMD_FRAGMENTHEADER 0x0010
+#define CONTROL_CMD_FRAGMENT 0x0011
/** If there is a complete control message waiting on buf, then store
* its contents into *<b>type_out</b>, store its body's length into
* *<b>len_out</b>, allocate and store a string for its body into
- * *<b>body_out</b>, and return -1. (body_out will always be NUL-terminated,
+ * *<b>body_out</b>, and return 1. (body_out will always be NUL-terminated,
* even if the control message body doesn't end with NUL.)
*
* If there is not a complete control message waiting, return 0.
*
* Return -1 on error.
*/
-int fetch_from_buf_control(buf_t *buf, uint16_t *len_out, uint16_t *type_out,
+int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out,
char **body_out)
{
- uint16_t len;
+ uint32_t msglen;
+ uint16_t type;
tor_assert(buf);
tor_assert(len_out);
@@ -668,23 +672,82 @@ int fetch_from_buf_control(buf_t *buf, uint16_t *len_out, uint16_t *type_out,
if (buf->datalen < 4)
return 0;
- len = ntohs(get_uint16(buf->mem));
- if (buf->datalen < 4 + (unsigned)len)
+ msglen = ntohs(get_uint16(buf->mem));
+ if (buf->datalen < 4 + (unsigned)msglen)
return 0;
- *len_out = len;
- *type_out = ntohs(get_uint16(buf->mem+2));
- if (len) {
- *body_out = tor_malloc(len+1);
- memcpy(*body_out, buf->mem+4, len);
- (*body_out)[len] = '\0';
+ type = ntohs(get_uint16(buf->mem+2));
+ if (type != CONTROL_CMD_FRAGMENTHEADER) {
+ *len_out = msglen;
+ *type_out = type;
+ if (msglen) {
+ *body_out = tor_malloc(msglen+1);
+ memcpy(*body_out, buf->mem+4, msglen);
+ (*body_out)[msglen] = '\0';
+ } else {
+ *body_out = NULL;
+ }
+ buf_remove_from_front(buf, 4+msglen);
+
+ return 1;
} else {
- *body_out = NULL;
- }
+ uint32_t totallen, sofar;
+ char *cp, *endp, *outp;
+
+ /* Okay, we have a fragmented message. Is it all here? */
+ if (msglen < 6)
+ return -1;
+ type = htons(get_uint16(buf->mem+4));
+ totallen = htonl(get_uint32(buf->mem+6));
+ if (totallen < 65536)
+ return -1;
+
+ if (buf->datalen<4+6+totallen)
+ /* The data can't possibly be here yet, no matter how well it's packed.*/
+ return 0;
+
+ /* Count how much data is really here. */
+ sofar = msglen-6;
+ cp = buf->mem+4+msglen;
+ endp = buf->mem+buf->datalen;
+ while (sofar < totallen) {
+ if ((endp-cp)<4)
+ return 0; /* Fragment header not all here. */
+ msglen = ntohs(get_uint16(cp));
+ if (ntohs(get_uint16(cp+2) != CONTROL_CMD_FRAGMENT))
+ return -1; /* Missing fragment message; error. */
+ if ((endp-cp) < 4+msglen)
+ return 0; /* Fragment not all here. */
+ sofar += msglen;
+ cp += (4+msglen);
+ }
+ if (sofar > totallen)
+ return -1; /* Fragments add to more than expected; error. */
+
+ /* Okay, everything is here. */
+ *len_out = totallen;
+ *type_out = type;
+ *body_out = tor_malloc(totallen+1);
+
+ /* copy FRAGMENTED packet contents. */
+ msglen = ntohs(get_uint16(buf->mem));
+ if (msglen>6)
+ memcpy(*body_out,buf->mem+4+6,msglen-6);
+ sofar = msglen-6;
+ outp = *body_out+sofar;
+ cp = buf->mem+4+msglen;
+ while (sofar < totallen) {
+ msglen = ntohs(get_uint16(cp));
+ memcpy(outp,cp+4,msglen);
+ outp += msglen;
+ cp += 4+msglen;
+ sofar -= msglen;
+ }
+ (*body_out)[totallen]='\0';
- buf_remove_from_front(buf, 4+len);
+ return 1;
+ }
- return 1;
}
/** Log an error and exit if <b>buf</b> is corrupted.
diff --git a/src/or/control.c b/src/or/control.c
index ff54f38959..4714e8ee7a 100644
--- a/src/or/control.c
+++ b/src/or/control.c
@@ -46,7 +46,9 @@ const char control_c_id[] = "$Id$";
#define CONTROL_CMD_EXTENDCIRCUIT 0x000D
#define CONTROL_CMD_ATTACHSTREAM 0x000E
#define CONTROL_CMD_POSTDESCRIPTOR 0x000F
-#define _CONTROL_CMD_MAX_RECOGNIZED 0x000F
+#define CONTROL_CMD_FRAGMENTHEADER 0x0010
+#define CONTROL_CMD_FRAGMENT 0x0011
+#define _CONTROL_CMD_MAX_RECOGNIZED 0x0011
/* Recognized error codes. */
#define ERR_UNSPECIFIED 0x0000
@@ -89,7 +91,9 @@ static const char * CONTROL_COMMANDS[_CONTROL_CMD_MAX_RECOGNIZED+1] = {
"infovalue",
"extendcircuit",
"attachstream",
- "postdescriptor"
+ "postdescriptor",
+ "fragmentheader",
+ "fragment",
};
/** Bitfield: The bit 1&lt;&lt;e is set if <b>any</b> open control
@@ -115,33 +119,33 @@ static char authentication_cookie[AUTHENTICATION_COOKIE_LEN];
static void update_global_event_mask(void);
static void send_control_message(connection_t *conn, uint16_t type,
- uint16_t len, const char *body);
+ uint32_t len, const char *body);
static void send_control_done(connection_t *conn);
static void send_control_done2(connection_t *conn, const char *msg, size_t len);
static void send_control_error(connection_t *conn, uint16_t error,
const char *message);
-static void send_control_event(uint16_t event, uint16_t len, const char *body);
-static int handle_control_setconf(connection_t *conn, uint16_t len,
+static void send_control_event(uint16_t event, uint32_t len, const char *body);
+static int handle_control_setconf(connection_t *conn, uint32_t len,
char *body);
-static int handle_control_getconf(connection_t *conn, uint16_t len,
+static int handle_control_getconf(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_setevents(connection_t *conn, uint16_t len,
+static int handle_control_setevents(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_authenticate(connection_t *conn, uint16_t len,
+static int handle_control_authenticate(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_saveconf(connection_t *conn, uint16_t len,
+static int handle_control_saveconf(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_signal(connection_t *conn, uint16_t len,
+static int handle_control_signal(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_mapaddress(connection_t *conn, uint16_t len,
+static int handle_control_mapaddress(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_getinfo(connection_t *conn, uint16_t len,
+static int handle_control_getinfo(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_extendcircuit(connection_t *conn, uint16_t len,
+static int handle_control_extendcircuit(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_attachstream(connection_t *conn, uint16_t len,
+static int handle_control_attachstream(connection_t *conn, uint32_t len,
const char *body);
-static int handle_control_postdescriptor(connection_t *conn, uint16_t len,
+static int handle_control_postdescriptor(connection_t *conn, uint32_t len,
const char *body);
/** Given a possibly invalid message type code <b>cmd</b>, return a
@@ -172,18 +176,38 @@ static void update_global_event_mask(void)
/** Send a message of type <b>type</b> containing <b>len</b> bytes
* from <b>body</b> along the control connection <b>conn</b> */
static void
-send_control_message(connection_t *conn, uint16_t type, uint16_t len,
+send_control_message(connection_t *conn, uint16_t type, uint32_t len,
const char *body)
{
- char buf[4];
+ char buf[10];
tor_assert(conn);
tor_assert(len || !body);
tor_assert(type <= _CONTROL_CMD_MAX_RECOGNIZED);
- set_uint16(buf, htons(len));
- set_uint16(buf+2, htons(type));
- connection_write_to_buf(buf, 4, conn);
- if (len)
- connection_write_to_buf(body, len, conn);
+ if (len < 65536) {
+ set_uint16(buf, htons(len));
+ set_uint16(buf+2, htons(type));
+ connection_write_to_buf(buf, 4, conn);
+ if (len)
+ connection_write_to_buf(body, len, conn);
+ } else {
+ set_uint16(buf, htons(65535));
+ set_uint16(buf+2, htons(CONTROL_CMD_FRAGMENTHEADER));
+ set_uint16(buf+4, htons(type));
+ set_uint32(buf+6, htonl(len));
+ connection_write_to_buf(buf, 10, conn);
+ connection_write_to_buf(body, 65535-6, conn);
+ len -= (65535-6);
+ body += (65535-6);
+ while (len) {
+ size_t chunklen = (len<65535)?len:65535;
+ set_uint16(buf, htons((uint16_t)chunklen));
+ set_uint16(buf+2, htons(CONTROL_CMD_FRAGMENT));
+ connection_write_to_buf(buf, 4, conn);
+ connection_write_to_buf(body, chunklen, conn);
+ len -= chunklen;
+ body += chunklen;
+ }
+ }
}
/** Send a "DONE" message down the control connection <b>conn</b> */
@@ -216,7 +240,7 @@ send_control_error(connection_t *conn, uint16_t error, const char *message)
* <b>len</b> bytes in <b>body</b> to every control connection that
* is interested in it. */
static void
-send_control_event(uint16_t event, uint16_t len, const char *body)
+send_control_event(uint16_t event, uint32_t len, const char *body)
{
connection_t **conns;
int n_conns, i;
@@ -233,7 +257,7 @@ send_control_event(uint16_t event, uint16_t len, const char *body)
if (conns[i]->type == CONN_TYPE_CONTROL &&
conns[i]->state == CONTROL_CONN_STATE_OPEN &&
conns[i]->event_mask & (1<<event)) {
- send_control_message(conns[i], CONTROL_CMD_EVENT, (uint16_t)(buflen), buf);
+ send_control_message(conns[i], CONTROL_CMD_EVENT, buflen, buf);
}
}
@@ -243,7 +267,7 @@ send_control_event(uint16_t event, uint16_t len, const char *body)
/** Called when we receive a SETCONF message: parse the body and try
* to update our configuration. Reply with a DONE or ERROR message. */
static int
-handle_control_setconf(connection_t *conn, uint16_t len, char *body)
+handle_control_setconf(connection_t *conn, uint32_t len, char *body)
{
int r;
struct config_line_t *lines=NULL;
@@ -278,7 +302,7 @@ handle_control_setconf(connection_t *conn, uint16_t len, char *body)
/** Called when we receive a GETCONF message. Parse the request, and
* reply with a CONFVALUE or an ERROR message */
static int
-handle_control_getconf(connection_t *conn, uint16_t body_len, const char *body)
+handle_control_getconf(connection_t *conn, uint32_t body_len, const char *body)
{
smartlist_t *questions = NULL;
smartlist_t *answers = NULL;
@@ -332,7 +356,7 @@ handle_control_getconf(connection_t *conn, uint16_t body_len, const char *body)
/** Called when we get a SETEVENTS message: update conn->event_mask,
* and reply with DONE or ERROR. */
static int
-handle_control_setevents(connection_t *conn, uint16_t len, const char *body)
+handle_control_setevents(connection_t *conn, uint32_t len, const char *body)
{
uint16_t event_code;
uint32_t event_mask = 0;
@@ -382,7 +406,7 @@ decode_hashed_password(char *buf, const char *hashed)
* OPEN. Reply with DONE or ERROR.
*/
static int
-handle_control_authenticate(connection_t *conn, uint16_t len, const char *body)
+handle_control_authenticate(connection_t *conn, uint32_t len, const char *body)
{
or_options_t *options = get_options();
if (options->CookieAuthentication) {
@@ -421,7 +445,7 @@ handle_control_authenticate(connection_t *conn, uint16_t len, const char *body)
}
static int
-handle_control_saveconf(connection_t *conn, uint16_t len,
+handle_control_saveconf(connection_t *conn, uint32_t len,
const char *body)
{
if (save_current_config()<0) {
@@ -434,7 +458,7 @@ handle_control_saveconf(connection_t *conn, uint16_t len,
}
static int
-handle_control_signal(connection_t *conn, uint16_t len,
+handle_control_signal(connection_t *conn, uint32_t len,
const char *body)
{
if (len != 1) {
@@ -449,7 +473,7 @@ handle_control_signal(connection_t *conn, uint16_t len,
}
static int
-handle_control_mapaddress(connection_t *conn, uint16_t len, const char *body)
+handle_control_mapaddress(connection_t *conn, uint32_t len, const char *body)
{
smartlist_t *elts;
smartlist_t *lines;
@@ -478,8 +502,9 @@ handle_control_mapaddress(connection_t *conn, uint16_t len, const char *body)
log_fn(LOG_WARN,
"Unable to allocate address for '%s' in AdressMap msg", line);
} else {
- char *ans = tor_malloc(strlen(addr)+strlen(to)+2);
- tor_snprintf(ans, "%s %s", addr, to);
+ size_t anslen = strlen(addr)+strlen(to)+2;
+ char *ans = tor_malloc(anslen);
+ tor_snprintf(ans, anslen, "%s %s", addr, to);
addressmap_register(addr, tor_strdup(to), 0);
smartlist_add(reply, ans);
}
@@ -542,7 +567,7 @@ handle_getinfo_helper(const char *question)
}
static int
-handle_control_getinfo(connection_t *conn, uint16_t len, const char *body)
+handle_control_getinfo(connection_t *conn, uint32_t len, const char *body)
{
smartlist_t *questions = NULL;
smartlist_t *answers = NULL;
@@ -574,7 +599,7 @@ handle_control_getinfo(connection_t *conn, uint16_t len, const char *body)
msg = smartlist_join_strings2(answers, "\0", 1, 1, &msg_len);
send_control_message(conn, CONTROL_CMD_INFOVALUE,
- (uint16_t)msg_len, msg_len?msg:NULL);
+ msg_len, msg_len?msg:NULL);
done:
if (answers) SMARTLIST_FOREACH(answers, char *, cp, tor_free(cp));
@@ -586,20 +611,20 @@ handle_control_getinfo(connection_t *conn, uint16_t len, const char *body)
return 0;
}
static int
-handle_control_extendcircuit(connection_t *conn, uint16_t len,
+handle_control_extendcircuit(connection_t *conn, uint32_t len,
const char *body)
{
send_control_error(conn,ERR_UNRECOGNIZED_TYPE,"not yet implemented");
return 0;
}
-static int handle_control_attachstream(connection_t *conn, uint16_t len,
+static int handle_control_attachstream(connection_t *conn, uint32_t len,
const char *body)
{
send_control_error(conn,ERR_UNRECOGNIZED_TYPE,"not yet implemented");
return 0;
}
static int
-handle_control_postdescriptor(connection_t *conn, uint16_t len,
+handle_control_postdescriptor(connection_t *conn, uint32_t len,
const char *body)
{
if (router_load_single_router(body)<0) {
@@ -634,7 +659,8 @@ int connection_control_reached_eof(connection_t *conn) {
*/
int
connection_control_process_inbuf(connection_t *conn) {
- uint16_t body_len, command_type;
+ uint32_t body_len;
+ uint16_t command_type;
char *body;
tor_assert(conn);
@@ -726,6 +752,10 @@ connection_control_process_inbuf(connection_t *conn) {
send_control_error(conn, ERR_UNRECOGNIZED_TYPE,
"Command type only valid from server to tor client");
break;
+ case CONTROL_CMD_FRAGMENTHEADER:
+ case CONTROL_CMD_FRAGMENT:
+ log_fn(LOG_WARN, "Recieved command fragment out of order; ignoring.");
+ send_control_error(conn, ERR_SYNTAX, "Bad fragmentation on command.");
default:
log_fn(LOG_WARN, "Received unrecognized command type %d; ignoring.",
(int)command_type);
@@ -756,7 +786,7 @@ control_event_circuit_status(circuit_t *circ, circuit_status_event_t tp)
set_uint32(msg+1, htonl(circ->global_identifier));
strlcpy(msg+5,path,path_len+1);
- send_control_event(EVENT_CIRCUIT_STATUS, (uint16_t)(path_len+6), msg);
+ send_control_event(EVENT_CIRCUIT_STATUS, (uint32_t)(path_len+6), msg);
tor_free(path);
tor_free(msg);
return 0;
@@ -784,7 +814,7 @@ control_event_stream_status(connection_t *conn, stream_status_event_t tp)
set_uint32(msg+1, htonl(conn->global_identifier));
strlcpy(msg+5, buf, len+1);
- send_control_event(EVENT_STREAM_STATUS, (uint16_t)(5+len+1), msg);
+ send_control_event(EVENT_STREAM_STATUS, (uint32_t)(5+len+1), msg);
tor_free(msg);
return 0;
}
@@ -805,7 +835,7 @@ control_event_or_conn_status(connection_t *conn,or_conn_status_event_t tp)
buf[0] = (uint8_t)tp;
strlcpy(buf+1,conn->nickname,sizeof(buf)-1);
len = strlen(buf+1);
- send_control_event(EVENT_OR_CONN_STATUS, (uint16_t)(len+1), buf);
+ send_control_event(EVENT_OR_CONN_STATUS, (uint32_t)(len+1), buf);
return 0;
}
@@ -837,7 +867,7 @@ control_event_logmsg(int severity, const char *msg)
return;
len = strlen(msg);
- send_control_event(EVENT_WARNING, (uint16_t)(len+1), msg);
+ send_control_event(EVENT_WARNING, (uint32_t)(len+1), msg);
}
/** Choose a random authentication cookie and write it to disk.
diff --git a/src/or/or.h b/src/or/or.h
index fa76f55d5c..f01b6b31cf 100644
--- a/src/or/or.h
+++ b/src/or/or.h
@@ -1100,7 +1100,7 @@ int fetch_from_buf_http(buf_t *buf,
char **headers_out, size_t max_headerlen,
char **body_out, size_t *body_used, size_t max_bodylen);
int fetch_from_buf_socks(buf_t *buf, socks_request_t *req);
-int fetch_from_buf_control(buf_t *buf, uint16_t *len_out, uint16_t *type_out,
+int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out,
char **body_out);
void assert_buf_ok(buf_t *buf);