aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2004-10-07 21:37:06 +0000
committerNick Mathewson <nickm@torproject.org>2004-10-07 21:37:06 +0000
commit6c970aec946942d10996cda55798859dbc90f105 (patch)
tree90681cc5e09b7454250d5c79bb336bebc89c9563
parentc7e8c2098abd93bf24439f869016c385067e5e5b (diff)
downloadtor-6c970aec946942d10996cda55798859dbc90f105.tar
tor-6c970aec946942d10996cda55798859dbc90f105.tar.gz
Turn tor_strpartion into a swiss-army-knife function, so it can terminate or not-terminate appropriately.
svn:r2429
-rw-r--r--src/common/crypto.c9
-rw-r--r--src/common/util.c41
-rw-r--r--src/common/util.h6
-rw-r--r--src/or/test.c14
4 files changed, 57 insertions, 13 deletions
diff --git a/src/common/crypto.c b/src/common/crypto.c
index 9f70b7ab5..1692aac23 100644
--- a/src/common/crypto.c
+++ b/src/common/crypto.c
@@ -514,9 +514,8 @@ crypto_pk_env_t *crypto_pk_DER64_decode_public_key(const char *in)
}
/* base64_decode doesn't work unless we insert linebreaks every 64
* characters. how dumb. */
- if (tor_strpartition(partitioned, sizeof(partitioned), in, "\n", 64))
- return NULL;
- if (strlcat(partitioned, "\n",sizeof(partitioned))>=sizeof(partitioned))
+ if (tor_strpartition(partitioned, sizeof(partitioned), in, "\n", 64,
+ ALWAYS_TERMINATE))
return NULL;
len = base64_decode(buf, sizeof(buf), partitioned, strlen(partitioned));
if (len<0) {
@@ -919,7 +918,9 @@ crypto_pk_get_fingerprint(crypto_pk_env_t *pk, char *fp_out, int add_space)
}
base16_encode(hexdigest,sizeof(hexdigest),digest,DIGEST_LEN);
if (add_space) {
- tor_strpartition(fp_out, FINGERPRINT_LEN+1, hexdigest, " ", 4);
+ if (tor_strpartition(fp_out, FINGERPRINT_LEN+1, hexdigest, " ", 4,
+ NEVER_TERMINATE)<0)
+ return -1;
} else {
strcpy(fp_out, hexdigest);
}
diff --git a/src/common/util.c b/src/common/util.c
index aa350b2ab..faa87c9c8 100644
--- a/src/common/util.c
+++ b/src/common/util.c
@@ -219,25 +219,56 @@ int tor_strstrip(char *s, const char *strip)
/** Set the <b>dest_len</b>-byte buffer <b>buf</b> to contain the
* string <b>s</b>, with the string <b>insert</b> inserted after every
* <b>n</b> characters. Return 0 on success, -1 on failure.
+ *
+ * If <b>rule</b> is ALWAYS_TERMINATE, then always end the string with
+ * <b>insert</b>, even if its length is not a multiple of <b>n</b>. If
+ * <b>rule</b> is NEVER_TERMINATE, then never end the string with
+ * <b>insert</b>, even if its length <i>is</i> a multiple of <b>n</b>.
+ * If <b>rule</b> is TERMINATE_IF_EVEN, then end the string with <b>insert</b>
+ * exactly when its length <i>is</i> a multiple of <b>n</b>.
*/
int tor_strpartition(char *dest, size_t dest_len,
- const char *s, const char *insert, size_t n)
+ const char *s, const char *insert, size_t n,
+ part_finish_rule_t rule)
{
tor_assert(s && insert && n > 0);
+ char *destp;
int len_in, len_out, len_ins;
+ int is_even;
len_in = strlen(s);
len_ins = strlen(insert);
len_out = len_in + (len_in/n)*len_ins;
+ is_even = (len_in%n) == 0;
+ switch(rule)
+ {
+ case ALWAYS_TERMINATE:
+ if (!is_even) len_out += len_ins;
+ break;
+ case NEVER_TERMINATE:
+ if (is_even && len_in) len_out -= len_ins;
+ break;
+ case TERMINATE_IF_EVEN:
+ break;
+ }
if (dest_len < len_out+1)
return -1;
+ destp = dest;
while(len_in) {
- strncpy(dest, s, n);
+ strncpy(destp, s, n);
len_in -= n;
- if (len_in < 0) break;
- strcpy(dest+n, insert);
+ if (len_in < 0) {
+ if (rule == ALWAYS_TERMINATE)
+ strcpy(destp+n+len_in,insert);
+ break;
+ } else if (len_in == 0 && rule == NEVER_TERMINATE) {
+ *(destp+n) = '\0';
+ break;
+ }
+ strcpy(destp+n, insert);
s += n;
- dest += n+len_ins;
+ destp += n+len_ins;
}
+ tor_assert(len_out == strlen(dest));
return 0;
}
diff --git a/src/common/util.h b/src/common/util.h
index c579c92a9..2b13f92b4 100644
--- a/src/common/util.h
+++ b/src/common/util.h
@@ -88,8 +88,12 @@ char *tor_strndup(const char *s, size_t n);
void tor_strlower(char *s);
int strcmpstart(const char *s1, const char *s2);
int tor_strstrip(char *s, const char *strip);
+typedef enum {
+ ALWAYS_TERMINATE, NEVER_TERMINATE, TERMINATE_IF_EVEN
+} part_finish_rule_t;
int tor_strpartition(char *dest, size_t dest_len,
- const char *s, const char *insert, size_t n);
+ const char *s, const char *insert, size_t n,
+ part_finish_rule_t rule);
/* Some platforms segfault when you try to access a multi-byte type
diff --git a/src/or/test.c b/src/or/test.c
index ed89df1f3..8b4917f38 100644
--- a/src/or/test.c
+++ b/src/or/test.c
@@ -620,10 +620,18 @@ test_util() {
test_streq(buf, "Testing123");
/* Test tor_strpartition() */
- test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefg", "##", 3));
+ test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefg", "##", 3,
+ TERMINATE_IF_EVEN));
test_streq(buf, "abc##def##g");
- test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefghi", "##", 3));
+ test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefg", "##", 3,
+ ALWAYS_TERMINATE));
+ test_streq(buf, "abc##def##g##");
+ test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefghi", "##", 3,
+ TERMINATE_IF_EVEN));
test_streq(buf, "abc##def##ghi##");
+ test_assert(! tor_strpartition(buf, sizeof(buf), "abcdefghi", "##", 3,
+ NEVER_TERMINATE));
+ test_streq(buf, "abc##def##ghi");
/* XXXX test older functions. */
smartlist_free(sl);
@@ -915,7 +923,7 @@ test_dir_format()
strcat(buf2, "\n"
"published 1970-01-01 00:00:00\n"
"opt fingerprint ");
- crypto_pk_get_fingerprint(pk2, fingerprint, 1);
+ test_assert(!crypto_pk_get_fingerprint(pk2, fingerprint, 1));
strcat(buf2, fingerprint);
strcat(buf2, "\nopt uptime 0\n"
/* XXX the "0" above is hardcoded, but even if we made it reflect