diff --git a/Makefile.am b/Makefile.am index 56aa08b3e87..3f5a987cbd0 100644 --- a/Makefile.am +++ b/Makefile.am @@ -61,7 +61,8 @@ common_srcs = \ prov/util/src/util_mr_map.c \ prov/util/src/util_ns.c \ prov/util/src/util_shm.c \ - prov/util/src/util_mem_monitor.c + prov/util/src/util_mem_monitor.c\ + prov/util/src/util_mr_cache.c if MACOS common_srcs += src/unix/osd.c diff --git a/include/fi_iov.h b/include/fi_iov.h index aa27b05174f..154b81ca7ad 100644 --- a/include/fi_iov.h +++ b/include/fi_iov.h @@ -36,6 +36,7 @@ #include "config.h" +#include #include #include #include @@ -82,6 +83,30 @@ ofi_copy_from_iov(void *buf, uint64_t bufsize, OFI_COPY_IOV_TO_BUF); } +static inline void * +ofi_iov_end(const struct iovec *iov) +{ + return ((char *) iov->iov_base) + iov->iov_len; +} + +static inline bool +ofi_iov_left(const struct iovec *iov1, const struct iovec *iov2) +{ + return ofi_iov_end(iov1) < iov2->iov_base; +} + +static inline bool +ofi_iov_right(const struct iovec *iov1, const struct iovec *iov2) +{ + return iov1->iov_base > ofi_iov_end(iov2); +} + +static inline bool +ofi_iov_within(const struct iovec *iov1, const struct iovec *iov2) +{ + return (iov1->iov_base >= iov2->iov_base) && + (ofi_iov_end(iov1) <= ofi_iov_end(iov2)); +} #endif /* IOV_H */ diff --git a/include/ofi_mr.h b/include/ofi_mr.h index 5f09eeb5ea8..ffaf0e79dd2 100644 --- a/include/ofi_mr.h +++ b/include/ofi_mr.h @@ -38,11 +38,14 @@ #endif /* HAVE_CONFIG_H */ #include +#include #include #include #include #include +#include + #define OFI_MR_BASIC_MAP (FI_MR_ALLOCATED | FI_MR_PROV_KEY | FI_MR_VIRT_ADDR) @@ -74,6 +77,7 @@ static inline uint64_t ofi_mr_get_prov_mode(uint32_t version, } } + /* * Memory notifier - Report memory mapping changes to address ranges */ @@ -113,9 +117,10 @@ int ofi_monitor_subscribe(struct ofi_notification_queue *nq, void *addr, size_t len, struct ofi_subscription *subscription); void ofi_monitor_unsubscribe(void *addr, size_t len, - struct ofi_subscription *subscription); + struct ofi_subscription *subscription); struct ofi_subscription *ofi_monitor_get_event(struct ofi_notification_queue *nq); + /* * MR map */ @@ -141,4 +146,50 @@ int ofi_mr_map_verify(struct ofi_mr_map *map, uintptr_t *io_addr, size_t len, uint64_t key, uint64_t access, void **context); + +/* + * Memory registration cache + */ + +struct ofi_mr_entry { + struct iovec iov; + unsigned int cached:1; + unsigned int subscribed:1; + int use_cnt; + struct dlist_entry lru_entry; + struct ofi_subscription subscription; + uint8_t data[]; +}; + +struct ofi_mr_cache { + struct util_domain *domain; + struct ofi_notification_queue nq; + size_t size; + size_t entry_data_size; + + RbtHandle mr_tree; + struct dlist_entry lru_list; + + uint64_t cached_cnt; + uint64_t search_cnt; + uint64_t delete_cnt; + uint64_t hit_cnt; + + int (*add_region)(struct ofi_mr_cache *cache, + struct ofi_mr_entry *entry); + void (*delete_region)(struct ofi_mr_cache *cache, + struct ofi_mr_entry *entry); +}; + +int ofi_mr_cache_init(struct util_domain *domain, struct ofi_mem_monitor *monitor, + struct ofi_mr_cache *cache); +void ofi_mr_cache_cleanup(struct ofi_mr_cache *cache); + +/* Caller must provide locking around calls */ +bool ofi_mr_cache_flush(struct ofi_mr_cache *cache); +int ofi_mr_cache_search(struct ofi_mr_cache *cache, const struct fi_mr_attr *attr, + struct ofi_mr_entry **entry); +void ofi_mr_cache_delete(struct ofi_mr_cache *cache, struct ofi_mr_entry *entry); + + #endif /* _OFI_MR_H_ */ diff --git a/prov/util/src/util_mr_cache.c b/prov/util/src/util_mr_cache.c new file mode 100644 index 00000000000..1a99987ac62 --- /dev/null +++ b/prov/util/src/util_mr_cache.c @@ -0,0 +1,292 @@ +/* + * Copyright (c) 2016-2017 Cray Inc. All rights reserved. + * Copyright (c) 2017 Intel Corporation, Inc. All rights reserved. + * + * This software is available to you under a choice of one of two + * licenses. You may choose to be licensed under the terms of the GNU + * General Public License (GPL) Version 2, available from the file + * COPYING in the main directory of this source tree, or the + * BSD license below: + * + * Redistribution and use in source and binary forms, with or + * without modification, are permitted provided that the following + * conditions are met: + * + * - Redistributions of source code must retain the above + * copyright notice, this list of conditions and the following + * disclaimer. + * + * - Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials + * provided with the distribution. + * + * 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 +#include +#include +#include +#include +#include + + +static int util_mr_find_overlap(void *a, void *b) +{ + struct iovec *iov1 = a, *iov2 = b; + + if (ofi_iov_left(iov1, iov2)) + return -1; + else if (ofi_iov_right(iov1, iov2)) + return 1; + else + return 0; +} + +static void util_mr_free_entry(struct ofi_mr_cache *cache, + struct ofi_mr_entry *entry) +{ + FI_DBG(cache->domain->prov, FI_LOG_MR, "free %p (len: %" PRIu64 ")\n", + entry->iov.iov_base, entry->iov.iov_len); + + assert(!entry->cached); + if (entry->subscribed) { + ofi_monitor_unsubscribe(entry->iov.iov_base, entry->iov.iov_len, + &entry->subscription); + } + cache->delete_region(cache, entry); + free(entry); + cache->cached_cnt--; +} + +static void util_mr_uncache_entry(struct ofi_mr_cache *cache, + struct ofi_mr_entry *entry) +{ + RbtIterator iter; + + assert(entry->cached); + iter = rbtFind(cache->mr_tree, &entry->iov); + assert(iter); + rbtErase(cache->mr_tree, iter); + entry->cached = 0; +} + +static void +util_mr_cache_process_events(struct ofi_mr_cache *cache) +{ + struct ofi_subscription *subscription; + struct ofi_mr_entry *entry; + + while ((subscription = ofi_monitor_get_event(&cache->nq))) { + entry = container_of(subscription, struct ofi_mr_entry, + subscription); + if (entry->cached) + util_mr_uncache_entry(cache, entry); + + if (entry->use_cnt == 0) { + dlist_remove(&entry->lru_entry); + util_mr_free_entry(cache, entry); + } + } +} + +bool ofi_mr_cache_flush(struct ofi_mr_cache *cache) +{ + struct ofi_mr_entry *entry; + + if (dlist_empty(&cache->lru_list)) + return false; + + dlist_pop_front(&cache->lru_list, struct ofi_mr_entry, + entry, lru_entry); + FI_DBG(cache->domain->prov, FI_LOG_MR, "flush %p (len: %" PRIu64 ")\n", + entry->iov.iov_base, entry->iov.iov_len); + + util_mr_uncache_entry(cache, entry); + util_mr_free_entry(cache, entry); + return true; +} + +void ofi_mr_cache_delete(struct ofi_mr_cache *cache, struct ofi_mr_entry *entry) +{ + FI_DBG(cache->domain->prov, FI_LOG_MR, "delete %p (len: %" PRIu64 ")\n", + entry->iov.iov_base, entry->iov.iov_len); + cache->delete_cnt++; + + util_mr_cache_process_events(cache); + + if (--entry->use_cnt == 0) { + if (entry->cached) { + dlist_insert_tail(&entry->lru_entry, &cache->lru_list); + } else { + util_mr_free_entry(cache, entry); + } + } +} + +static int +util_mr_cache_create(struct ofi_mr_cache *cache, const struct iovec *iov, + uint64_t access, struct ofi_mr_entry **entry) +{ + int ret; + + FI_DBG(cache->domain->prov, FI_LOG_MR, "create %p (len: %" PRIu64 ")\n", + iov->iov_base, iov->iov_len); + + *entry = calloc(1, sizeof(**entry) + cache->entry_data_size); + if (!*entry) + return -FI_ENOMEM; + + (*entry)->iov = *iov; + (*entry)->use_cnt = 1; + + ret = cache->add_region(cache, *entry); + if (ret) { + free(*entry); + return ret; + } + + if (++cache->cached_cnt > cache->size) { + (*entry)->cached = 0; + } else { + ret = ofi_monitor_subscribe(&cache->nq, iov->iov_base, iov->iov_len, + &(*entry)->subscription); + if (ret) + goto err; + (*entry)->subscribed = 1; + + if (rbtInsert(cache->mr_tree, &(*entry)->iov, *entry)) { + ret = -FI_ENOMEM; + goto err; + } + (*entry)->cached = 1; + } + + return 0; + +err: + util_mr_free_entry(cache, *entry); + return ret; +} + +static int +util_mr_cache_merge(struct ofi_mr_cache *cache, const struct fi_mr_attr *attr, + RbtIterator iter, struct ofi_mr_entry **entry) +{ + struct iovec iov, *old_iov; + struct ofi_mr_entry *old_entry; + + iov = *attr->mr_iov; + do { + rbtKeyValue(cache->mr_tree, iter, (void **) &old_iov, + (void **) &old_entry); + + FI_DBG(cache->domain->prov, FI_LOG_MR, + "merging %p (len: %" PRIu64 ") with %p (len: %" PRIu64 ")\n", + iov.iov_base, iov.iov_len, + old_entry->iov.iov_base, old_entry->iov.iov_len); + + iov.iov_len = ((uintptr_t) + MAX(ofi_iov_end(&iov), ofi_iov_end(old_iov))) - + ((uintptr_t) MIN(iov.iov_base, old_iov->iov_base)); + iov.iov_base = MIN(iov.iov_base, old_iov->iov_base); + FI_DBG(cache->domain->prov, FI_LOG_MR, "merged %p (len: %" PRIu64 ")\n", + iov.iov_base, iov.iov_len); + + rbtErase(cache->mr_tree, iter); + old_entry->cached = 0; + if (old_entry->use_cnt == 0) { + dlist_remove(&old_entry->lru_entry); + util_mr_free_entry(cache, old_entry); + } + + } while ((iter = rbtFind(cache->mr_tree, &iov))); + + return util_mr_cache_create(cache, &iov, attr->access, entry); +} + +int ofi_mr_cache_search(struct ofi_mr_cache *cache, const struct fi_mr_attr *attr, + struct ofi_mr_entry **entry) +{ + RbtIterator iter; + struct iovec *iov; + + util_mr_cache_process_events(cache); + + assert(attr->iov_count == 1); + FI_DBG(cache->domain->prov, FI_LOG_MR, "search %p (len: %" PRIu64 ")\n", + attr->mr_iov->iov_base, attr->mr_iov->iov_len); + cache->search_cnt++; + + while ((cache->cached_cnt >= cache->size) && ofi_mr_cache_flush(cache)) + ; + + iter = rbtFind(cache->mr_tree, (void *) attr->mr_iov); + if (!iter) { + return util_mr_cache_create(cache, attr->mr_iov, + attr->access, entry); + } + + rbtKeyValue(cache->mr_tree, iter, (void **) &iov, (void **) entry); + + if (!ofi_iov_within(attr->mr_iov, iov)) + return util_mr_cache_merge(cache, attr, iter, entry); + + cache->hit_cnt++; + if ((*entry)->use_cnt++ == 0) + dlist_remove(&(*entry)->lru_entry); + + return 0; +} + +void ofi_mr_cache_cleanup(struct ofi_mr_cache *cache) +{ + struct ofi_mr_entry *entry; + struct dlist_entry *tmp; + + FI_INFO(cache->domain->prov, FI_LOG_MR, "MR cache stats: " + "searches %" PRIu64 ", deletes %" PRIu64 ", hits %" PRIu64 "\n", + cache->search_cnt, cache->delete_cnt, cache->hit_cnt); + + dlist_foreach_container_safe(&cache->lru_list, struct ofi_mr_entry, + entry, lru_entry, tmp) { + assert(entry->use_cnt == 0); + util_mr_uncache_entry(cache, entry); + dlist_remove(&entry->lru_entry); + util_mr_free_entry(cache, entry); + } + rbtDelete(cache->mr_tree); + ofi_monitor_del_queue(&cache->nq); + ofi_atomic_dec32(&cache->domain->ref); + assert(cache->cached_cnt == 0); +} + +int ofi_mr_cache_init(struct util_domain *domain, struct ofi_mem_monitor *monitor, + struct ofi_mr_cache *cache) +{ + assert(cache->add_region && cache->delete_region); + + cache->mr_tree = rbtNew(util_mr_find_overlap); + if (!cache->mr_tree) + return -FI_ENOMEM; + + cache->domain = domain; + ofi_atomic_inc32(&domain->ref); + + dlist_init(&cache->lru_list); + cache->cached_cnt = 0; + cache->search_cnt = 0; + cache->delete_cnt = 0; + cache->hit_cnt = 0; + ofi_monitor_add_queue(monitor, &cache->nq); + + return 0; +}