X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;f=src%2Fnetwork%2Fdn_expand.c;h=eac343af5ed4f8142985e214c4c531397678d0b1;hb=51d4669fb97782f6a66606da852b5afd49a08001;hp=01b449bb21224433a85ea9094f34b5e4445cec20;hpb=0b44a0315b47dd8eced9f3b7f31580cf14bbfc01;p=musl diff --git a/src/network/dn_expand.c b/src/network/dn_expand.c index 01b449bb..eac343af 100644 --- a/src/network/dn_expand.c +++ b/src/network/dn_expand.c @@ -1,28 +1,33 @@ #include -#include -#define BITOP(a,b,op) \ - ((a)[(size_t)(b)/(8*sizeof *(a))] op (size_t)1<<((size_t)(b)%(8*sizeof *(a)))) - -int dn_expand(unsigned char *b, unsigned char *pz, unsigned char *p, unsigned char *s, int outlen) +int __dn_expand(const unsigned char *base, const unsigned char *end, const unsigned char *src, char *dest, int space) { - /* Remember jump destinations to detect loops and abort */ - size_t seen[512/8/sizeof(size_t)] = { 0 }; - unsigned char *sz = s + outlen; - if (pz-b > 512) return -1; - 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; + const unsigned char *p = src; + char *dend, *dbegin = dest; + int len = -1, i, j; + if (p==end || space <= 0) return -1; + dend = dest + (space > 254 ? 254 : space); + /* detect reference loop using an iteration counter */ + for (i=0; i < end-base; i+=2) { + /* loop invariants: p= end-base) return -1; + p = base+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; + if (dest != dbegin) *dest++ = '.'; + j = *p++; + if (j >= end-p || j >= dend-dest) return -1; + while (j--) *dest++ = *p++; + } else { + *dest = 0; + if (len < 0) len = p+1-src; + return len; + } } + return -1; } + +weak_alias(__dn_expand, dn_expand);