6 * This version of ecore_con_info uses dns.c to provide asynchronous dns lookup.
8 * dns.c is written by William Ahern:
9 * http://25thandclement.com/~william/projects/dns.c.html
13 #include <sys/types.h>
15 #ifdef HAVE_NETINET_IN_H
16 # include <netinet/in.h>
19 #ifdef HAVE_ARPA_INET_H
20 # include <arpa/inet.h>
26 #include "Ecore_Con.h"
27 #include "ecore_con_private.h"
29 typedef struct dns_addrinfo dns_addrinfo;
30 typedef struct dns_resolv_conf dns_resolv_conf;
31 typedef struct dns_resolver dns_resolver;
32 typedef struct dns_hosts dns_hosts;
34 typedef struct _Ecore_Con_DNS Ecore_Con_DNS;
38 Ecore_Con_Server *svr;
39 Ecore_Con_Info_Cb done_cb;
43 struct addrinfo hints;
44 Ecore_Fd_Handler *fdh;
48 static int _ecore_con_dns_init = 0;
49 static dns_resolv_conf *resconf = NULL;
50 static dns_hosts *hosts = NULL;
53 _ecore_con_dns_free(Ecore_Con_DNS *dns)
55 if (dns->svr->infos) dns->svr->infos = eina_list_remove(dns->svr->infos, dns);
56 if (dns->timer) ecore_timer_del(dns->timer);
57 if (dns->fdh) ecore_main_fd_handler_del(dns->fdh);
58 dns_res_close(dns->resolv);
63 _dns_addrinfo_get(Ecore_Con_DNS *dns, const char *addr, int port)
66 char service[NI_MAXSERV];
68 snprintf(service, sizeof(service), "%d", port);
69 dns->ai = dns_ai_open(addr, service, DNS_T_A, (const struct addrinfo *)&dns->hints, dns->resolv, &error);
74 _ecore_con_dns_check(Ecore_Con_DNS *dns)
76 struct addrinfo *ent = NULL;
79 error = dns_ai_nextent(&ent, dns->ai);
88 ERR("resolve failed: %s", dns_strerror(error));
93 Ecore_Con_Info result = {0, .ip = {0}, .service = {0}};
96 dns_ai_print(pretty, sizeof(pretty), ent, dns->ai);
97 printf("%s\n", pretty);
100 dns_inet_ntop(dns_sa_family(ent->ai_addr), dns_sa_addr(dns_sa_family(ent->ai_addr), ent->ai_addr), result.ip, sizeof(result.ip));
101 snprintf(result.service, sizeof(result.service), "%u", ntohs(*dns_sa_port(dns_sa_family(ent->ai_addr), ent->ai_addr)));
102 memcpy(&result.info, ent, sizeof(result.info));
103 if (dns->fdh) ecore_main_fd_handler_del(dns->fdh);
105 dns->done_cb(dns->data, &result);
107 _ecore_con_dns_free(dns);
112 dns->done_cb(dns->data, NULL);
113 _ecore_con_dns_free(dns);
118 _dns_fd_cb(Ecore_Con_DNS *dns, Ecore_Fd_Handler *fdh)
120 if (_ecore_con_dns_check(dns) == 1)
121 ecore_main_fd_handler_active_set(fdh, dns_ai_events(dns->ai));
122 return ECORE_CALLBACK_RENEW;
126 _dns_timer_cb(Ecore_Con_DNS *dns)
128 dns->done_cb(dns->data, NULL);
129 _ecore_con_dns_free(dns);
135 ecore_con_info_init(void)
138 if (_ecore_con_dns_init) return ++_ecore_con_dns_init;
140 resconf = dns_resconf_local(&err);
143 ERR("resconf_open: %s", dns_strerror(err));
146 hosts = dns_hosts_local(&err);
149 ERR("hosts_open: %s", dns_strerror(err));
150 dns_resconf_close(resconf);
154 /* this is super slow don't do it */
155 //resconf->options.recurse = 1;
156 return ++_ecore_con_dns_init;
160 ecore_con_info_shutdown(void)
162 if (!_ecore_con_dns_init) return 0;
163 if (--_ecore_con_dns_init) return _ecore_con_dns_init;
164 dns_resconf_close(resconf);
166 dns_hosts_close(hosts);
172 ecore_con_info_data_clear(void *info)
174 Ecore_Con_DNS *dns = info;
175 if (dns) dns->data = NULL;
179 ecore_con_info_tcp_connect(Ecore_Con_Server *svr,
180 Ecore_Con_Info_Cb done_cb,
183 struct addrinfo hints;
185 memset(&hints, 0, sizeof(struct addrinfo));
187 hints.ai_family = AF_INET6;
189 hints.ai_family = AF_INET;
191 hints.ai_socktype = SOCK_STREAM;
192 hints.ai_flags = AI_CANONNAME;
193 hints.ai_protocol = IPPROTO_TCP;
195 return ecore_con_info_get(svr, done_cb, data, &hints);
199 ecore_con_info_tcp_listen(Ecore_Con_Server *svr,
200 Ecore_Con_Info_Cb done_cb,
203 struct addrinfo hints;
205 memset(&hints, 0, sizeof(struct addrinfo));
207 hints.ai_family = AF_INET6;
209 hints.ai_family = AF_INET;
211 hints.ai_socktype = SOCK_STREAM;
212 hints.ai_flags = AI_PASSIVE;
213 hints.ai_protocol = IPPROTO_TCP;
215 return ecore_con_info_get(svr, done_cb, data, &hints);
219 ecore_con_info_udp_connect(Ecore_Con_Server *svr,
220 Ecore_Con_Info_Cb done_cb,
223 struct addrinfo hints;
225 memset(&hints, 0, sizeof(struct addrinfo));
227 hints.ai_family = AF_INET6;
229 hints.ai_family = AF_INET;
231 hints.ai_socktype = SOCK_DGRAM;
232 hints.ai_flags = AI_CANONNAME;
233 hints.ai_protocol = IPPROTO_UDP;
235 return ecore_con_info_get(svr, done_cb, data, &hints);
239 ecore_con_info_udp_listen(Ecore_Con_Server *svr,
240 Ecore_Con_Info_Cb done_cb,
243 struct addrinfo hints;
245 memset(&hints, 0, sizeof(struct addrinfo));
247 hints.ai_family = AF_INET6;
249 hints.ai_family = AF_INET;
251 hints.ai_socktype = SOCK_DGRAM;
252 hints.ai_flags = AI_PASSIVE;
253 hints.ai_protocol = IPPROTO_UDP;
255 return ecore_con_info_get(svr, done_cb, data, &hints);
259 ecore_con_info_mcast_listen(Ecore_Con_Server *svr,
260 Ecore_Con_Info_Cb done_cb,
263 struct addrinfo hints;
265 memset(&hints, 0, sizeof(struct addrinfo));
267 hints.ai_family = AF_INET6;
269 hints.ai_family = AF_INET;
271 hints.ai_socktype = SOCK_DGRAM;
272 hints.ai_protocol = IPPROTO_UDP;
274 return ecore_con_info_get(svr, done_cb, data, &hints);
278 ecore_con_info_get(Ecore_Con_Server *svr,
279 Ecore_Con_Info_Cb done_cb,
281 struct addrinfo *hints)
286 dns = calloc(1, sizeof(Ecore_Con_DNS));
290 dns->done_cb = done_cb;
294 memcpy(&dns->hints, hints, sizeof(struct addrinfo));
296 if (!(dns->resolv = dns_res_open(resconf, hosts, dns_hints_mortal(dns_hints_local(resconf, &error)), NULL, dns_opts(), &error)))
298 ERR("res_open: %s", dns_strerror(error));
303 error = _dns_addrinfo_get(dns, svr->ecs ? svr->ecs->ip : svr->name, dns->svr->ecs ? dns->svr->ecs->port : dns->svr->port);
304 if (error && (error != EAGAIN))
306 ERR("resolver: %s", dns_strerror(error));
310 switch (_ecore_con_dns_check(dns))
315 dns->fdh = ecore_main_fd_handler_add(dns_ai_pollfd(dns->ai), dns_ai_events(dns->ai), (Ecore_Fd_Cb)_dns_fd_cb, dns, NULL, NULL);
316 svr->infos = eina_list_append(svr->infos, dns);
317 dns->timer = ecore_timer_add(5.0, (Ecore_Task_Cb)_dns_timer_cb, dns);
325 if (dns->resolv) dns_res_close(dns->resolv);