X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~ianmdlvl/git?p=secnet.git;a=blobdiff_plain;f=util.c;h=241196227a80cd71e655daea6d5368fabadf37dd;hp=703606172e6b550dd16dc81926ed4359f6eecc5c;hb=7dae56bc17f51e875b436ec63e035be12c4a6fcf;hpb=2d80199d7bc25b4c6e4a5ac986f8902770e82d96 diff --git a/util.c b/util.c index 7036061..2411962 100644 --- a/util.c +++ b/util.c @@ -11,7 +11,7 @@ * * secnet is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version d of the License, or + * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * secnet is distributed in the hope that it will be useful, but @@ -93,31 +93,23 @@ void *safe_malloc_ary(size_t size, size_t count, const char *message) { return safe_realloc_ary(0,size,count,message); } -/* Convert a buffer into its MP_INT representation */ -void read_mpbin(MP_INT *a, uint8_t *bin, int binsize) +void hex_encode(const uint8_t *bin, int binsize, char *buff) { - char *buff; int i; - buff=safe_malloc(binsize*2 + 1,"read_mpbin"); - for (i=0; i> 4]; buff[i*2+1]=hexdigits[(bin[i] & 0xf)]; } buff[binsize*2]=0; - - mpz_set_str(a, buff, 16); - free(buff); } -/* Convert a MP_INT into a hex string */ -char *write_mpstring(MP_INT *a) +string_t hex_encode_alloc(const uint8_t *bin, int binsize) { char *buff; - buff=safe_malloc(mpz_sizeinbase(a,16)+2,"write_mpstring"); - mpz_get_str(buff, 16, a); + buff=safe_malloc(hex_encode_size(binsize),"hex_encode"); + hex_encode(bin,binsize,buff); return buff; } @@ -150,27 +142,54 @@ static uint8_t hexval(uint8_t c) return -1; } -/* Convert a MP_INT into a buffer; return length; truncate if necessary */ -int32_t write_mpbin(MP_INT *a, uint8_t *buffer, int32_t buflen) +bool_t hex_decode(uint8_t *buffer, int32_t buflen, int32_t *outlen, + cstring_t hb, bool_t allow_odd_nibble) { - char *hb; - int i,j,l; - - if (buflen==0) return 0; - hb=write_mpstring(a); - - l=strlen(hb); - i=0; j=0; + int i = 0, j = 0, l = strlen(hb), hi, lo; + bool_t ok = False; + + if (!l || !buflen) { ok = !l; goto done; } if (l&1) { /* The number starts with a half-byte */ - buffer[i++]=hexval(hb[j++]); + if (!allow_odd_nibble) goto done; + lo = hexval(hb[j++]); if (lo < 0) goto done; + buffer[i++] = lo; } - for (; hb[j] && i 0) { + c = *s++; + if (c >= ' ' && c <= 126 && c != '\\' && c != '"' && c != '\'') { + if (!buf_remaining_space(buf)) break; + buf->start[buf->size++] = c; + continue; + } + char quoted[5]; + quoted[0] = '\\'; + quoted[2] = 0; + switch (c) { + case '\n': quoted[1] = 'n'; break; + case '\r': quoted[1] = 'r'; break; + case '\t': quoted[1] = 't'; break; + case '\\': case '"': case '\'': quoted[1] = c; break; + default: sprintf(quoted, "\\x%02x", (unsigned)c); + } + truncmsg_add_string(buf, quoted); + } +} +const char *truncmsg_terminate(const struct buffer_if *buf) +{ + if (buf_remaining_space(buf)) { + buf->start[buf->size] = 0; + } else { + assert(buf->size >= 4); + strcpy(buf->start + buf->size - 4, "..."); + } + return buf->start; +} + +void priomsg_new(struct priomsg *pm, int32_t maxlen) +{ + buffer_new(&pm->m, maxlen); + pm->prio = INT_MIN; +} +void priomsg_reset(struct priomsg *pm) +{ + buffer_init(&pm->m, 0); + pm->prio = INT_MIN; +} +bool_t priomsg_update_p(struct priomsg *pm, int prio) +{ + if (!pm) return False; + if (prio <= pm->prio) return False; + buffer_init(&pm->m, 0); + pm->prio = prio; + return True; +} +const char *priomsg_getmessage(const struct priomsg *pm, const char *defmsg) +{ + if (pm->prio >= INT_MIN) + return truncmsg_terminate(&pm->m); + else + return defmsg; +} + + void buffer_new(struct buffer_if *buf, int32_t len) { buf->free=True; buf->owner=NULL; - buf->flags=0; buf->loc.file=NULL; buf->loc.line=0; buf->size=0; @@ -361,7 +444,6 @@ void buffer_readonly_view(struct buffer_if *buf, const void *data, int32_t len) { buf->free=False; buf->owner="READONLY"; - buf->flags=0; buf->loc.file=NULL; buf->loc.line=0; buf->size=buf->alloclen=len; @@ -467,6 +549,14 @@ int consttime_memeq(const void *s1in, const void *s2in, size_t n) return accumulator; } +void hash_hash(const struct hash_if *hashi, const void *msg, int32_t len, + uint8_t *digest) { + uint8_t hst[hashi->slen]; + hashi->init(hst); + hashi->update(hst,msg,len); + hashi->final(hst,digest); +} + void util_module(dict_t *dict) { add_closure(dict,"sysbuffer",buffer_apply); @@ -521,7 +611,7 @@ void string_item_to_iaddr(const item_t *item, uint16_t port, union iaddr *ia, #ifndef CONFIG_IPV6 ia->sin.sin_family=AF_INET; - ia->sin.sin_addr.s_addr=string_item_to_ipaddr(item,desc); + ia->sin.sin_addr.s_addr=htonl(string_item_to_ipaddr(item,desc)); ia->sin.sin_port=htons(port); #else /* CONFIG_IPV6 => we have adns_text2addr */ @@ -565,7 +655,7 @@ const char *iaddr_to_string(const union iaddr *ia) int r = adns_addr2text(&ia->sa, 0, addrbuf, &addrbuflen, &port); if (r) { - const char fmt[]= "scoped IPv6 addr, error: %.*s"; + const char fmt[]= "bad addr, error: %.*s"; sprintf(addrbuf, fmt, (int)(ADNS_ADDR2TEXT_BUFLEN - sizeof(fmt)) /* underestimate */, strerror(r));