2 This file is part of catta.
4 catta is free software; you can redistribute it and/or modify it
5 under the terms of the GNU Lesser General Public License as
6 published by the Free Software Foundation; either version 2.1 of the
7 License, or (at your option) any later version.
9 catta is distributed in the hope that it will be useful, but WITHOUT
10 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
11 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
12 Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with catta; if not, write to the Free Software
16 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
26 #include <catta/domain.h>
27 #include <catta/malloc.h>
28 #include <catta/error.h>
31 #include <catta/log.h>
34 typedef struct CattaDNSServerInfo CattaDNSServerInfo;
36 struct CattaDNSServerInfo {
37 CattaSDNSServerBrowser *browser;
40 CattaProtocol protocol;
41 CattaRecord *srv_record;
42 CattaSHostNameResolver *host_name_resolver;
44 CattaLookupResultFlags flags;
46 CATTA_LLIST_FIELDS(CattaDNSServerInfo, info);
49 struct CattaSDNSServerBrowser {
52 CattaSRecordBrowser *record_browser;
53 CattaSDNSServerBrowserCallback callback;
55 CattaProtocol aprotocol;
56 CattaLookupFlags user_flags;
60 CATTA_LLIST_FIELDS(CattaSDNSServerBrowser, browser);
61 CATTA_LLIST_HEAD(CattaDNSServerInfo, info);
64 static CattaDNSServerInfo* get_server_info(CattaSDNSServerBrowser *b, CattaIfIndex iface, CattaProtocol protocol, CattaRecord *r) {
65 CattaDNSServerInfo *i;
70 for (i = b->info; i; i = i->info_next)
71 if (i->iface == iface &&
72 i->protocol == protocol &&
73 catta_record_equal_no_ttl(r, i->srv_record))
79 static void server_info_free(CattaSDNSServerBrowser *b, CattaDNSServerInfo *i) {
83 catta_record_unref(i->srv_record);
84 if (i->host_name_resolver)
85 catta_s_host_name_resolver_free(i->host_name_resolver);
87 CATTA_LLIST_REMOVE(CattaDNSServerInfo, info, b->info, i);
89 assert(b->n_info >= 1);
95 static void host_name_resolver_callback(
96 CattaSHostNameResolver *r,
97 CATTA_GCC_UNUSED CattaIfIndex iface,
98 CATTA_GCC_UNUSED CattaProtocol protocol,
99 CattaResolverEvent event,
100 const char *host_name,
101 const CattaAddress *a,
102 CattaLookupResultFlags flags,
105 CattaDNSServerInfo *i = userdata;
112 case CATTA_RESOLVER_FOUND: {
115 i->browser->callback(
120 i->srv_record->data.srv.name,
122 i->srv_record->data.srv.port,
124 i->browser->userdata);
129 case CATTA_RESOLVER_FAILURE:
134 catta_s_host_name_resolver_free(i->host_name_resolver);
135 i->host_name_resolver = NULL;
138 static void record_browser_callback(
139 CattaSRecordBrowser*rr,
141 CattaProtocol protocol,
142 CattaBrowserEvent event,
144 CattaLookupResultFlags flags,
147 CattaSDNSServerBrowser *b = userdata;
153 flags &= CATTA_LOOKUP_RESULT_CACHED | CATTA_LOOKUP_RESULT_MULTICAST | CATTA_LOOKUP_RESULT_WIDE_AREA;
156 case CATTA_BROWSER_NEW: {
157 CattaDNSServerInfo *i;
160 assert(record->key->type == CATTA_DNS_TYPE_SRV);
162 if (get_server_info(b, iface, protocol, record))
168 if (!(i = catta_new(CattaDNSServerInfo, 1)))
173 i->protocol = protocol;
174 i->srv_record = catta_record_ref(record);
175 i->host_name_resolver = catta_s_host_name_resolver_new(
178 record->data.srv.name,
181 host_name_resolver_callback, i);
184 CATTA_LLIST_PREPEND(CattaDNSServerInfo, info, b->info, i);
190 case CATTA_BROWSER_REMOVE: {
191 CattaDNSServerInfo *i;
194 assert(record->key->type == CATTA_DNS_TYPE_SRV);
196 if (!(i = get_server_info(b, iface, protocol, record)))
199 if (!i->host_name_resolver)
205 i->srv_record->data.srv.name,
207 i->srv_record->data.srv.port,
211 server_info_free(b, i);
215 case CATTA_BROWSER_FAILURE:
216 case CATTA_BROWSER_ALL_FOR_NOW:
217 case CATTA_BROWSER_CACHE_EXHAUSTED:
234 CattaSDNSServerBrowser *catta_s_dns_server_browser_new(
237 CattaProtocol protocol,
239 CattaDNSServerType type,
240 CattaProtocol aprotocol,
241 CattaLookupFlags flags,
242 CattaSDNSServerBrowserCallback callback,
245 static const char * const type_table[CATTA_DNS_SERVER_MAX] = {
250 CattaSDNSServerBrowser *b;
252 char n[CATTA_DOMAIN_NAME_MAX];
258 CATTA_CHECK_VALIDITY_RETURN_NULL(server, CATTA_IF_VALID(iface), CATTA_ERR_INVALID_INTERFACE);
259 CATTA_CHECK_VALIDITY_RETURN_NULL(server, CATTA_PROTO_VALID(protocol), CATTA_ERR_INVALID_PROTOCOL);
260 CATTA_CHECK_VALIDITY_RETURN_NULL(server, CATTA_PROTO_VALID(aprotocol), CATTA_ERR_INVALID_PROTOCOL);
261 CATTA_CHECK_VALIDITY_RETURN_NULL(server, !domain || catta_is_valid_domain_name(domain), CATTA_ERR_INVALID_DOMAIN_NAME);
262 CATTA_CHECK_VALIDITY_RETURN_NULL(server, CATTA_FLAGS_VALID(flags, CATTA_LOOKUP_USE_WIDE_AREA|CATTA_LOOKUP_USE_MULTICAST), CATTA_ERR_INVALID_FLAGS);
263 CATTA_CHECK_VALIDITY_RETURN_NULL(server, type < CATTA_DNS_SERVER_MAX, CATTA_ERR_INVALID_FLAGS);
266 domain = server->domain_name;
268 if ((r = catta_service_name_join(n, sizeof(n), NULL, type_table[type], domain)) < 0) {
269 catta_server_set_errno(server, r);
273 if (!(b = catta_new(CattaSDNSServerBrowser, 1))) {
274 catta_server_set_errno(server, CATTA_ERR_NO_MEMORY);
279 b->callback = callback;
280 b->userdata = userdata;
281 b->aprotocol = aprotocol;
283 b->user_flags = flags;
285 CATTA_LLIST_HEAD_INIT(CattaDNSServerInfo, b->info);
286 CATTA_LLIST_PREPEND(CattaSDNSServerBrowser, browser, server->dns_server_browsers, b);
288 if (!(k = catta_key_new(n, CATTA_DNS_CLASS_IN, CATTA_DNS_TYPE_SRV))) {
289 catta_server_set_errno(server, CATTA_ERR_NO_MEMORY);
293 if (!(b->record_browser = catta_s_record_browser_new(server, iface, protocol, k, flags, record_browser_callback, b)))
305 catta_s_dns_server_browser_free(b);
309 void catta_s_dns_server_browser_free(CattaSDNSServerBrowser *b) {
313 server_info_free(b, b->info);
315 CATTA_LLIST_REMOVE(CattaSDNSServerBrowser, browser, b->server->dns_server_browsers, b);
317 if (b->record_browser)
318 catta_s_record_browser_free(b->record_browser);