diff options
author | Nick Mathewson <nickm@torproject.org> | 2005-04-26 20:53:22 +0000 |
---|---|---|
committer | Nick Mathewson <nickm@torproject.org> | 2005-04-26 20:53:22 +0000 |
commit | ef2409e4ed367f7fe6f8b95ca874fbaf81f96e4b (patch) | |
tree | e887bf8760020aabb83879c74c9094fdd0328da2 /src/or | |
parent | a1fee5db63bdb2044950c48e4b7e4ffd8961bb5c (diff) | |
download | tor-ef2409e4ed367f7fe6f8b95ca874fbaf81f96e4b.tar.gz tor-ef2409e4ed367f7fe6f8b95ca874fbaf81f96e4b.zip |
Change the implementation of buf_t a lot: make it a ring buffer to minimize memmove on flush. This may break the universe, but it is probably Necessary For Perfomance.
svn:r4121
Diffstat (limited to 'src/or')
-rw-r--r-- | src/or/buffers.c | 490 | ||||
-rw-r--r-- | src/or/test.c | 157 |
2 files changed, 495 insertions, 152 deletions
diff --git a/src/or/buffers.c b/src/or/buffers.c index c437d49872..c5fcd6fe56 100644 --- a/src/or/buffers.c +++ b/src/or/buffers.c @@ -12,10 +12,32 @@ const char buffers_c_id[] = "$Id$"; #include "or.h" +#undef SENTINALS +#undef CHECK_AFTER_RESIZE + +#ifdef SENTINALS +/* If SENTINALS is defined, check for attempts to write beyond the + * end/before the start of the buffer. + */ +#define START_MAGIC 0x70370370u +#define END_MAGIC 0xA0B0C0D0u +#define RAW_MEM(m) ((m)-4) +#define GUARDED_MEM(m) ((m)+4) +#define ALLOC_LEN(ln) ((ln)+8) +#define SET_GUARDS(m, ln) \ + do { set_uint32((m)-4, START_MAGIC); set_uint32((m)+ln, END_MAGIC); }while(0) +#else +#define RAW_MEM(m) (m) +#define GUARDED_MEM(m) (m) +#define ALLOC_LEN(ln) (ln) +#define SET_GUARDS(m,ln) do {} while(0) +#endif + #define BUFFER_MAGIC 0xB0FFF312u struct buf_t { uint32_t magic; /**< Magic cookie for debugging: Must be set to BUFFER_MAGIC */ char *mem; /**< Storage for data in the buffer */ + char *start; /**< The first byte used for storing data in the buffer. */ size_t len; /**< Maximum amount of data that <b>mem</b> can hold. */ size_t datalen; /**< Number of bytes currently in <b>mem</b>. */ }; @@ -27,13 +49,128 @@ struct buf_t { * than this size. */ #define MIN_BUF_SHRINK_SIZE (16*1024) -/** Change a buffer's capacity. <b>new_capacity</b> must be \<= buf->datalen. */ +static void peek_from_buf(char *string, size_t string_len, buf_t *buf); + +static void buf_normalize(buf_t *buf) +{ + if (buf->start + buf->datalen <= buf->mem+buf->len) { + return; + } else { + char *newmem; + size_t sz = (buf->mem+buf->len)-buf->start; + log_fn(LOG_WARN, "Unexpected non-normalized buffer."); + newmem = GUARDED_MEM(tor_malloc(ALLOC_LEN(buf->len))); + SET_GUARDS(newmem, buf->len); + memcpy(newmem, buf->start, sz); + memcpy(newmem+sz, buf->mem, buf->datalen-sz); + free(RAW_MEM(buf->mem)); + buf->mem = buf->start = newmem; + } +} + +/** Return the point in the buffer where the next byte will get stored. */ +static INLINE char *_buf_end(buf_t *buf) +{ + char *next = buf->start + buf->datalen; + char *end = buf->mem + buf->len; + return (next < end) ? next : (next - buf->len); +} + + +/** If the pointer <b>cp</b> has passed beyond the end of the buffer, wrap it + * around. */ +static INLINE char *_wrap_ptr(buf_t *buf, char *cp) { + return (cp >= buf->mem + buf->len) ? (cp - buf->len) : cp; +} + +/** If the range of *<b>len</b> bytes starting at <b>at</b> wraps around the + * end of the buffer, then set *<b>len</b> to the number of bytes starting + * at <b>at</b>, and set *<b>more_len</b> to the number of bytes starting + * at <b>buf->mem</b>. Otherwise, set *<b>more_len</b> to 0. + */ +static INLINE void _split_range(buf_t *buf, char *at, size_t *len, + size_t *more_len) +{ + char *eos = at + *len; + if (eos >= (buf->mem + buf->len)) { + *more_len = eos - (buf->mem + buf->len); + *len -= *more_len; + } else { + *more_len = 0; + } +} + +/** Change a buffer's capacity. <b>new_capacity</b> must be \>= buf->datalen. */ static INLINE void buf_resize(buf_t *buf, size_t new_capacity) { + off_t offset; +#ifdef CHECK_AFTER_RESIZE + char *tmp, *tmp2; +#endif tor_assert(buf->datalen <= new_capacity); tor_assert(new_capacity); - buf->mem = tor_realloc(buf->mem, new_capacity); + +#ifdef CHECK_AFTER_RESIZE + assert_buf_ok(buf); + tmp = tor_malloc(buf->datalen); + tmp2 = tor_malloc(buf->datalen); + peek_from_buf(tmp, buf->datalen, buf); +#endif + + offset = buf->start - buf->mem; + if (offset + buf->datalen >= new_capacity) { + /* We need to move stuff before we shrink. */ + if (offset+buf->datalen >= buf->len) { + /* We have: + * + * mem[0] ... mem[datalen-(len-offset)] (end of data) + * mem[offset] ... mem[len-1] (the start of the data) + * + * We're shrinking the buffer by (len-new_capacity) bytes, so we need + * to move the start portion back by that many bytes. + */ + memmove(buf->start-(buf->len-new_capacity), buf->start, + buf->len-offset); + offset -= (buf->len-new_capacity); + } else { + /* The data doen't wrap around, but it does extend beyond the new + * buffer length: + * mem[offset] ... mem[offset+datalen-1] (the data) + */ + memmove(buf->mem, buf->start, buf->datalen); + offset = 0; + } + } + buf->mem = GUARDED_MEM(tor_realloc(RAW_MEM(buf->mem), + ALLOC_LEN(new_capacity))); + SET_GUARDS(buf->mem, new_capacity); + buf->start = buf->mem+offset; + if (offset + buf->datalen >= buf->len) { + /* We need to move data now that we are done growing. The buffer + * now contains: + * + * mem[0] ... mem[datalen-(len-offset)] (end of data) + * mem[offset] ... mem[len-1] (the start of the data) + * mem[len]...mem[new_capacity] (empty space) + * + * We're growing by (new_capacity-len) bytes, so we need to move the + * end portion forward by that many bytes. + */ + memmove(buf->start+(new_capacity-buf->len), buf->start, + buf->len-offset); + buf->start += new_capacity-buf->len; + } buf->len = new_capacity; + +#ifdef CHECK_AFTER_RESIZE + assert_buf_ok(buf); + peek_from_buf(tmp2, buf->datalen, buf); + if (memcmp(tmp, tmp2, buf->datalen)) { + tor_assert(0); + } + tor_free(tmp); + tor_free(tmp2); +#endif } /** If the buffer is not large enough to hold <b>capacity</b> bytes, resize @@ -88,7 +225,7 @@ static INLINE void buf_shrink_if_underfull(buf_t *buf) { static INLINE void buf_remove_from_front(buf_t *buf, size_t n) { tor_assert(buf->datalen >= n); buf->datalen -= n; - memmove(buf->mem, buf->mem+n, buf->datalen); + buf->start = _wrap_ptr(buf, buf->start+n); buf_shrink_if_underfull(buf); } @@ -97,7 +234,7 @@ static INLINE int buf_nul_terminate(buf_t *buf) { if (buf_ensure_capacity(buf,buf->datalen+1)<0) return -1; - buf->mem[buf->datalen] = '\0'; + *_buf_end(buf) = '\0'; return 0; } @@ -107,7 +244,8 @@ buf_t *buf_new_with_capacity(size_t size) { buf_t *buf; buf = tor_malloc(sizeof(buf_t)); buf->magic = BUFFER_MAGIC; - buf->mem = tor_malloc(size); + buf->start = buf->mem = GUARDED_MEM(tor_malloc(ALLOC_LEN(size))); + SET_GUARDS(buf->mem, size); buf->len = size; buf->datalen = 0; // memset(buf->mem,0,size); @@ -126,6 +264,7 @@ buf_t *buf_new() void buf_clear(buf_t *buf) { buf->datalen = 0; + buf->start = buf->mem; } /** Return the number of bytes stored in <b>buf</b> */ @@ -145,7 +284,7 @@ size_t buf_capacity(const buf_t *buf) */ const char *_buf_peek_raw_buffer(const buf_t *buf) { - return buf->mem; + return buf->start; } /** Release storage held by <b>buf</b>. @@ -153,19 +292,46 @@ const char *_buf_peek_raw_buffer(const buf_t *buf) void buf_free(buf_t *buf) { assert_buf_ok(buf); buf->magic = 0xDEADBEEF; - tor_free(buf->mem); + free(RAW_MEM(buf->mem)); tor_free(buf); } +static INLINE int read_to_buf_impl(int s, size_t at_most, buf_t *buf, + char *pos, int *reached_eof) +{ + int read_result; + +// log_fn(LOG_DEBUG,"reading at most %d bytes.",at_most); + read_result = recv(s, pos, at_most, 0); + if (read_result < 0) { + int e = tor_socket_errno(s); + if (!ERRNO_IS_EAGAIN(e)) { /* it's a real error */ + return -1; + } + return 0; /* would block. */ + } else if (read_result == 0) { + log_fn(LOG_DEBUG,"Encountered eof"); + *reached_eof = 1; + return 0; + } else { /* we read some bytes */ + buf->datalen += read_result; + log_fn(LOG_DEBUG,"Read %d bytes. %d on inbuf.",read_result, + (int)buf->datalen); + return read_result; + } +} + /** Read from socket <b>s</b>, writing onto end of <b>buf</b>. Read at most * <b>at_most</b> bytes, resizing the buffer as necessary. If recv() * returns 0, set <b>*reached_eof</b> to 1 and return 0. Return -1 on error; * else return the number of bytes read. Return 0 if recv() would * block. */ -int read_to_buf(int s, size_t at_most, buf_t *buf, int *reached_eof) { - - int read_result; +int read_to_buf(int s, size_t at_most, buf_t *buf, int *reached_eof) +{ + int r; + char *next; + size_t at_start; assert_buf_ok(buf); tor_assert(reached_eof); @@ -180,30 +346,52 @@ int read_to_buf(int s, size_t at_most, buf_t *buf, int *reached_eof) { if (at_most == 0) return 0; /* we shouldn't read anything */ -// log_fn(LOG_DEBUG,"reading at most %d bytes.",at_most); - read_result = recv(s, buf->mem+buf->datalen, at_most, 0); - if (read_result < 0) { - int e = tor_socket_errno(s); - if (!ERRNO_IS_EAGAIN(e)) { /* it's a real error */ - return -1; + next = _buf_end(buf); + _split_range(buf, next, &at_most, &at_start); + + r = read_to_buf_impl(s, at_most, buf, next, reached_eof); + + if (r < 0 || (size_t)r < at_most) { + return r; /* Either error, eof, block, or no more to read. */ + } + + if (at_start) { + int r2; + tor_assert(_buf_end(buf) == buf->mem); + r2 = read_to_buf_impl(s, at_start, buf, buf->start, reached_eof); + if (r2 < 0) { + return r2; + } else { + r += r2; } - return 0; /* would block. */ - } else if (read_result == 0) { - log_fn(LOG_DEBUG,"Encountered eof"); - *reached_eof = 1; - return 0; - } else { /* we read some bytes */ - buf->datalen += read_result; - log_fn(LOG_DEBUG,"Read %d bytes. %d on inbuf.",read_result, - (int)buf->datalen); - return read_result; } + return r; +} + +static INLINE int +read_to_buf_tls_impl(tor_tls *tls, size_t at_most, buf_t *buf, char *next) +{ + int r; + + log_fn(LOG_DEBUG,"before: %d on buf, %d pending, at_most %d.", + (int)buf_datalen(buf), (int)tor_tls_get_pending_bytes(tls), + (int)at_most); + r = tor_tls_read(tls, next, at_most); + if (r<0) + return r; + buf->datalen += r; + log_fn(LOG_DEBUG,"Read %d bytes. %d on inbuf; %d pending",r, + (int)buf->datalen,(int)tor_tls_get_pending_bytes(tls)); + return r; } /** As read_to_buf, but reads from a TLS connection. */ int read_to_buf_tls(tor_tls *tls, size_t at_most, buf_t *buf) { int r; + char *next; + size_t at_start; + tor_assert(tls); assert_buf_ok(buf); @@ -220,20 +408,48 @@ int read_to_buf_tls(tor_tls *tls, size_t at_most, buf_t *buf) { if (at_most == 0) return 0; - log_fn(LOG_DEBUG,"before: %d on buf, %d pending, at_most %d.", - (int)buf_datalen(buf), (int)tor_tls_get_pending_bytes(tls), - (int)at_most); - - check_no_tls_errors(); - r = tor_tls_read(tls, buf->mem+buf->datalen, at_most); - if (r<0) - return r; - buf->datalen += r; - log_fn(LOG_DEBUG,"Read %d bytes. %d on inbuf; %d pending",r, - (int)buf->datalen,(int)tor_tls_get_pending_bytes(tls)); + next = _buf_end(buf); + _split_range(buf, next, &at_most, &at_start); + + r = read_to_buf_tls_impl(tls, at_most, buf, next); + if (r < 0 || (size_t)r < at_most) + return r; /* Either error, eof, block, or no more to read. */ + + if (at_start) { + int r2; + tor_assert(_buf_end(buf) == buf->mem); + r2 = read_to_buf_tls_impl(tls, at_most, buf, buf->mem); + if (r2 < 0) + return r2; + else + r += r2; + } return r; } +static INLINE int +flush_buf_impl(int s, buf_t *buf, size_t sz, size_t *buf_flushlen) +{ + int write_result; + + write_result = send(s, buf->start, sz, 0); + if (write_result < 0) { + int e = tor_socket_errno(s); + if (!ERRNO_IS_EAGAIN(e)) { /* it's a real error */ + return -1; + } + log_fn(LOG_DEBUG,"write() would block, returning."); + return 0; + } else { + *buf_flushlen -= write_result; + + buf_remove_from_front(buf, write_result); + + return write_result; + } +} + + /** Write data from <b>buf</b> to the socket <b>s</b>. Write at most * <b>*buf_flushlen</b> bytes, decrement <b>*buf_flushlen</b> by * the number of bytes actually written, and remove the written bytes @@ -242,7 +458,9 @@ int read_to_buf_tls(tor_tls *tls, size_t at_most, buf_t *buf) { */ int flush_buf(int s, buf_t *buf, size_t *buf_flushlen) { - int write_result; + int r; + size_t flushed = 0; + size_t flushlen0, flushlen1; assert_buf_ok(buf); tor_assert(buf_flushlen); @@ -252,29 +470,53 @@ int flush_buf(int s, buf_t *buf, size_t *buf_flushlen) if (*buf_flushlen == 0) /* nothing to flush */ return 0; - write_result = send(s, buf->mem, *buf_flushlen, 0); - if (write_result < 0) { - int e = tor_socket_errno(s); - if (!ERRNO_IS_EAGAIN(e)) { /* it's a real error */ - return -1; - } - log_fn(LOG_DEBUG,"write() would block, returning."); - return 0; - } else { - *buf_flushlen -= write_result; + flushlen0 = *buf_flushlen; + _split_range(buf, buf->start, &flushlen0, &flushlen1); + + r = flush_buf_impl(s, buf, flushlen0, buf_flushlen); + + log_fn(LOG_DEBUG,"%d: flushed %d bytes, %d ready to flush, %d remain.", + s,r,(int)*buf_flushlen,(int)buf->datalen); + if (r < 0 || (size_t)r < flushlen0) + return r; /* Error, or can't flush any more now. */ + flushed = r; + + if (flushlen1) { + tor_assert(buf->start == buf->mem); + r = flush_buf_impl(s, buf, flushlen1, buf_flushlen); log_fn(LOG_DEBUG,"%d: flushed %d bytes, %d ready to flush, %d remain.", - s,write_result,(int)*buf_flushlen,(int)buf->datalen-write_result); - buf_remove_from_front(buf, write_result); + s,r,(int)*buf_flushlen,(int)buf->datalen); + if (r<0) + return r; + flushed += r; + } + return flushed; +} - return write_result; +static INLINE int +flush_buf_tls_impl(tor_tls *tls, buf_t *buf, size_t sz, size_t *buf_flushlen) +{ + int r; + + r = tor_tls_write(tls, buf->start, sz); + if (r < 0) { + return r; } + *buf_flushlen -= r; + buf_remove_from_front(buf, r); + log_fn(LOG_DEBUG,"flushed %d bytes, %d ready to flush, %d remain.", + r,(int)*buf_flushlen,(int)buf->datalen); + return r; } + /** As flush_buf, but writes data to a TLS connection. */ int flush_buf_tls(tor_tls *tls, buf_t *buf, size_t *buf_flushlen) { int r; + size_t flushed=0; + size_t flushlen0, flushlen1; assert_buf_ok(buf); tor_assert(tls); tor_assert(buf_flushlen); @@ -282,15 +524,23 @@ int flush_buf_tls(tor_tls *tls, buf_t *buf, size_t *buf_flushlen) /* we want to let tls write even if flushlen is zero, because it might * have a partial record pending */ check_no_tls_errors(); - r = tor_tls_write(tls, buf->mem, *buf_flushlen); - if (r < 0) { - return r; + + flushlen0 = *buf_flushlen; + _split_range(buf, buf->start, &flushlen0, &flushlen1); + + r = flush_buf_tls_impl(tls, buf, flushlen0, buf_flushlen); + if (r < 0 || (size_t)r < flushlen0) + return r; /* Error, or can't flush any more now. */ + flushed = r; + + if (flushlen1) { + tor_assert(buf->start == buf->mem); + r = flush_buf_tls_impl(tls, buf, flushlen1, buf_flushlen); + if (r<0) + return r; + flushed += r; } - *buf_flushlen -= r; - buf_remove_from_front(buf, r); - log_fn(LOG_DEBUG,"flushed %d bytes, %d ready to flush, %d remain.", - r,(int)*buf_flushlen,(int)buf->datalen); - return r; + return flushed; } /** Append <b>string_len</b> bytes from <b>string</b> to the end of @@ -298,7 +548,11 @@ int flush_buf_tls(tor_tls *tls, buf_t *buf, size_t *buf_flushlen) * * Return the new length of the buffer on success, -1 on failure. */ -int write_to_buf(const char *string, size_t string_len, buf_t *buf) { +int +write_to_buf(const char *string, size_t string_len, buf_t *buf) +{ + char *next; + size_t len2; /* append string to buf (growing as needed, return -1 if "too big") * return total number of bytes on the buf @@ -312,17 +566,24 @@ int write_to_buf(const char *string, size_t string_len, buf_t *buf) { return -1; } - memcpy(buf->mem+buf->datalen, string, string_len); + next = _buf_end(buf); + _split_range(buf, next, &string_len, &len2); + + memcpy(next, string, string_len); buf->datalen += string_len; + + if (len2) { + tor_assert(_buf_end(buf) == buf->mem); + memcpy(buf->mem, string+string_len, len2); + buf->datalen += len2; + } log_fn(LOG_DEBUG,"added %d bytes to buf (now %d total).",(int)string_len, (int)buf->datalen); return buf->datalen; } -/** Remove <b>string_len</b> bytes from the front of <b>buf</b>, and store them - * into <b>string</b>. Return the new buffer size. <b>string_len</b> must be \<= - * the number of bytes on the buffer. - */ -int fetch_from_buf(char *string, size_t string_len, buf_t *buf) { +static INLINE void peek_from_buf(char *string, size_t string_len, buf_t *buf) +{ + size_t len2; /* There must be string_len bytes in buf; write them onto string, * then memmove buf back (that is, remove them from buf). @@ -333,7 +594,26 @@ int fetch_from_buf(char *string, size_t string_len, buf_t *buf) { tor_assert(string_len <= buf->datalen); /* make sure we don't ask for too much */ assert_buf_ok(buf); - memcpy(string,buf->mem,string_len); + _split_range(buf, buf->start, &string_len, &len2); + + memcpy(string, buf->start, string_len); + if (len2) { + memcpy(string+string_len,buf->mem,len2); + } +} + +/** Remove <b>string_len</b> bytes from the front of <b>buf</b>, and store them + * into <b>string</b>. Return the new buffer size. <b>string_len</b> must be \<= + * the number of bytes on the buffer. + */ +int fetch_from_buf(char *string, size_t string_len, buf_t *buf) +{ + /* There must be string_len bytes in buf; write them onto string, + * then memmove buf back (that is, remove them from buf). + * + * Return the number of bytes still on the buffer. */ + + peek_from_buf(string, string_len, buf); buf_remove_from_front(buf, string_len); return buf->datalen; } @@ -362,12 +642,13 @@ int fetch_from_buf_http(buf_t *buf, size_t headerlen, bodylen, contentlen; assert_buf_ok(buf); + buf_normalize(buf); - headers = buf->mem; if (buf_nul_terminate(buf)<0) { log_fn(LOG_WARN,"Couldn't nul-terminate buffer"); return -1; } + headers = buf->start; body = strstr(headers,"\r\n\r\n"); if (!body) { log_fn(LOG_DEBUG,"headers not all here yet."); @@ -412,14 +693,14 @@ int fetch_from_buf_http(buf_t *buf, /* all happy. copy into the appropriate places, and return 1 */ if (headers_out) { *headers_out = tor_malloc(headerlen+1); - memcpy(*headers_out,buf->mem,headerlen); + memcpy(*headers_out,buf->start,headerlen); (*headers_out)[headerlen] = 0; /* null terminate it */ } if (body_out) { tor_assert(body_used); *body_used = bodylen; *body_out = tor_malloc(bodylen+1); - memcpy(*body_out,buf->mem+headerlen,bodylen); + memcpy(*body_out,buf->start+headerlen,bodylen); (*body_out)[bodylen] = 0; /* null terminate it */ } buf_remove_from_front(buf, headerlen+bodylen); @@ -459,16 +740,18 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { if (buf->datalen < 2) /* version and another byte */ return 0; - switch (*(buf->mem)) { /* which version of socks? */ + buf_normalize(buf); + + switch (*(buf->start)) { /* which version of socks? */ case 5: /* socks5 */ if (req->socks_version != 5) { /* we need to negotiate a method */ - unsigned char nummethods = (unsigned char)*(buf->mem+1); + unsigned char nummethods = (unsigned char)*(buf->start+1); tor_assert(!req->socks_version); if (buf->datalen < 2u+nummethods) return 0; - if (!nummethods || !memchr(buf->mem+2, 0, nummethods)) { + if (!nummethods || !memchr(buf->start+2, 0, nummethods)) { log_fn(LOG_WARN,"socks5: offered methods don't include 'no auth'. Rejecting."); req->replylen = 2; /* 2 bytes of response */ req->reply[0] = 5; @@ -488,7 +771,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { log_fn(LOG_DEBUG,"socks5: checking request"); if (buf->datalen < 8) /* basic info plus >=2 for addr plus 2 for port */ return 0; /* not yet */ - req->command = (unsigned char) *(buf->mem+1); + req->command = (unsigned char) *(buf->start+1); if (req->command != SOCKS_COMMAND_CONNECT && req->command != SOCKS_COMMAND_RESOLVE) { /* not a connect or resolve? we don't support it. */ @@ -496,13 +779,13 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { req->command); return -1; } - switch (*(buf->mem+3)) { /* address type */ + switch (*(buf->start+3)) { /* address type */ case 1: /* IPv4 address */ log_fn(LOG_DEBUG,"socks5: ipv4 address type"); if (buf->datalen < 10) /* ip/port there? */ return 0; /* not yet */ - destip = ntohl(*(uint32_t*)(buf->mem+4)); + destip = ntohl(*(uint32_t*)(buf->start+4)); in.s_addr = htonl(destip); tor_inet_ntoa(&in,tmpbuf,sizeof(tmpbuf)); if (strlen(tmpbuf)+1 > MAX_SOCKS_ADDR_LEN) { @@ -511,7 +794,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { return -1; } strlcpy(req->address,tmpbuf,sizeof(req->address)); - req->port = ntohs(*(uint16_t*)(buf->mem+8)); + req->port = ntohs(*(uint16_t*)(buf->start+8)); buf_remove_from_front(buf, 10); if (!have_warned_about_unsafe_socks) { log_fn(LOG_WARN,"Your application (using socks5 on port %d) is giving Tor only an IP address. Applications that do DNS resolves themselves may leak information. Consider using Socks4A (e.g. via privoxy or socat) instead.", req->port); @@ -520,7 +803,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { return 1; case 3: /* fqdn */ log_fn(LOG_DEBUG,"socks5: fqdn address type"); - len = (unsigned char)*(buf->mem+4); + len = (unsigned char)*(buf->start+4); if (buf->datalen < 7u+len) /* addr/port there? */ return 0; /* not yet */ if (len+1 > MAX_SOCKS_ADDR_LEN) { @@ -528,13 +811,13 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { len+1,MAX_SOCKS_ADDR_LEN); return -1; } - memcpy(req->address,buf->mem+5,len); + memcpy(req->address,buf->start+5,len); req->address[len] = 0; - req->port = ntohs(get_uint16(buf->mem+5+len)); + req->port = ntohs(get_uint16(buf->start+5+len)); buf_remove_from_front(buf, 5+len+2); return 1; default: /* unsupported */ - log_fn(LOG_WARN,"socks5: unsupported address type %d. Rejecting.",*(buf->mem+3)); + log_fn(LOG_WARN,"socks5: unsupported address type %d. Rejecting.",*(buf->start+3)); return -1; } tor_assert(0); @@ -546,7 +829,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { if (buf->datalen < SOCKS4_NETWORK_LEN) /* basic info available? */ return 0; /* not yet */ - req->command = (unsigned char) *(buf->mem+1); + req->command = (unsigned char) *(buf->start+1); if (req->command != SOCKS_COMMAND_CONNECT && req->command != SOCKS_COMMAND_RESOLVE) { /* not a connect or resolve? we don't support it. */ @@ -555,7 +838,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { return -1; } - req->port = ntohs(*(uint16_t*)(buf->mem+2)); + req->port = ntohs(*(uint16_t*)(buf->start+2)); destip = ntohl(*(uint32_t*)(buf->mem+4)); if ((!req->port && req->command!=SOCKS_COMMAND_RESOLVE) || !destip) { log_fn(LOG_WARN,"socks4: Port or DestIP is zero. Rejecting."); @@ -574,13 +857,13 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { socks4_prot = socks4; } - next = memchr(buf->mem+SOCKS4_NETWORK_LEN, 0, + next = memchr(buf->start+SOCKS4_NETWORK_LEN, 0, buf->datalen-SOCKS4_NETWORK_LEN); if (!next) { log_fn(LOG_DEBUG,"socks4: Username not here yet."); return 0; } - tor_assert(next < buf->mem+buf->datalen); + tor_assert(next < buf->start+buf->datalen); startaddr = NULL; if (socks4_prot != socks4a && !have_warned_about_unsafe_socks) { @@ -588,12 +871,12 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { // have_warned_about_unsafe_socks = 1; // (for now, warn every time) } if (socks4_prot == socks4a) { - if (next+1 == buf->mem+buf->datalen) { + if (next+1 == buf->start+buf->datalen) { log_fn(LOG_DEBUG,"socks4: No part of destaddr here yet."); return 0; } startaddr = next+1; - next = memchr(startaddr, 0, buf->mem+buf->datalen-startaddr); + next = memchr(startaddr, 0, buf->start+buf->datalen-startaddr); if (!next) { log_fn(LOG_DEBUG,"socks4: Destaddr not all here yet."); return 0; @@ -602,12 +885,12 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { log_fn(LOG_WARN,"socks4: Destaddr too long. Rejecting."); return -1; } - tor_assert(next < buf->mem+buf->datalen); + tor_assert(next < buf->start+buf->datalen); } log_fn(LOG_DEBUG,"socks4: Everything is here. Success."); strlcpy(req->address, startaddr ? startaddr : tmpbuf, sizeof(req->address)); - buf_remove_from_front(buf, next-buf->mem+1); /* next points to the final \0 on inbuf */ + buf_remove_from_front(buf, next-buf->start+1); /* next points to the final \0 on inbuf */ return 1; case 'G': /* get */ @@ -639,7 +922,7 @@ int fetch_from_buf_socks(buf_t *buf, socks_request_t *req) { /* fall through */ default: /* version is not socks4 or socks5 */ log_fn(LOG_WARN,"Socks version %d not recognized. (Tor is not an http proxy.)", - *(buf->mem)); + *(buf->start)); return -1; } } @@ -661,6 +944,7 @@ int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out, { uint32_t msglen; uint16_t type; + char tmp[10]; tor_assert(buf); tor_assert(len_out); @@ -670,23 +954,24 @@ int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out, if (buf->datalen < 4) return 0; - msglen = ntohs(get_uint16(buf->mem)); + peek_from_buf(tmp, 4, buf); + + msglen = ntohs(get_uint16(tmp)); if (buf->datalen < 4 + (unsigned)msglen) return 0; - type = ntohs(get_uint16(buf->mem+2)); + type = ntohs(get_uint16(tmp+2)); if (type != CONTROL_CMD_FRAGMENTHEADER) { *len_out = msglen; *type_out = type; + buf_remove_from_front(buf, 4); if (msglen) { *body_out = tor_malloc(msglen+1); - memcpy(*body_out, buf->mem+4, msglen); + fetch_from_buf(*body_out, msglen, buf); (*body_out)[msglen] = '\0'; } else { *body_out = NULL; } - buf_remove_from_front(buf, 4+msglen); - return 1; } else { uint32_t totallen, sofar; @@ -695,8 +980,9 @@ int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out, /* 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)); + peek_from_buf(tmp, 10, buf); + type = htons(get_uint16(tmp+4)); + totallen = htonl(get_uint32(tmp+6)); if (totallen < 65536) return -1; @@ -706,8 +992,9 @@ int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out, /* Count how much data is really here. */ sofar = msglen-6; - cp = buf->mem+4+msglen; - endp = buf->mem+buf->datalen; + cp = buf->start+4+msglen; + endp = buf->start+buf->datalen; + /* XXXXX!!!!!! This will not handle fragmented messages right now. */ while (sofar < totallen) { if ((endp-cp)<4) return 0; /* Fragment header not all here. */ @@ -745,7 +1032,6 @@ int fetch_from_buf_control(buf_t *buf, uint32_t *len_out, uint16_t *type_out, return 1; } - } /** Log an error and exit if <b>buf</b> is corrupted. @@ -756,4 +1042,8 @@ void assert_buf_ok(buf_t *buf) tor_assert(buf->magic == BUFFER_MAGIC); tor_assert(buf->mem); tor_assert(buf->datalen <= buf->len); +#ifdef SENTINALS + tor_assert(get_uint32(buf->mem - 4) == START_MAGIC); + tor_assert(get_uint32(buf->mem + buf->len) == END_MAGIC); +#endif } diff --git a/src/or/test.c b/src/or/test.c index 0e1c473010..2a568b54d1 100644 --- a/src/or/test.c +++ b/src/or/test.c @@ -132,26 +132,122 @@ test_buffers(void) { if (!(buf = buf_new())) test_fail(); - test_eq(buf_capacity(buf), 512*1024); + test_eq(buf_capacity(buf), 4096); test_eq(buf_datalen(buf), 0); /**** - * read_to_buf - ****/ - s = open(get_fname("data"), O_WRONLY|O_CREAT|O_TRUNC, 0600); + * General pointer frobbing + */ for (j=0;j<256;++j) { str[j] = (char)j; } + write_to_buf(str, 256, buf); + write_to_buf(str, 256, buf); + test_eq(buf_datalen(buf), 512); + fetch_from_buf(str2, 200, buf); + test_memeq(str, str2, 200); + test_eq(buf_datalen(buf), 312); + memset(str2, 0, sizeof(str2)); + + fetch_from_buf(str2, 256, buf); + test_memeq(str+200, str2, 56); + test_memeq(str, str2+56, 200); + test_eq(buf_datalen(buf), 56); + memset(str2, 0, sizeof(str2)); + /* Okay, now we should be 512 bytes into the 4096-byte buffer. If we add + * another 3584 bytes, we hit the end. */ + for(j=0;j<15;++j) { + write_to_buf(str, 256, buf); + } + assert_buf_ok(buf); + test_eq(buf_datalen(buf), 3896); + fetch_from_buf(str2, 56, buf); + test_eq(buf_datalen(buf), 3840); + test_memeq(str+200, str2, 56); + for(j=0;j<15;++j) { + memset(str2, 0, sizeof(str2)); + fetch_from_buf(str2, 256, buf); + test_memeq(str, str2, 256); + } + test_eq(buf_datalen(buf), 0); + buf_free(buf); + + /* Okay, now make sure growing can work. */ + buf = buf_new_with_capacity(16); + test_eq(buf_capacity(buf), 16); + write_to_buf(str+1, 255, buf); + test_eq(buf_capacity(buf), 256); + fetch_from_buf(str2, 254, buf); + test_memeq(str+1, str2, 254); + printf("%d, %d, %d\n", (int)buf, buf_capacity(buf), buf_datalen(buf)); + test_eq(buf_capacity(buf), 256); + assert_buf_ok(buf); + write_to_buf(str, 32, buf); + test_eq(buf_capacity(buf), 256); + assert_buf_ok(buf); + write_to_buf(str, 256, buf); + assert_buf_ok(buf); + test_eq(buf_capacity(buf), 512); + test_eq(buf_datalen(buf), 33+256); + fetch_from_buf(str2, 33, buf); + test_eq(*str2, str[255]); + + test_memeq(str2+1, str, 32); + test_eq(buf_capacity(buf), 512); + test_eq(buf_datalen(buf), 256); + fetch_from_buf(str2, 256, buf); + test_memeq(str, str2, 256); + + /* now try shrinking: case 1. */ + buf_free(buf); + buf = buf_new_with_capacity(33668); + for (j=0;j<67;++j) { + write_to_buf(str,255, buf); + } + test_eq(buf_capacity(buf), 33668); + test_eq(buf_datalen(buf), 17085); + for (j=0; j < 40; ++j) { + fetch_from_buf(str2, 255,buf); + test_memeq(str2, str, 255); + } + + /* now try shrinking: case 2. */ + buf_free(buf); + buf = buf_new_with_capacity(33668); + for (j=0;j<67;++j) { + write_to_buf(str,255, buf); + } + for (j=0; j < 20; ++j) { + fetch_from_buf(str2, 255,buf); + test_memeq(str2, str, 255); + } + for (j=0;j<80;++j) { + write_to_buf(str,255, buf); + } + test_eq(buf_capacity(buf),33668); + for (j=0; j < 120; ++j) { + fetch_from_buf(str2, 255,buf); + test_memeq(str2, str, 255); + } + + + /**** + * read_to_buf + ****/ + s = open(get_fname("data"), O_WRONLY|O_CREAT|O_TRUNC, 0600); write(s, str, 256); close(s); s = open(get_fname("data"), O_RDONLY, 0); eof = 0; + errno = 0; /* XXXX */ i = read_to_buf(s, 10, buf, &eof); - test_eq(buf_capacity(buf), 512*1024); - test_eq(buf_datalen(buf), 10); - test_eq(eof, 0); + printf("%s\n", strerror(errno)); test_eq(i, 10); + test_eq(eof, 0); + test_eq(buf_capacity(buf), 4096); + test_eq(buf_datalen(buf), 10); + test_memeq(str, (char*)_buf_peek_raw_buffer(buf), 10); /* Test reading 0 bytes. */ @@ -195,49 +291,6 @@ test_buffers(void) { test_eq(buf_datalen(buf), 256-6-32); test_eq(eof, 1); - close(s); - - /**** - * fetch_from_buf - ****/ - memset(str2, 255, 256); - test_eq(246, fetch_from_buf(str2, 10, buf)); - test_memeq(str2, str, 10); - test_memeq(str+10,(char*)_buf_peek_raw_buffer(buf),246); - test_eq(buf_datalen(buf),246); - - test_eq(0, fetch_from_buf(str2, 246, buf)); - test_memeq(str2, str+10, 246); - test_eq(buf_capacity(buf),MAX_BUF_SIZE); - test_eq(buf_datalen(buf),0); - - /**** - * write_to_buf - ****/ - memset((char *)_buf_peek_raw_buffer(buf), (int)'-', 256); - i = write_to_buf("Hello world", 11, buf); - test_eq(i, 11); - test_eq(buf_datalen(buf), 11); - test_memeq((char*)_buf_peek_raw_buffer(buf), "Hello world", 11); - i = write_to_buf("XYZZY", 5, buf); - test_eq(i, 16); - test_eq(buf_datalen(buf), 16); - test_memeq((char*)_buf_peek_raw_buffer(buf), "Hello worldXYZZY", 16); - /* Test when buffer is overfull. */ -#if 0 - buflen = 18; - test_eq(-1, write_to_buf("This string will not fit.", 25, - &buf, &buflen, &buf_datalen)); - test_eq(buf_datalen, 16); - test_memeq(buf, "Hello worldXYZZY--", 18); - buflen = MAX_BUF_SIZE; -#endif - - /**** - * flush_buf - ****/ - /* XXXX Needs tests. */ - buf_free(buf); } @@ -1319,8 +1372,8 @@ main(int c, char**v) { printf("Running Tor unit tests on %s\n", get_uname()); -// puts("========================== Buffers ========================="); - if (0) test_buffers(); + puts("========================== Buffers ========================="); + test_buffers(); puts("\n========================== Crypto =========================="); // add_stream_log(LOG_DEBUG, LOG_ERR, "<stdout>", stdout); test_crypto(); |