iov[1].iov_len = len;
DBG(1, "Q <- %s %s\n", dns_type_string(type), question);
- if (interface_send_packet(iface, iov, ARRAY_SIZE(iov)) < 0)
- perror("failed to send question :");
+ if (interface_send_packet(iface, NULL, iov, ARRAY_SIZE(iov)) < 0)
+ perror("failed to send question");
}
}
void
-dns_send_answer(struct interface *iface, const char *answer)
+dns_send_answer(struct interface *iface, struct sockaddr *to, const char *answer)
{
uint8_t buffer[256];
struct blob_attr *attr;
DBG(1, "A <- %s %s\n", dns_type_string(be16_to_cpu(a->type)), answer);
}
- if (interface_send_packet(iface, iov, n_iov) < 0)
- fprintf(stderr, "failed to send question\n");
+ if (interface_send_packet(iface, to, iov, n_iov) < 0)
+ perror("failed to send answer");
}
void
-dns_reply_a(struct interface *iface, int ttl)
+dns_reply_a(struct interface *iface, struct sockaddr *to, int ttl)
{
struct ifaddrs *ifap, *ifa;
struct sockaddr_in *sa;
dns_add_answer(TYPE_AAAA, (uint8_t *) &sa6->sin6_addr, 16, ttl);
}
}
- dns_send_answer(iface, mdns_hostname_local);
+ dns_send_answer(iface, to, mdns_hostname_local);
freeifaddrs(ifap);
}
}
static void
-parse_question(struct interface *iface, char *name, struct dns_question *q)
+parse_question(struct interface *iface, struct sockaddr *from, char *name, struct dns_question *q)
{
+ struct sockaddr *to;
char *host;
- if ((q->class & CLASS_UNICAST) && iface->multicast)
+ /* TODO: Multicast if more than one quarter of TTL has passed */
+ if ((q->class & CLASS_UNICAST) && iface->multicast) {
iface = iface->peer;
+ to = from;
+ } else {
+ to = NULL;
+ }
DBG(1, "Q -> %s %s\n", dns_type_string(q->type), name);
switch (q->type) {
case TYPE_ANY:
if (!strcmp(name, mdns_hostname_local)) {
- service_reply(iface, NULL, announce_ttl);
- dns_reply_a(iface, announce_ttl);
+ dns_reply_a(iface, to, announce_ttl);
+ service_reply(iface, to, NULL, announce_ttl);
}
break;
case TYPE_PTR:
- service_announce_services(iface, name, announce_ttl);
- service_reply(iface, name, announce_ttl);
+ if (!strcmp(name, sdudp)) {
+ dns_reply_a(iface, to, announce_ttl);
+ service_announce_services(iface, to, announce_ttl);
+ } else {
+ /* First dot separates instance name from the rest */
+ char *dot = strchr(name, '.');
+ /* Length of queried instance */
+ size_t len = dot ? dot - name : 0;
+
+ /* Make sure it's query for the instance name we use */
+ if (len && len == strlen(mdns_hostname) &&
+ !strncmp(name, mdns_hostname, len))
+ service_reply(iface, to, dot + 1, announce_ttl);
+ }
break;
case TYPE_AAAA:
if (host)
*host = '\0';
if (!strcmp(mdns_hostname, name))
- dns_reply_a(iface, announce_ttl);
+ dns_reply_a(iface, to, announce_ttl);
break;
};
}
}
if (!(h->flags & FLAG_RESPONSE))
- parse_question(iface, name, q);
+ parse_question(iface, s, name, q);
}
if (!(h->flags & FLAG_RESPONSE))