/** * * /brief getdns contect management functions * * This is the meat of the API * Originally taken from the getdns API description pseudo implementation. * */ /* The MIT License (MIT) * Copyright (c) 2013 Verisign, Inc. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "config.h" #ifdef HAVE_EVENT2_EVENT_H # include #else # include #endif #include #include #include #include #include #include #include #include "context.h" #include "types-internal.h" #include "util-internal.h" void *plain_mem_funcs_user_arg = MF_PLAIN; /* Private functions */ static uint16_t *create_default_namespaces(struct getdns_context *context); static struct getdns_list *create_default_root_servers(); static getdns_return_t add_ip_str(struct getdns_dict *); static struct getdns_dict *create_ipaddr_dict_from_rdf(struct getdns_context *, ldns_rdf *); static struct getdns_list *create_from_ldns_list(struct getdns_context *, ldns_rdf **, size_t); static getdns_return_t set_os_defaults(struct getdns_context *); static int transaction_id_cmp(const void *, const void *); static void set_ub_string_opt(struct getdns_context *, char *, char *); static void set_ub_number_opt(struct getdns_context *, char *, uint16_t); static inline void clear_resolution_type_set_flag(struct getdns_context *, uint16_t); static void dispatch_updated(struct getdns_context *, uint16_t); static void cancel_dns_req(getdns_dns_req *); /* Stuff to make it compile pedantically */ #define UNUSED_PARAM(x) ((void)(x)) /** * Helper to get default lookup namespaces. * TODO: Determine from OS */ static uint16_t * create_default_namespaces(struct getdns_context *context) { uint16_t *result = GETDNS_XMALLOC(context, uint16_t, 2); result[0] = GETDNS_CONTEXT_NAMESPACE_LOCALNAMES; result[1] = GETDNS_CONTEXT_NAMESPACE_DNS; return result; } /** * Helper to get the default root servers. * TODO: Implement */ static struct getdns_list * create_default_root_servers() { return NULL; } static getdns_return_t add_ip_str(struct getdns_dict * ip) { struct sockaddr_storage storage; char buff[256]; getdns_return_t r = dict_to_sockaddr(ip, &storage); if (r != GETDNS_RETURN_GOOD) { return r; } if (storage.ss_family == AF_INET) { struct sockaddr_in *addr = (struct sockaddr_in *) &storage; const char *ipStr = inet_ntop(AF_INET, &(addr->sin_addr), buff, 256); if (!ipStr) { return GETDNS_RETURN_GENERIC_ERROR; } getdns_dict_util_set_string(ip, GETDNS_STR_ADDRESS_STRING, ipStr); } else if (storage.ss_family == AF_INET6) { struct sockaddr_in6 *addr = (struct sockaddr_in6 *) &storage; const char *ipStr = inet_ntop(AF_INET6, &(addr->sin6_addr), buff, 256); if (!ipStr) { return GETDNS_RETURN_GENERIC_ERROR; } getdns_dict_util_set_string(ip, GETDNS_STR_ADDRESS_STRING, ipStr); } else { /* unknown */ return GETDNS_RETURN_GENERIC_ERROR; } return GETDNS_RETURN_GOOD; } static struct getdns_dict * create_ipaddr_dict_from_rdf(struct getdns_context *context, ldns_rdf * rdf) { ldns_rdf_type rt = ldns_rdf_get_type(rdf); size_t sz = ldns_rdf_size(rdf); struct getdns_dict *result = getdns_dict_create_with_context(context); /* set type */ if (rt == LDNS_RDF_TYPE_A) { getdns_dict_util_set_string(result, GETDNS_STR_ADDRESS_TYPE, GETDNS_STR_IPV4); } else { getdns_dict_util_set_string(result, GETDNS_STR_ADDRESS_TYPE, GETDNS_STR_IPV6); } /* set data */ struct getdns_bindata data_bin = { sz, ldns_rdf_data(rdf) }; getdns_dict_set_bindata(result, GETDNS_STR_ADDRESS_DATA, &data_bin); add_ip_str(result); return result; } static struct getdns_list * create_from_ldns_list(struct getdns_context *context, ldns_rdf ** ldns_list, size_t count) { size_t i = 0; size_t idx = 0; struct getdns_list *result = getdns_list_create_with_context(context); for (i = 0; i < count; ++i) { ldns_rdf *rdf = ldns_list[i]; switch (ldns_rdf_get_type(rdf)) { case LDNS_RDF_TYPE_A: case LDNS_RDF_TYPE_AAAA: { struct getdns_dict *ipaddr = create_ipaddr_dict_from_rdf(context, rdf); getdns_list_add_item(result, &idx); getdns_list_set_dict(result, idx, ipaddr); getdns_dict_destroy(ipaddr); } break; case LDNS_RDF_TYPE_DNAME: { struct getdns_bindata item; char *srch = ldns_rdf2str(rdf); item.size = strlen(srch) + 1; item.data = (uint8_t *) srch; getdns_list_add_item(result, &idx); getdns_list_set_bindata(result, idx, &item); free(srch); } break; default: break; } } return result; } static getdns_return_t set_os_defaults(struct getdns_context *context) { ldns_resolver *lr = NULL; if (ldns_resolver_new_frm_file(&lr, NULL) != LDNS_STATUS_OK) { return GETDNS_RETURN_GENERIC_ERROR; } ldns_rdf **rdf_list = ldns_resolver_nameservers(lr); size_t rdf_list_sz = ldns_resolver_nameserver_count(lr); if (rdf_list_sz > 0) { context->upstream_list = create_from_ldns_list(context, rdf_list, rdf_list_sz); } rdf_list = ldns_resolver_searchlist(lr); rdf_list_sz = ldns_resolver_searchlist_count(lr); if (rdf_list_sz > 0) { context->suffix = create_from_ldns_list(context, rdf_list, rdf_list_sz); } /** cleanup **/ ldns_resolver_deep_free(lr); return GETDNS_RETURN_GOOD; } static int transaction_id_cmp(const void *id1, const void *id2) { if (id1 == NULL && id2 == NULL) { return 0; } else if (id1 == NULL && id2 != NULL) { return 1; } else if (id1 != NULL && id2 == NULL) { return -1; } else { getdns_transaction_t t1 = *((const getdns_transaction_t *) id1); getdns_transaction_t t2 = *((const getdns_transaction_t *) id2); if (t1 == t2) { return 0; } else if (t1 < t2) { return -1; } else { return 1; } } } /* * getdns_context_create * * Call this to initialize the context that is used in other getdns calls. */ getdns_return_t getdns_context_create_with_memory_functions(struct getdns_context ** context, int set_from_os, void *(*malloc)(size_t), void *(*realloc)(void *, size_t), void (*free)(void *) ) { struct getdns_context *result = NULL; if (!context || !malloc || !realloc || !free) return GETDNS_RETURN_GENERIC_ERROR; /** default init **/ result = (*malloc)(sizeof(struct getdns_context)); if (!result) { return GETDNS_RETURN_GENERIC_ERROR; } result->update_callback = NULL; result->mf_arg = MF_PLAIN; result->mf.pln.malloc = malloc; result->mf.pln.realloc = realloc; result->mf.pln.free = free; result->event_base_sync = event_base_new(); result->unbound_sync = ub_ctx_create_event(result->event_base_sync); /* create the async one also so options are kept up to date */ result->unbound_async = ub_ctx_create_event(result->event_base_sync); result->event_base_async = NULL; result->resolution_type_set = 0; result->outbound_requests = ldns_rbtree_create(transaction_id_cmp); result->resolution_type = GETDNS_CONTEXT_RECURSING; result->namespaces = create_default_namespaces(result); result->timeout = 5000; result->follow_redirects = GETDNS_CONTEXT_FOLLOW_REDIRECTS; result->dns_root_servers = create_default_root_servers(); result->append_name = GETDNS_CONTEXT_APPEND_NAME_ALWAYS; result->suffix = NULL; result->dnssec_trust_anchors = NULL; result->upstream_list = NULL; result->edns_extended_rcode = 0; result->edns_version = 0; result->edns_do_bit = 0; if (set_from_os) { if (GETDNS_RETURN_GOOD != set_os_defaults(result)) { getdns_context_destroy(result); return GETDNS_RETURN_GENERIC_ERROR; } } *context = result; /* other opts */ getdns_context_set_dnssec_allowed_skew(result, 0); getdns_context_set_edns_maximum_udp_payload_size(result, 512); getdns_context_set_dns_transport(result, GETDNS_CONTEXT_UDP_FIRST_AND_FALL_BACK_TO_TCP); return GETDNS_RETURN_GOOD; } /* getdns_context_create */ /* * getdns_context_create * * Call this to initialize the context that is used in other getdns calls. */ getdns_return_t getdns_context_create(struct getdns_context ** context, int set_from_os) { return getdns_context_create_with_memory_functions(context, set_from_os, malloc, realloc, free); } /* getdns_context_create */ /* * getdns_context_destroy * * Call this to dispose of resources associated with a context once you * are done with it. */ void getdns_context_destroy(struct getdns_context *context) { if (context == NULL) { return; } if (context->namespaces) GETDNS_FREE(context, context->namespaces); getdns_list_destroy(context->dns_root_servers); getdns_list_destroy(context->suffix); getdns_list_destroy(context->dnssec_trust_anchors); getdns_list_destroy(context->upstream_list); /* destroy the ub context */ ub_ctx_delete(context->unbound_async); ub_ctx_delete(context->unbound_sync); event_base_free(context->event_base_sync); ldns_rbtree_free(context->outbound_requests); GETDNS_FREE(context, context); return; } /* getdns_context_destroy */ /* * getdns_context_set_context_update_callback * */ getdns_return_t getdns_context_set_context_update_callback(struct getdns_context *context, void (*value) (struct getdns_context *context, uint16_t changed_item) ) { context->update_callback = value; return GETDNS_RETURN_GOOD; } /* getdns_context_set_context_update_callback */ /* * Helpers to set options on the unbound ctx */ static void set_ub_string_opt(struct getdns_context *ctx, char *opt, char *value) { ub_ctx_set_option(ctx->unbound_sync, opt, value); ub_ctx_set_option(ctx->unbound_async, opt, value); } static void set_ub_number_opt(struct getdns_context *ctx, char *opt, uint16_t value) { char buffer[64]; snprintf(buffer, 64, "%hu", value); set_ub_string_opt(ctx, opt, buffer); } /* * Clear the resolution type set flag if needed */ static inline void clear_resolution_type_set_flag(struct getdns_context *context, uint16_t type) { if (context->resolution_type_set == type) { context->resolution_type_set = 0; } } /* * getdns_context_set_context_update * */ getdns_return_t getdns_context_set_context_update(struct getdns_context *context, uint16_t value) { UNUSED_PARAM(context); UNUSED_PARAM(value); return GETDNS_RETURN_GOOD; } /* getdns_context_set_context_update */ /** * Helper to dispatch the updated callback */ static void dispatch_updated(struct getdns_context *context, uint16_t item) { if (context->update_callback) { context->update_callback(context, item); } } /* * getdns_context_set_resolution_type * */ getdns_return_t getdns_context_set_resolution_type(struct getdns_context *context, uint16_t value) { if (value != GETDNS_CONTEXT_STUB && value != GETDNS_CONTEXT_RECURSING) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } context->resolution_type = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_RESOLUTION_TYPE); return GETDNS_RETURN_GOOD; } /* getdns_context_set_resolution_type */ /* * getdns_context_set_namespaces * */ getdns_return_t getdns_context_set_namespaces(struct getdns_context *context, size_t namespace_count, uint16_t * namespaces) { if (namespace_count == 0 || namespaces == NULL) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } /** clean up old namespaces **/ GETDNS_FREE(context, context->namespaces); /** duplicate **/ context->namespaces = GETDNS_XMALLOC(context, uint16_t, namespace_count); memcpy(context->namespaces, namespaces, namespace_count * sizeof(uint16_t)); dispatch_updated(context, GETDNS_CONTEXT_CODE_NAMESPACES); return GETDNS_RETURN_GOOD; } /* getdns_context_set_namespaces */ /* * getdns_context_set_dns_transport * */ getdns_return_t getdns_context_set_dns_transport(struct getdns_context *context, uint16_t value) { switch (value) { case GETDNS_CONTEXT_UDP_FIRST_AND_FALL_BACK_TO_TCP: set_ub_string_opt(context, "do-udp", "yes"); set_ub_string_opt(context, "do-tcp", "yes"); break; case GETDNS_CONTEXT_UDP_ONLY: set_ub_string_opt(context, "do-udp", "yes"); set_ub_string_opt(context, "do-tcp", "no"); break; case GETDNS_CONTEXT_TCP_ONLY: set_ub_string_opt(context, "do-udp", "no"); set_ub_string_opt(context, "do-tcp", "yes"); break; default: /* TODO GETDNS_CONTEXT_TCP_ONLY_KEEP_CONNECTIONS_OPEN */ return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } dispatch_updated(context, GETDNS_CONTEXT_CODE_DNS_TRANSPORT); return GETDNS_RETURN_GOOD; } /* getdns_context_set_dns_transport */ /* * getdns_context_set_limit_outstanding_queries * */ getdns_return_t getdns_context_set_limit_outstanding_queries(struct getdns_context *context, uint16_t limit) { /* num-queries-per-thread */ set_ub_number_opt(context, "num-queries-per-thread", limit); dispatch_updated(context, GETDNS_CONTEXT_CODE_LIMIT_OUTSTANDING_QUERIES); return GETDNS_RETURN_GOOD; } /* getdns_context_set_limit_outstanding_queries */ /* * getdns_context_set_timeout * */ getdns_return_t getdns_context_set_timeout(struct getdns_context *context, uint16_t timeout) { context->timeout = timeout; dispatch_updated(context, GETDNS_CONTEXT_CODE_TIMEOUT); return GETDNS_RETURN_GOOD; } /* getdns_context_set_timeout */ /* * getdns_context_set_follow_redirects * */ getdns_return_t getdns_context_set_follow_redirects(struct getdns_context *context, uint16_t value) { context->follow_redirects = value; clear_resolution_type_set_flag(context, GETDNS_CONTEXT_RECURSING); dispatch_updated(context, GETDNS_CONTEXT_CODE_FOLLOW_REDIRECTS); return GETDNS_RETURN_GOOD; } /* getdns_context_set_follow_redirects */ /* * getdns_context_set_dns_root_servers * */ getdns_return_t getdns_context_set_dns_root_servers(struct getdns_context *context, struct getdns_list * addresses) { struct getdns_list *copy = NULL; size_t count = 0; if (addresses != NULL) { if (getdns_list_copy(addresses, ©) != GETDNS_RETURN_GOOD) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } addresses = copy; getdns_list_get_length(addresses, &count); if (count == 0) { getdns_list_destroy(addresses); addresses = NULL; } else { size_t i = 0; getdns_return_t r = GETDNS_RETURN_GOOD; /* validate and add ip str */ for (i = 0; i < count; ++i) { struct getdns_dict *dict = NULL; getdns_list_get_dict(addresses, i, &dict); r = add_ip_str(dict); if (r != GETDNS_RETURN_GOOD) { break; } } if (r != GETDNS_RETURN_GOOD) { getdns_list_destroy(addresses); return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } } } getdns_list_destroy(context->dns_root_servers); context->dns_root_servers = addresses; clear_resolution_type_set_flag(context, GETDNS_CONTEXT_RECURSING); dispatch_updated(context, GETDNS_CONTEXT_CODE_DNS_ROOT_SERVERS); return GETDNS_RETURN_GOOD; } /* getdns_context_set_dns_root_servers */ /* * getdns_context_set_append_name * */ getdns_return_t getdns_context_set_append_name(struct getdns_context *context, uint16_t value) { if (value != GETDNS_CONTEXT_APPEND_NAME_ALWAYS && value != GETDNS_CONTEXT_APPEND_NAME_ONLY_TO_SINGLE_LABEL_AFTER_FAILURE && value != GETDNS_CONTEXT_APPEND_NAME_ONLY_TO_MULTIPLE_LABEL_NAME_AFTER_FAILURE && value != GETDNS_CONTEXT_DO_NOT_APPEND_NAMES) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } context->append_name = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_APPEND_NAME); return GETDNS_RETURN_GOOD; } /* getdns_context_set_append_name */ /* * getdns_context_set_suffix * */ getdns_return_t getdns_context_set_suffix(struct getdns_context *context, struct getdns_list * value) { struct getdns_list *copy = NULL; if (value != NULL) { if (getdns_list_copy(value, ©) != GETDNS_RETURN_GOOD) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } value = copy; } getdns_list_destroy(context->suffix); context->suffix = value; clear_resolution_type_set_flag(context, GETDNS_CONTEXT_STUB); dispatch_updated(context, GETDNS_CONTEXT_CODE_SUFFIX); return GETDNS_RETURN_GOOD; } /* getdns_context_set_suffix */ /* * getdns_context_set_dnssec_trust_anchors * */ getdns_return_t getdns_context_set_dnssec_trust_anchors(struct getdns_context *context, struct getdns_list * value) { struct getdns_list *copy = NULL; if (value != NULL) { if (getdns_list_copy(value, ©) != GETDNS_RETURN_GOOD) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } value = copy; } getdns_list_destroy(context->dnssec_trust_anchors); context->dnssec_trust_anchors = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_DNSSEC_TRUST_ANCHORS); return GETDNS_RETURN_GOOD; } /* getdns_context_set_dnssec_trust_anchors */ /* * getdns_context_set_dnssec_allowed_skew * */ getdns_return_t getdns_context_set_dnssec_allowed_skew(struct getdns_context *context, uint16_t value) { set_ub_number_opt(context, "val-sig-skew-min", value); set_ub_number_opt(context, "val-sig-skew-max", value); dispatch_updated(context, GETDNS_CONTEXT_CODE_DNSSEC_ALLOWED_SKEW); return GETDNS_RETURN_GOOD; } /* getdns_context_set_dnssec_allowed_skew */ /* * getdns_context_set_stub_resolution * */ getdns_return_t getdns_context_set_stub_resolution(struct getdns_context *context, struct getdns_list * upstream_list) { size_t count = 0; size_t i = 0; getdns_return_t r = getdns_list_get_length(upstream_list, &count); if (count == 0 || r != GETDNS_RETURN_GOOD) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } struct getdns_list *copy = NULL; if (getdns_list_copy(upstream_list, ©) != GETDNS_RETURN_GOOD) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } upstream_list = copy; /* validate and add ip str */ for (i = 0; i < count; ++i) { struct getdns_dict *dict = NULL; getdns_list_get_dict(upstream_list, i, &dict); r = add_ip_str(dict); if (r != GETDNS_RETURN_GOOD) { break; } } if (r != GETDNS_RETURN_GOOD) { getdns_list_destroy(upstream_list); return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } getdns_list_destroy(context->upstream_list); context->upstream_list = upstream_list; clear_resolution_type_set_flag(context, GETDNS_CONTEXT_STUB); dispatch_updated(context, GETDNS_CONTEXT_CODE_UPSTREAM_RECURSIVE_SERVERS); return GETDNS_RETURN_GOOD; } /* getdns_context_set_stub_resolution */ /* * getdns_context_set_edns_maximum_udp_payload_size * */ getdns_return_t getdns_context_set_edns_maximum_udp_payload_size(struct getdns_context *context, uint16_t value) { /* check for < 512. uint16_t won't let it go above max) */ if (value < 512) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } /* max-udp-size */ set_ub_number_opt(context, "max-udp-size", value); dispatch_updated(context, GETDNS_CONTEXT_CODE_EDNS_MAXIMUM_UDP_PAYLOAD_SIZE); return GETDNS_RETURN_GOOD; } /* getdns_context_set_edns_maximum_udp_payload_size */ /* * getdns_context_set_edns_extended_rcode * */ getdns_return_t getdns_context_set_edns_extended_rcode(struct getdns_context *context, uint8_t value) { context->edns_extended_rcode = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_EDNS_EXTENDED_RCODE); return GETDNS_RETURN_GOOD; } /* getdns_context_set_edns_extended_rcode */ /* * getdns_context_set_edns_version * */ getdns_return_t getdns_context_set_edns_version(struct getdns_context *context, uint8_t value) { context->edns_version = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_EDNS_VERSION); return GETDNS_RETURN_GOOD; } /* getdns_context_set_edns_version */ /* * getdns_context_set_edns_do_bit * */ getdns_return_t getdns_context_set_edns_do_bit(struct getdns_context *context, uint8_t value) { /* 0 or 1 */ if (value > 1) { return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; } context->edns_do_bit = value; dispatch_updated(context, GETDNS_CONTEXT_CODE_EDNS_DO_BIT); return GETDNS_RETURN_GOOD; } /* getdns_context_set_edns_do_bit */ /* * getdns_context_set_memory_functions * */ getdns_return_t getdns_context_set_memory_functions(struct getdns_context *context, void *(*malloc) (size_t), void *(*realloc) (void *, size_t), void (*free) (void *) ) { if (!context) return GETDNS_RETURN_BAD_CONTEXT; if (!malloc || !realloc || !free) return GETDNS_RETURN_CONTEXT_UPDATE_FAIL; context->mf_arg = MF_PLAIN; context->mf.pln.malloc = malloc; context->mf.pln.realloc = realloc; context->mf.pln.free = free; dispatch_updated(context, GETDNS_CONTEXT_CODE_MEMORY_FUNCTIONS); return GETDNS_RETURN_GOOD; } /* getdns_context_set_memory_functions*/ /* * getdns_extension_set_libevent_base * */ getdns_return_t getdns_extension_set_libevent_base(struct getdns_context *context, struct event_base * this_event_base) { if (this_event_base) { ub_ctx_set_event(context->unbound_async, this_event_base); context->event_base_async = this_event_base; } else { ub_ctx_set_event(context->unbound_async, context->event_base_sync); context->event_base_async = NULL; } return GETDNS_RETURN_GOOD; } /* getdns_extension_set_libevent_base */ /* cancel the request */ static void cancel_dns_req(getdns_dns_req * req) { getdns_network_req *netreq = req->first_req; while (netreq) { if (netreq->state == NET_REQ_IN_FLIGHT) { /* for ev based ub, this should always prevent * the callback from firing */ ub_cancel(req->unbound, netreq->unbound_id); netreq->state = NET_REQ_CANCELED; } else if (netreq->state == NET_REQ_NOT_SENT) { netreq->state = NET_REQ_CANCELED; } netreq = netreq->next; } req->canceled = 1; } getdns_return_t getdns_context_cancel_request(struct getdns_context *context, getdns_transaction_t transaction_id, int fire_callback) { getdns_dns_req *req = NULL; /* delete the node from the tree */ ldns_rbnode_t *node = ldns_rbtree_delete(context->outbound_requests, &transaction_id); if (!node) { return GETDNS_RETURN_UNKNOWN_TRANSACTION; } req = (getdns_dns_req *) node->data; /* do the cancel */ cancel_dns_req(req); if (fire_callback) { getdns_callback_t cb = NULL; void *user_pointer = NULL; cb = req->user_callback; user_pointer = req->user_pointer; /* clean up */ GETDNS_FREE(context, node); dns_req_free(req); /* fire callback */ cb(context, GETDNS_CALLBACK_CANCEL, NULL, user_pointer, transaction_id); } return GETDNS_RETURN_GOOD; } /* * getdns_cancel_callback * */ getdns_return_t getdns_cancel_callback(struct getdns_context *context, getdns_transaction_t transaction_id) { return getdns_context_cancel_request(context, transaction_id, 1); } /* getdns_cancel_callback */ static void ub_setup_stub(struct ub_ctx *ctx, struct getdns_list * upstreams, size_t count) { size_t i; /* reset forwarding servers */ ub_ctx_set_fwd(ctx, NULL); for (i = 0; i < count; ++i) { struct getdns_dict *dict = NULL; char *ip_str = NULL; getdns_list_get_dict(upstreams, i, &dict); getdns_dict_util_get_string(dict, GETDNS_STR_ADDRESS_STRING, &ip_str); ub_ctx_set_fwd(ctx, ip_str); } } getdns_return_t getdns_context_prepare_for_resolution(struct getdns_context *context) { if (context->resolution_type_set == context->resolution_type) { /* already set and no config changes have caused this to be * bad. */ return GETDNS_RETURN_GOOD; } if (context->resolution_type == GETDNS_CONTEXT_STUB) { size_t upstream_len = 0; getdns_return_t r = getdns_list_get_length(context->upstream_list, &upstream_len); if (r != GETDNS_RETURN_GOOD || upstream_len == 0) { return GETDNS_RETURN_BAD_CONTEXT; } /* set upstreams */ ub_setup_stub(context->unbound_async, context->upstream_list, upstream_len); ub_setup_stub(context->unbound_sync, context->upstream_list, upstream_len); /* use /etc/hosts */ ub_ctx_hosts(context->unbound_sync, NULL); ub_ctx_hosts(context->unbound_async, NULL); } else if (context->resolution_type == GETDNS_CONTEXT_RECURSING) { /* set recursive */ /* TODO: use the root servers via root hints file */ ub_ctx_set_fwd(context->unbound_async, NULL); ub_ctx_set_fwd(context->unbound_sync, NULL); } else { /* bogus? */ return GETDNS_RETURN_BAD_CONTEXT; } context->resolution_type_set = context->resolution_type; return GETDNS_RETURN_GOOD; } getdns_return_t getdns_context_track_outbound_request(getdns_dns_req * req) { if (!req) { return GETDNS_RETURN_GENERIC_ERROR; } struct getdns_context *context = req->context; ldns_rbnode_t *node = GETDNS_MALLOC(context, ldns_rbnode_t); if (!node) { return GETDNS_RETURN_GENERIC_ERROR; } node->key = &(req->trans_id); node->data = req; if (!ldns_rbtree_insert(context->outbound_requests, node)) { /* free the node */ GETDNS_FREE(context, node); return GETDNS_RETURN_GENERIC_ERROR; } return GETDNS_RETURN_GOOD; } getdns_return_t getdns_context_clear_outbound_request(getdns_dns_req * req) { if (!req) { return GETDNS_RETURN_GENERIC_ERROR; } struct getdns_context *context = req->context; ldns_rbnode_t *node = ldns_rbtree_delete(context->outbound_requests, &(req->trans_id)); if (node) { GETDNS_FREE(context, node); } return GETDNS_RETURN_GOOD; } char * getdns_strdup(void *(*malloc)(size_t), const char *s) { size_t sz = strlen(s) + 1; char *r = (char *)(*malloc)(sizeof(char) * sz); if (r) return memcpy(r, s, sz); else return NULL; } struct getdns_bindata * getdns_bindata_copy(void *(*malloc)(size_t), void (*free)(void *), const struct getdns_bindata *src) { struct getdns_bindata *dst; if (!src) return NULL; dst = (struct getdns_bindata *)(*malloc)(sizeof(struct getdns_bindata)); if (!dst) return NULL; dst->size = src->size; dst->data = (uint8_t *)(*malloc)(sizeof(uint8_t) * src->size); if (!dst->data) { (*free)(dst); return NULL; } (void) memcpy(dst->data, src->data, src->size); return dst; } void getdns_bindata_destroy(void (*free)(void *), struct getdns_bindata *bindata) { if (!bindata) return; (*free)(bindata->data); (*free)(bindata); } /* getdns_context.c */