aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/restricted/aws/aws-c-io/source/host_resolver.c
diff options
context:
space:
mode:
authorrobot-contrib <robot-contrib@yandex-team.com>2023-03-07 09:11:05 +0300
committerrobot-contrib <robot-contrib@yandex-team.com>2023-03-07 09:11:05 +0300
commit980f2a05eb72bde52c212d5c67f8860e45ef1b10 (patch)
treeaea8ee1e3d4e5c51e3de18069d4bc56a2021c31b /contrib/restricted/aws/aws-c-io/source/host_resolver.c
parentdb461a7619dcd5bc525a809660d32a5331638e85 (diff)
downloadydb-980f2a05eb72bde52c212d5c67f8860e45ef1b10.tar.gz
Update contrib/restricted/aws/aws-c-io to 0.13.17
Diffstat (limited to 'contrib/restricted/aws/aws-c-io/source/host_resolver.c')
-rw-r--r--contrib/restricted/aws/aws-c-io/source/host_resolver.c268
1 files changed, 217 insertions, 51 deletions
diff --git a/contrib/restricted/aws/aws-c-io/source/host_resolver.c b/contrib/restricted/aws/aws-c-io/source/host_resolver.c
index 050f00a187..4854ab16b7 100644
--- a/contrib/restricted/aws/aws-c-io/source/host_resolver.c
+++ b/contrib/restricted/aws/aws-c-io/source/host_resolver.c
@@ -2,6 +2,7 @@
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0.
*/
+#include <aws/io/event_loop.h>
#include <aws/io/host_resolver.h>
#include <aws/common/atomics.h>
@@ -11,6 +12,7 @@
#include <aws/common/lru_cache.h>
#include <aws/common/mutex.h>
#include <aws/common/string.h>
+#include <aws/common/task_scheduler.h>
#include <aws/common/thread.h>
#include <aws/io/logging.h>
@@ -22,18 +24,7 @@ const uint64_t NS_PER_SEC = 1000000000;
int aws_host_address_copy(const struct aws_host_address *from, struct aws_host_address *to) {
to->allocator = from->allocator;
to->address = aws_string_new_from_string(to->allocator, from->address);
-
- if (!to->address) {
- return AWS_OP_ERR;
- }
-
to->host = aws_string_new_from_string(to->allocator, from->host);
-
- if (!to->host) {
- aws_string_destroy((void *)to->address);
- return AWS_OP_ERR;
- }
-
to->record_type = from->record_type;
to->use_count = from->use_count;
to->connection_failure_count = from->connection_failure_count;
@@ -80,7 +71,37 @@ int aws_host_resolver_purge_cache(struct aws_host_resolver *resolver) {
return resolver->vtable->purge_cache(resolver);
}
-int aws_host_resolver_record_connection_failure(struct aws_host_resolver *resolver, struct aws_host_address *address) {
+int aws_host_resolver_purge_cache_with_callback(
+ struct aws_host_resolver *resolver,
+ aws_simple_completion_callback *on_purge_cache_complete_callback,
+ void *user_data) {
+ AWS_PRECONDITION(resolver);
+ AWS_PRECONDITION(resolver->vtable);
+
+ if (!resolver->vtable->purge_cache_with_callback) {
+ AWS_LOGF_ERROR(AWS_LS_IO_DNS, "purge_cache_with_callback function is not supported");
+ return aws_raise_error(AWS_ERROR_UNSUPPORTED_OPERATION);
+ }
+
+ return resolver->vtable->purge_cache_with_callback(resolver, on_purge_cache_complete_callback, user_data);
+}
+
+int aws_host_resolver_purge_host_cache(
+ struct aws_host_resolver *resolver,
+ const struct aws_host_resolver_purge_host_options *options) {
+ AWS_PRECONDITION(resolver);
+ AWS_PRECONDITION(resolver->vtable);
+
+ if (!resolver->vtable->purge_host_cache) {
+ AWS_LOGF_ERROR(AWS_LS_IO_DNS, "purge_host_cache function is not supported");
+ return aws_raise_error(AWS_ERROR_UNSUPPORTED_OPERATION);
+ }
+ return resolver->vtable->purge_host_cache(resolver, options);
+}
+
+int aws_host_resolver_record_connection_failure(
+ struct aws_host_resolver *resolver,
+ const struct aws_host_address *address) {
AWS_ASSERT(resolver->vtable && resolver->vtable->record_connection_failure);
return resolver->vtable->record_connection_failure(resolver, address);
}
@@ -153,6 +174,8 @@ struct default_host_resolver {
* Function to use to query current time. Overridable in construction options.
*/
aws_io_clock_fn *system_clock_fn;
+
+ struct aws_event_loop_group *event_loop_group;
};
/* Default host resolver implementation for listener. */
@@ -228,6 +251,8 @@ struct host_entry {
enum default_resolver_state state;
struct aws_array_list new_addresses;
struct aws_array_list expired_addresses;
+ aws_simple_completion_callback *on_host_purge_complete;
+ void *on_host_purge_complete_user_data;
};
/*
@@ -286,10 +311,42 @@ static void s_remove_host_listener_from_entry(
static void s_host_listener_destroy(struct host_listener *listener);
+struct host_purge_callback_options {
+ struct aws_allocator *allocator;
+ struct aws_ref_count ref_count;
+ aws_simple_completion_callback *on_purge_cache_complete_callback;
+ void *user_data;
+};
+
+static void s_host_purge_callback_options_destroy(void *user_data) {
+ struct host_purge_callback_options *options = user_data;
+ options->on_purge_cache_complete_callback(options->user_data);
+ aws_mem_release(options->allocator, options);
+}
+
+static struct host_purge_callback_options *s_host_purge_callback_options_new(
+ struct aws_allocator *allocator,
+ aws_simple_completion_callback *on_purge_cache_complete_callback,
+ void *user_data) {
+ struct host_purge_callback_options *purge_callback_options =
+ aws_mem_calloc(allocator, 1, sizeof(struct host_purge_callback_options));
+ purge_callback_options->allocator = allocator;
+ aws_ref_count_init(
+ &purge_callback_options->ref_count, purge_callback_options, s_host_purge_callback_options_destroy);
+ purge_callback_options->on_purge_cache_complete_callback = on_purge_cache_complete_callback;
+ purge_callback_options->user_data = user_data;
+ return purge_callback_options;
+}
+
+static void s_purge_cache_callback(void *user_data) {
+ struct host_purge_callback_options *purge_callback_options = user_data;
+ aws_ref_count_release(&purge_callback_options->ref_count);
+}
+
/*
* resolver lock must be held before calling this function
*/
-static void s_clear_default_resolver_entry_table(struct default_host_resolver *resolver) {
+static void s_clear_default_resolver_entry_table_synced(struct default_host_resolver *resolver) {
struct aws_hash_table *table = &resolver->host_entry_table;
for (struct aws_hash_iter iter = aws_hash_iter_begin(table); !aws_hash_iter_done(&iter);
aws_hash_iter_next(&iter)) {
@@ -300,18 +357,75 @@ static void s_clear_default_resolver_entry_table(struct default_host_resolver *r
aws_hash_table_clear(table);
}
-static int resolver_purge_cache(struct aws_host_resolver *resolver) {
+static int s_resolver_purge_cache(struct aws_host_resolver *resolver) {
struct default_host_resolver *default_host_resolver = resolver->impl;
aws_mutex_lock(&default_host_resolver->resolver_lock);
- s_clear_default_resolver_entry_table(default_host_resolver);
+ s_clear_default_resolver_entry_table_synced(default_host_resolver);
aws_mutex_unlock(&default_host_resolver->resolver_lock);
return AWS_OP_SUCCESS;
}
+static void s_purge_host_cache_callback_task(struct aws_task *task, void *arg, enum aws_task_status status) {
+ (void)status;
+ struct host_purge_callback_options *options = arg;
+ aws_mem_release(options->allocator, task);
+ aws_ref_count_release(&options->ref_count);
+}
+
+static void s_sechdule_purge_cache_callback_async(
+ struct default_host_resolver *default_host_resolver,
+ struct host_purge_callback_options *purge_callback_options) {
+
+ struct aws_task *task = aws_mem_calloc(default_host_resolver->allocator, 1, sizeof(struct aws_task));
+ aws_task_init(task, s_purge_host_cache_callback_task, purge_callback_options, "async_purge_host_callback_task");
+
+ struct aws_event_loop *loop = aws_event_loop_group_get_next_loop(default_host_resolver->event_loop_group);
+ AWS_FATAL_ASSERT(loop != NULL);
+ aws_event_loop_schedule_task_now(loop, task);
+}
+
+static int s_resolver_purge_cache_with_callback(
+ struct aws_host_resolver *resolver,
+ aws_simple_completion_callback *on_purge_cache_complete_callback,
+ void *user_data) {
+
+ if (!on_purge_cache_complete_callback) {
+ return s_resolver_purge_cache(resolver);
+ }
+
+ struct default_host_resolver *default_host_resolver = resolver->impl;
+ aws_mutex_lock(&default_host_resolver->resolver_lock);
+ struct aws_hash_table *table = &default_host_resolver->host_entry_table;
+
+ struct host_purge_callback_options *purge_callback_options = s_host_purge_callback_options_new(
+ default_host_resolver->allocator, on_purge_cache_complete_callback, user_data);
+
+ /* purge all cache */
+ for (struct aws_hash_iter iter = aws_hash_iter_begin(table); !aws_hash_iter_done(&iter);
+ aws_hash_iter_next(&iter)) {
+ struct host_entry *entry = iter.element.value;
+ /* acquire a refernce to wait for the callback to trigger */
+ aws_ref_count_acquire(&purge_callback_options->ref_count);
+ aws_mutex_lock(&entry->entry_lock);
+ entry->on_host_purge_complete = s_purge_cache_callback;
+ entry->on_host_purge_complete_user_data = purge_callback_options;
+ entry->state = DRS_SHUTTING_DOWN;
+ aws_mutex_unlock(&entry->entry_lock);
+ }
+
+ aws_hash_table_clear(table);
+ aws_mutex_unlock(&default_host_resolver->resolver_lock);
+
+ /* release the original reference async */
+ s_sechdule_purge_cache_callback_async(default_host_resolver, purge_callback_options);
+ return AWS_OP_SUCCESS;
+}
+
static void s_cleanup_default_resolver(struct aws_host_resolver *resolver) {
struct default_host_resolver *default_host_resolver = resolver->impl;
+ aws_event_loop_group_release(default_host_resolver->event_loop_group);
aws_hash_table_clean_up(&default_host_resolver->host_entry_table);
aws_hash_table_clean_up(&default_host_resolver->listener_entry_table);
@@ -337,7 +451,7 @@ static void resolver_destroy(struct aws_host_resolver *resolver) {
AWS_FATAL_ASSERT(default_host_resolver->state == DRS_ACTIVE);
- s_clear_default_resolver_entry_table(default_host_resolver);
+ s_clear_default_resolver_entry_table_synced(default_host_resolver);
default_host_resolver->state = DRS_SHUTTING_DOWN;
if (default_host_resolver->pending_host_entry_shutdown_completion_callbacks == 0) {
cleanup_resolver = true;
@@ -547,7 +661,58 @@ static inline void process_records(
}
}
-static int resolver_record_connection_failure(struct aws_host_resolver *resolver, struct aws_host_address *address) {
+static int s_resolver_purge_host_cache(
+ struct aws_host_resolver *resolver,
+ const struct aws_host_resolver_purge_host_options *options) {
+
+ AWS_PRECONDITION(resolver);
+ if (options == NULL) {
+ AWS_LOGF_ERROR(AWS_LS_IO_DNS, "Cannot purge host cache; options structure is NULL.");
+ return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
+ }
+
+ struct default_host_resolver *default_host_resolver = resolver->impl;
+ AWS_LOGF_INFO(AWS_LS_IO_DNS, "id=%p: purging record for %s", (void *)resolver, options->host->bytes);
+
+ aws_mutex_lock(&default_host_resolver->resolver_lock);
+
+ struct aws_hash_element *element = NULL;
+ aws_hash_table_find(&default_host_resolver->host_entry_table, options->host, &element);
+
+ /* Success if entry doesn't exist in cache. */
+ if (element == NULL) {
+ aws_mutex_unlock(&default_host_resolver->resolver_lock);
+ if (options->on_host_purge_complete_callback != NULL) {
+ /* Schedule completion callback asynchronouly */
+ struct host_purge_callback_options *purge_callback_options = s_host_purge_callback_options_new(
+ default_host_resolver->allocator, options->on_host_purge_complete_callback, options->user_data);
+
+ s_sechdule_purge_cache_callback_async(default_host_resolver, purge_callback_options);
+ }
+ return AWS_OP_SUCCESS;
+ }
+
+ struct host_entry *host_entry = element->value;
+ AWS_FATAL_ASSERT(host_entry);
+
+ /* Setup the on_host_purge_complete callback. */
+ aws_mutex_lock(&host_entry->entry_lock);
+ AWS_FATAL_ASSERT(!host_entry->on_host_purge_complete);
+ AWS_FATAL_ASSERT(!host_entry->on_host_purge_complete_user_data);
+ host_entry->on_host_purge_complete = options->on_host_purge_complete_callback;
+ host_entry->on_host_purge_complete_user_data = options->user_data;
+ aws_mutex_unlock(&host_entry->entry_lock);
+
+ s_shutdown_host_entry(host_entry);
+ aws_hash_table_remove_element(&default_host_resolver->host_entry_table, element);
+
+ aws_mutex_unlock(&default_host_resolver->resolver_lock);
+ return AWS_OP_SUCCESS;
+}
+
+static int resolver_record_connection_failure(
+ struct aws_host_resolver *resolver,
+ const struct aws_host_address *address) {
struct default_host_resolver *default_host_resolver = resolver->impl;
AWS_LOGF_INFO(
@@ -1211,6 +1376,10 @@ done:
aws_array_list_clean_up(&new_address_list);
aws_array_list_clean_up(&expired_address_list);
+ /* trigger the purge complete callback */
+ if (host_entry->on_host_purge_complete != NULL) {
+ host_entry->on_host_purge_complete(host_entry->on_host_purge_complete_user_data);
+ }
/* please don't fail */
aws_thread_current_at_exit(s_on_host_entry_shutdown_completion, host_entry);
}
@@ -1346,10 +1515,7 @@ static inline int create_and_init_host_entry(
new_host_entry->resolution_config = *config;
aws_condition_variable_init(&new_host_entry->entry_signal);
- if (aws_thread_init(&new_host_entry->resolver_thread, resolver->allocator)) {
- goto setup_host_entry_error;
- }
-
+ aws_thread_init(&new_host_entry->resolver_thread, resolver->allocator);
thread_init = true;
struct default_host_resolver *default_host_resolver = resolver->impl;
if (AWS_UNLIKELY(
@@ -1360,8 +1526,10 @@ static inline int create_and_init_host_entry(
struct aws_thread_options thread_options = *aws_default_thread_options();
thread_options.join_strategy = AWS_TJS_MANAGED;
thread_options.name = aws_byte_cursor_from_c_str("AwsHostResolver"); /* 15 characters is max for Linux */
-
- aws_thread_launch(&new_host_entry->resolver_thread, aws_host_resolver_thread, new_host_entry, &thread_options);
+ if (aws_thread_launch(
+ &new_host_entry->resolver_thread, aws_host_resolver_thread, new_host_entry, &thread_options)) {
+ goto setup_host_entry_error;
+ }
++default_host_resolver->pending_host_entry_shutdown_completion_callbacks;
return AWS_OP_SUCCESS;
@@ -1371,6 +1539,10 @@ setup_host_entry_error:
if (thread_init) {
aws_thread_clean_up(&new_host_entry->resolver_thread);
}
+ // If we registered a callback, clear it. So that we don’t trigger callback and return an error.
+ if (!aws_linked_list_empty(&new_host_entry->pending_resolution_callbacks)) {
+ aws_linked_list_remove(&pending_callback->node);
+ }
s_clean_up_host_entry(new_host_entry);
@@ -1446,37 +1618,31 @@ static int default_resolve_host(
/* these will all need to be copied so that we don't hold the lock during the callback. */
if (aaaa_record) {
struct aws_host_address aaaa_record_cpy;
- if (!aws_host_address_copy(aaaa_record, &aaaa_record_cpy)) {
- aws_array_list_push_back(&callback_address_list, &aaaa_record_cpy);
- AWS_LOGF_TRACE(
- AWS_LS_IO_DNS,
- "id=%p: vending address %s for host %s to caller",
- (void *)resolver,
- aaaa_record->address->bytes,
- host_entry->host_name->bytes);
- }
+ aws_host_address_copy(aaaa_record, &aaaa_record_cpy);
+ aws_array_list_push_back(&callback_address_list, &aaaa_record_cpy);
+ AWS_LOGF_TRACE(
+ AWS_LS_IO_DNS,
+ "id=%p: vending address %s for host %s to caller",
+ (void *)resolver,
+ aaaa_record->address->bytes,
+ host_entry->host_name->bytes);
}
if (a_record) {
struct aws_host_address a_record_cpy;
- if (!aws_host_address_copy(a_record, &a_record_cpy)) {
- aws_array_list_push_back(&callback_address_list, &a_record_cpy);
- AWS_LOGF_TRACE(
- AWS_LS_IO_DNS,
- "id=%p: vending address %s for host %s to caller",
- (void *)resolver,
- a_record->address->bytes,
- host_entry->host_name->bytes);
- }
+ aws_host_address_copy(a_record, &a_record_cpy);
+ aws_array_list_push_back(&callback_address_list, &a_record_cpy);
+ AWS_LOGF_TRACE(
+ AWS_LS_IO_DNS,
+ "id=%p: vending address %s for host %s to caller",
+ (void *)resolver,
+ a_record->address->bytes,
+ host_entry->host_name->bytes);
}
aws_mutex_unlock(&host_entry->entry_lock);
/* we don't want to do the callback WHILE we hold the lock someone may reentrantly call us. */
- if (aws_array_list_length(&callback_address_list)) {
- res(resolver, host_name, AWS_OP_SUCCESS, &callback_address_list, user_data);
- } else {
- res(resolver, host_name, aws_last_error(), NULL, user_data);
- result = AWS_OP_ERR;
- }
+ // TODO: Fire the callback asynchronously
+ res(resolver, host_name, AWS_OP_SUCCESS, &callback_address_list, user_data);
for (size_t i = 0; i < aws_array_list_length(&callback_address_list); ++i) {
struct aws_host_address *address_ptr = NULL;
@@ -1538,13 +1704,15 @@ static size_t default_get_host_address_count(
}
static struct aws_host_resolver_vtable s_vtable = {
- .purge_cache = resolver_purge_cache,
+ .purge_cache = s_resolver_purge_cache,
+ .purge_cache_with_callback = s_resolver_purge_cache_with_callback,
.resolve_host = default_resolve_host,
.record_connection_failure = resolver_record_connection_failure,
.get_host_address_count = default_get_host_address_count,
.add_host_listener = default_add_host_listener,
.remove_host_listener = default_remove_host_listener,
.destroy = resolver_destroy,
+ .purge_host_cache = s_resolver_purge_host_cache,
};
static void s_aws_host_resolver_destroy(struct aws_host_resolver *resolver) {
@@ -1557,9 +1725,6 @@ struct aws_host_resolver *aws_host_resolver_new_default(
const struct aws_host_resolver_default_options *options) {
AWS_FATAL_ASSERT(options != NULL);
- /* NOTE: we don't use el_group yet, but we will in the future. Also, we
- don't want host resolvers getting cleaned up after el_groups; this will force that
- in bindings, and encourage it in C land. */
AWS_ASSERT(options->el_group);
struct aws_host_resolver *resolver = NULL;
@@ -1587,6 +1752,7 @@ struct aws_host_resolver *aws_host_resolver_new_default(
resolver->allocator = allocator;
resolver->impl = default_host_resolver;
+ default_host_resolver->event_loop_group = aws_event_loop_group_acquire(options->el_group);
default_host_resolver->allocator = allocator;
default_host_resolver->pending_host_entry_shutdown_completion_callbacks = 0;
default_host_resolver->state = DRS_ACTIVE;