#include #include #include #include #include #include #include #include #include #include #include #include #include #include "lookup.h" #include "stdio_impl.h" #include "syscall.h" #include "locale_impl.h" enum { base = 36, tmin = 1, tmax = 26, skew = 38, damp = 700, initial_bias = 72, initial_n = 128, }; static int is_valid_hostname(const char *host) { const unsigned char *s; if (strnlen(host, 255)-1 >= 254 || mbstowcs(0, host, 0) == -1) return 0; for (s=(void *)host; *s>=0x80 || *s=='.' || *s=='-' || isalnum(*s); s++); return !*s; } static unsigned int adapt(unsigned int delta, unsigned int numpoints, int firsttime) { unsigned int k = 0; delta /= firsttime ? damp : 2; delta += delta / numpoints; while (delta > ((base - tmin) * tmax) / 2) { delta /= base - tmin; k += base; } return k + ((base - tmin + 1) * delta) / (delta + skew); } static ssize_t punyenc(char *dst, const char *src, size_t len, size_t max) { static const char *const tbl = "abcdefghijklmnopqrstuvwxyz0123456789"; const unsigned char *usrc = (void *)src; unsigned int codepoints = 0; unsigned int dlen = 0; unsigned int si, mi; unsigned int n = initial_n; unsigned int delta = 0; unsigned int bias = initial_bias; unsigned int h, b; for (si = 0; si < len; ++si) { if (usrc[si] < 0x80) { if (dlen == max) return -1; dst[dlen++] = src[si]; } else if ((usrc[si] & 0xC0) == 0xC0) { ++codepoints; } } codepoints += dlen; h = b = dlen; if (dlen) { if (dlen == max) return -1; dst[dlen++] = '-'; } while (h < codepoints) { unsigned int m = (unsigned int)-1; unsigned int c; wchar_t wc; for (mi = 0; mi < len; ) { mi += mbtowc(&wc, src + mi, len - mi); c = (unsigned int)wc; if (c >= n && c < m) m = c; } if (((unsigned int)-1 - delta) / (h + 1) < m - n) return -1; delta += (m - n) * (h + 1); n = m; for (mi = 0; mi < len; ) { mi += mbtowc(&wc, src + mi, len - mi); c = (unsigned int)wc; if (c < n /* || c < 0x80 not necessary*/) if (++delta == 0) return -1; if (c == n) { unsigned int q = delta; unsigned int k; for (k = base; ; k += base) { unsigned int t; if (k <= bias + tmin) { t = tmin; } else if (k >= bias + tmax) { t = tmax; } else { t = k - bias; } if (q < t) break; if (dlen == max) return -1; dst[dlen++] = tbl[t + ((q - t) % (base - t))]; q = (q - t) / (base - t); } if (dlen == max) return -1; dst[dlen++] = tbl[q]; bias = adapt(delta, h + 1, h == b); delta = 0; ++h; } } ++delta; ++n; } return dlen; } static ssize_t idnaenc(char dst[static 256], const char *src) { size_t left = strlen(src); size_t olen = 0; while (left) { const char *dot; size_t len, i; int basic = 1; dot = memchr(src, '.', left); if (!dot) { dot = src + left; } len = dot - src; if (len == 0) { return -1; } left -= len + !!*dot; for (i = 0; i < len; ) { unsigned int c; wchar_t wc; int n = mbtowc(&wc, src + i, len - i); c = (n <= 0) ? 0 : (unsigned int)wc; if (c < 0x80) { if (!isalnum(c) && !(i > 0 && c == '-')) return -1; } else { if (c >= 0x7F && c <= 0x9F) return -1; basic = 0; } i += n; } if (basic) { if (len > 63 || len > 254 - olen) return -1; for (i = 0; i < len; ++i) dst[olen + i] = tolower(src[i]); olen += len; } else { ssize_t r; size_t max; if (olen >= 254 - 4) return -1; max = 254 - 4 - olen; if (max > 63 - 4) max = 63 - 4; memcpy(dst + olen, "xn--", 4); r = punyenc(dst + olen + 4, src, len, max); if (r <= 0) return -1; olen += r + 4; } if (olen == 255 || !*dot && olen == 254) return -1; if (*dot) dst[olen++] = *dot; src = dot + !!*dot; } if (olen == 0) return -1; dst[olen] = 0; return olen; } static int name_from_null(struct address buf[static 2], const char *name, int family, int flags) { int cnt = 0; if (name) return 0; if (flags & AI_PASSIVE) { if (family != AF_INET6) buf[cnt++] = (struct address){ .family = AF_INET }; if (family != AF_INET) buf[cnt++] = (struct address){ .family = AF_INET6 }; } else { if (family != AF_INET6) buf[cnt++] = (struct address){ .family = AF_INET, .addr = { 127,0,0,1 } }; if (family != AF_INET) buf[cnt++] = (struct address){ .family = AF_INET6, .addr = { [15] = 1 } }; } return cnt; } static int name_from_numeric(struct address buf[static 1], const char *name, int family) { return __lookup_ipliteral(buf, name, family); } static int name_from_hosts(struct address buf[static MAXADDRS], char canon[static 256], const char *name, int family) { char line[512]; size_t l = strlen(name); int cnt = 0, badfam = 0; unsigned char _buf[1032]; FILE _f, *f = __fopen_rb_ca("/etc/hosts", &_f, _buf, sizeof _buf); if (!f) switch (errno) { case ENOENT: case ENOTDIR: case EACCES: return 0; default: return EAI_SYSTEM; } while (fgets(line, sizeof line, f) && cnt < MAXADDRS) { char idna[256]; ssize_t r; char *p, *z, c; if ((p=strchr(line, '#'))) *p++='\n', *p=0; /* skip ip address and canonicalize names */ for (p=line; *p && !isspace(*p); p++); while (*p) { for (; *p && isspace(*p); p++); for (z=p; *z && !isspace(*z); z++); c = *z; *z = 0; r = idnaenc(idna, p); *z = c; if (r == l && memcmp(idna, name, l) == 0) break; p = z; } if (!*p) continue; /* Isolate IP address to parse */ for (p=line; *p && !isspace(*p); p++); *p++ = 0; switch (name_from_numeric(buf+cnt, line, family)) { case 1: cnt++; break; case 0: continue; default: badfam = EAI_NONAME; continue; } /* Extract first name as canonical name */ for (; *p && isspace(*p); p++); for (z=p; *z && !isspace(*z); z++); *z = 0; if ((r = idnaenc(idna, p)) > 0) memcpy(canon, idna, r); } __fclose_ca(f); return cnt ? cnt : badfam; } struct dpc_ctx { struct address *addrs; char *canon; int cnt; }; int __dns_parse(const unsigned char *, int, int (*)(void *, int, const void *, int, const void *), void *); int __dn_expand(const unsigned char *, const unsigned char *, const unsigned char *, char *, int); int __res_mkquery(int, const char *, int, int, const unsigned char *, int, const unsigned char*, unsigned char *, int); int __res_msend_rc(int, const unsigned char *const *, const int *, unsigned char *const *, int *, int, const struct resolvconf *); #define RR_A 1 #define RR_CNAME 5 #define RR_AAAA 28 static int dns_parse_callback(void *c, int rr, const void *data, int len, const void *packet) { char tmp[256]; struct dpc_ctx *ctx = c; switch (rr) { case RR_A: if (len != 4) return -1; ctx->addrs[ctx->cnt].family = AF_INET; ctx->addrs[ctx->cnt].scopeid = 0; memcpy(ctx->addrs[ctx->cnt++].addr, data, 4); break; case RR_AAAA: if (len != 16) return -1; ctx->addrs[ctx->cnt].family = AF_INET6; ctx->addrs[ctx->cnt].scopeid = 0; memcpy(ctx->addrs[ctx->cnt++].addr, data, 16); break; case RR_CNAME: if (__dn_expand(packet, (const unsigned char *)packet + 512, data, tmp, sizeof tmp) > 0 && is_valid_hostname(tmp)) strcpy(ctx->canon, tmp); break; } return 0; } static int name_from_dns(struct address buf[static MAXADDRS], char canon[static 256], const char *name, int family, const struct resolvconf *conf) { unsigned char qbuf[2][280], abuf[2][512]; const unsigned char *qp[2] = { qbuf[0], qbuf[1] }; unsigned char *ap[2] = { abuf[0], abuf[1] }; int qlens[2], alens[2]; int i, nq = 0; struct dpc_ctx ctx = { .addrs = buf, .canon = canon }; static const struct { int af; int rr; } afrr[2] = { { .af = AF_INET6, .rr = RR_A }, { .af = AF_INET, .rr = RR_AAAA }, }; for (i=0; i<2; i++) { if (family != afrr[i].af) { qlens[nq] = __res_mkquery(0, name, 1, afrr[i].rr, 0, 0, 0, qbuf[nq], sizeof *qbuf); if (qlens[nq] == -1) return EAI_NONAME; nq++; } } if (__res_msend_rc(nq, qp, qlens, ap, alens, sizeof *abuf, conf) < 0) return EAI_SYSTEM; for (i=0; i=ndots or name ends in * a dot, which is an explicit request for global scope. */ for (dots=l=0; name[l]; l++) if (name[l]=='.') dots++; if (dots >= conf.ndots || name[l-1]=='.') *search = 0; /* This can never happen; the caller already checked length. */ if (l >= 256) return EAI_NONAME; /* Name with search domain appended is setup in canon[]. This both * provides the desired default canonical name (if the requested * name is not a CNAME record) and serves as a buffer for passing * the full requested name to name_from_dns. */ memcpy(canon, name, l); canon[l] = '.'; for (p=search; *p; p=z) { for (; isspace(*p); p++); for (z=p; *z && !isspace(*z); z++); if (z==p) break; if (z-p < 256 - l - 1) { memcpy(canon+l+1, p, z-p); canon[z-p+1+l] = 0; int cnt = name_from_dns(buf, canon, canon, family, &conf); if (cnt) return cnt; } } canon[l] = 0; return name_from_dns(buf, canon, name, family, &conf); } static const struct policy { unsigned char addr[16]; unsigned char len, mask; unsigned char prec, label; } defpolicy[] = { { "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\1", 15, 0xff, 50, 0 }, { "\0\0\0\0\0\0\0\0\0\0\xff\xff", 11, 0xff, 35, 4 }, { "\x20\2", 1, 0xff, 30, 2 }, { "\x20\1", 3, 0xff, 5, 5 }, { "\xfc", 0, 0xfe, 3, 13 }, #if 0 /* These are deprecated and/or returned to the address * pool, so despite the RFC, treating them as special * is probably wrong. */ { "", 11, 0xff, 1, 3 }, { "\xfe\xc0", 1, 0xc0, 1, 11 }, { "\x3f\xfe", 1, 0xff, 1, 12 }, #endif /* Last rule must match all addresses to stop loop. */ { "", 0, 0, 40, 1 }, }; static const struct policy *policyof(const struct in6_addr *a) { int i; for (i=0; ; i++) { if (memcmp(a->s6_addr, defpolicy[i].addr, defpolicy[i].len)) continue; if ((a->s6_addr[defpolicy[i].len] & defpolicy[i].mask) != defpolicy[i].addr[defpolicy[i].len]) continue; return defpolicy+i; } } static int labelof(const struct in6_addr *a) { return policyof(a)->label; } static int scopeof(const struct in6_addr *a) { if (IN6_IS_ADDR_MULTICAST(a)) return a->s6_addr[1] & 15; if (IN6_IS_ADDR_LINKLOCAL(a)) return 2; if (IN6_IS_ADDR_LOOPBACK(a)) return 2; if (IN6_IS_ADDR_SITELOCAL(a)) return 5; return 14; } static int prefixmatch(const struct in6_addr *s, const struct in6_addr *d) { /* FIXME: The common prefix length should be limited to no greater * than the nominal length of the prefix portion of the source * address. However the definition of the source prefix length is * not clear and thus this limiting is not yet implemented. */ unsigned i; for (i=0; i<128 && !((s->s6_addr[i/8]^d->s6_addr[i/8])&(128>>(i%8))); i++); return i; } #define DAS_USABLE 0x40000000 #define DAS_MATCHINGSCOPE 0x20000000 #define DAS_MATCHINGLABEL 0x10000000 #define DAS_PREC_SHIFT 20 #define DAS_SCOPE_SHIFT 16 #define DAS_PREFIX_SHIFT 8 #define DAS_ORDER_SHIFT 0 static int addrcmp(const void *_a, const void *_b) { const struct address *a = _a, *b = _b; return b->sortkey - a->sortkey; } int __lookup_name(struct address buf[static MAXADDRS], char canon[static 256], const char *name, int family, int flags) { locale_t *ploc = &CURRENT_LOCALE, loc = *ploc; char _name[256]; int cnt = 0, i, j; *ploc = UTF8_LOCALE; *canon = 0; if (name) { /* reject empty name and check len so it fits into temp bufs */ ssize_t l; if ((l = idnaenc(_name, name)) <= 0) return EAI_NONAME; memcpy(canon, _name, l+1); name = _name; } /* Procedurally, a request for v6 addresses with the v4-mapped * flag set is like a request for unspecified family, followed * by filtering of the results. */ if (flags & AI_V4MAPPED) { if (family == AF_INET6) family = AF_UNSPEC; else flags -= AI_V4MAPPED; } /* Try each backend until there's at least one result. */ cnt = name_from_null(buf, name, family, flags); if (!cnt) cnt = name_from_numeric(buf, name, family); if (!cnt && !(flags & AI_NUMERICHOST)) { cnt = name_from_hosts(buf, canon, name, family); if (!cnt) cnt = name_from_dns_search(buf, canon, name, family); } *ploc = loc; if (cnt<=0) return cnt ? cnt : EAI_NONAME; /* Filter/transform results for v4-mapped lookup, if requested. */ if (flags & AI_V4MAPPED) { if (!(flags & AI_ALL)) { /* If any v6 results exist, remove v4 results. */ for (i=0; ilabel; int dprec = dpolicy->prec; int prefixlen = 0; int fd = socket(AF_INET6, SOCK_DGRAM|SOCK_CLOEXEC, IPPROTO_UDP); if (fd >= 0) { if (!connect(fd, (void *)&da, sizeof da)) { key |= DAS_USABLE; if (!getsockname(fd, (void *)&sa, &(socklen_t){sizeof sa})) { if (dscope == scopeof(&sa.sin6_addr)) key |= DAS_MATCHINGSCOPE; if (dlabel == labelof(&sa.sin6_addr)) key |= DAS_MATCHINGLABEL; prefixlen = prefixmatch(&sa.sin6_addr, &da.sin6_addr); } } close(fd); } key |= dprec << DAS_PREC_SHIFT; key |= (15-dscope) << DAS_SCOPE_SHIFT; key |= prefixlen << DAS_PREFIX_SHIFT; key |= (MAXADDRS-i) << DAS_ORDER_SHIFT; buf[i].sortkey = key; } qsort(buf, cnt, sizeof *buf, addrcmp); pthread_setcancelstate(cs, 0); return cnt; }