summaryrefslogtreecommitdiff
path: root/src/network/__dns.c
diff options
context:
space:
mode:
authorRich Felker <dalias@aerifal.cx>2013-08-14 17:58:20 -0400
committerRich Felker <dalias@aerifal.cx>2013-08-14 17:58:20 -0400
commitfcc522c92335783293ac19df318415cd97fbf66b (patch)
tree35de93be8123cb7cd075994087d132e3c8bbb906 /src/network/__dns.c
parentcccc1844be95549e5b6c91ffc1f2c2ba3d3aab16 (diff)
downloadmusl-fcc522c92335783293ac19df318415cd97fbf66b.tar.gz
de-duplicate dn_expand, fix return value and signature, clean up
the duplicate code in dn_expand and its incorrect return values are both results of the history of the code: the version in __dns.c was originally written with no awareness of the legacy resolver API, and was later copy-and-paste duplicated to provide the legacy API. this commit is the first of a series that will restructure the internal dns code to share as much code as possible with the legacy resolver API functions. I have also removed the loop detection logic, since the output buffer length limit naturally prevents loops. in order to avoid long runtime when encountering a loop if the caller provided a ridiculously long buffer, the caller-provided length is clamped at the maximum dns name length.
Diffstat (limited to 'src/network/__dns.c')
-rw-r--r--src/network/__dns.c30
1 files changed, 3 insertions, 27 deletions
diff --git a/src/network/__dns.c b/src/network/__dns.c
index 372a5871..8f3c6370 100644
--- a/src/network/__dns.c
+++ b/src/network/__dns.c
@@ -195,32 +195,7 @@ int __dns_query(unsigned char *r, const void *a, int family, int ptr)
return __dns_doqueries(r, a, rr, rrcnt);
}
-
-#define BITOP(a,b,op) \
- ((a)[(size_t)(b)/(8*sizeof *(a))] op (size_t)1<<((size_t)(b)%(8*sizeof *(a))))
-
-static int decname(char *s, const unsigned char *b, const unsigned char *p)
-{
- /* Remember jump destinations to detect loops and abort */
- size_t seen[PACKET_MAX/8/sizeof(size_t)] = { 0 };
- char *sz = s + HOST_NAME_MAX;
- const unsigned char *pz = b+512;
- for (;;) {
- if (p>=pz) return -1;
- else if (*p&0xc0) {
- int j = (p[0]&1) | p[1];
- if (BITOP(seen, j, &)) return -1;
- BITOP(seen, j, |=);
- p = b + j;
- } else if (*p) {
- if (p+*p+1>=pz || s+*p>=sz) return -1;
- memcpy(s, p+1, *p);
- s += *p+1;
- p += *p+1;
- s[-1] = *p ? '.' : 0;
- } else return 0;
- }
-}
+int __dn_expand(const unsigned char *, const unsigned char *, const unsigned char *, char *, int);
int __dns_get_rr(void *dest, size_t stride, size_t maxlen, size_t limit, const unsigned char *r, int rr, int dec)
{
@@ -249,7 +224,8 @@ int __dns_get_rr(void *dest, size_t stride, size_t maxlen, size_t limit, const u
len = p[8]*256 + p[9];
if (p+len > r+512) return -1;
if (p[1]==rr && len <= maxlen) {
- if (dec && decname(tmp, r, p+10)<0) return -1;
+ if (dec && __dn_expand(r, r+512, p+10, tmp, sizeof tmp)<0)
+ return -1;
if (dest && limit) {
if (dec) strcpy(dest, tmp);
else memcpy(dest, p+10, len);