Blame memkind-1.10.1/src/memkind_default.c

Packit Service 7f3b24
// SPDX-License-Identifier: BSD-2-Clause
Packit Service 7f3b24
/* Copyright (C) 2014 - 2020 Intel Corporation. */
Packit Service 7f3b24
Packit Service 7f3b24
#include <memkind/internal/memkind_arena.h>
Packit Service 7f3b24
#include <memkind/internal/memkind_default.h>
Packit Service 7f3b24
#include <memkind/internal/memkind_private.h>
Packit Service 7f3b24
#include <memkind/internal/memkind_log.h>
Packit Service 7f3b24
#include <memkind/internal/tbb_wrapper.h>
Packit Service 7f3b24
#include <memkind/internal/heap_manager.h>
Packit Service 7f3b24
Packit Service 7f3b24
#include <numa.h>
Packit Service 7f3b24
#include <numaif.h>
Packit Service 7f3b24
#include <sys/mman.h>
Packit Service 7f3b24
#include <errno.h>
Packit Service 7f3b24
#include <jemalloc/jemalloc.h>
Packit Service 7f3b24
#include <stdint.h>
Packit Service 7f3b24
Packit Service 7f3b24
#include "config.h"
Packit Service 7f3b24
Packit Service 7f3b24
#ifndef MADV_NOHUGEPAGE
Packit Service 7f3b24
#define MADV_NOHUGEPAGE 15
Packit Service 7f3b24
#endif
Packit Service 7f3b24
Packit Service 7f3b24
static int memkind_default_get_kind_stat(struct memkind *kind,
Packit Service 7f3b24
                                         memkind_stat_type stat, size_t *value)
Packit Service 7f3b24
{
Packit Service 7f3b24
    return memkind_arena_get_stat_with_check_init(kind, stat, true, value);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT struct memkind_ops MEMKIND_DEFAULT_OPS = {
Packit Service 7f3b24
    .create = memkind_default_create,
Packit Service 7f3b24
    .destroy = memkind_default_destroy,
Packit Service 7f3b24
    .malloc = memkind_default_malloc,
Packit Service 7f3b24
    .calloc = memkind_default_calloc,
Packit Service 7f3b24
    .posix_memalign = memkind_default_posix_memalign,
Packit Service 7f3b24
    .realloc = memkind_default_realloc,
Packit Service 7f3b24
    .free = memkind_default_free,
Packit Service 7f3b24
    .init_once = memkind_default_init_once,
Packit Service 7f3b24
    .malloc_usable_size = memkind_default_malloc_usable_size,
Packit Service 7f3b24
    .finalize = memkind_default_destroy,
Packit Service 7f3b24
    .get_stat = memkind_default_get_kind_stat,
Packit Service 7f3b24
    .defrag_reallocate = memkind_arena_defrag_reallocate
Packit Service 7f3b24
};
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_create(struct memkind *kind,
Packit Service 7f3b24
                                          struct memkind_ops *ops, const char *name)
Packit Service 7f3b24
{
Packit Service 7f3b24
    int err = 0;
Packit Service 7f3b24
Packit Service 7f3b24
    kind->ops = ops;
Packit Service 7f3b24
    if (strlen(name) >= MEMKIND_NAME_LENGTH_PRIV) {
Packit Service 7f3b24
        kind->name[0] = '\0';
Packit Service 7f3b24
        err = MEMKIND_ERROR_INVALID;
Packit Service 7f3b24
    } else {
Packit Service 7f3b24
        strcpy(kind->name, name);
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return err;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_destroy(struct memkind *kind)
Packit Service 7f3b24
{
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void *memkind_default_malloc(struct memkind *kind, size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    if(MEMKIND_UNLIKELY(size_out_of_bounds(size))) {
Packit Service 7f3b24
        return NULL;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return jemk_malloc(size);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void *memkind_default_calloc(struct memkind *kind, size_t num,
Packit Service 7f3b24
                                            size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    if(MEMKIND_UNLIKELY(size_out_of_bounds(num) || size_out_of_bounds(size))) {
Packit Service 7f3b24
        return NULL;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return jemk_calloc(num, size);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_posix_memalign(struct memkind *kind,
Packit Service 7f3b24
                                                  void **memptr, size_t alignment, size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    if(MEMKIND_UNLIKELY(size_out_of_bounds(size))) {
Packit Service 7f3b24
        *memptr = NULL;
Packit Service 7f3b24
        return 0;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return jemk_posix_memalign(memptr, alignment, size);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void *memkind_default_realloc(struct memkind *kind, void *ptr,
Packit Service 7f3b24
                                             size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    if(MEMKIND_UNLIKELY(size_out_of_bounds(size))) {
Packit Service 7f3b24
        jemk_free(ptr);
Packit Service 7f3b24
        return NULL;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return jemk_realloc(ptr, size);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void memkind_default_free(struct memkind *kind, void *ptr)
Packit Service 7f3b24
{
Packit Service 7f3b24
    jemk_free(ptr);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT size_t memkind_default_malloc_usable_size(struct memkind *kind,
Packit Service 7f3b24
                                                         void *ptr)
Packit Service 7f3b24
{
Packit Service 7f3b24
    return jemk_malloc_usable_size(ptr);
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void *memkind_default_mmap(struct memkind *kind, void *addr,
Packit Service 7f3b24
                                          size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    void *result = MAP_FAILED;
Packit Service 7f3b24
    int err = 0;
Packit Service 7f3b24
    int flags;
Packit Service 7f3b24
Packit Service 7f3b24
    if (kind->ops->get_mmap_flags) {
Packit Service 7f3b24
        err = kind->ops->get_mmap_flags(kind, &flags);
Packit Service 7f3b24
    } else {
Packit Service 7f3b24
        err = memkind_default_get_mmap_flags(kind, &flags);
Packit Service 7f3b24
    }
Packit Service 7f3b24
    if (MEMKIND_LIKELY(!err)) {
Packit Service 7f3b24
        result = mmap(addr, size, PROT_READ | PROT_WRITE, flags, -1, 0);
Packit Service 7f3b24
        if (result == MAP_FAILED) {
Packit Service 7f3b24
            log_err("syscall mmap() returned: %p", result);
Packit Service 7f3b24
            return result;
Packit Service 7f3b24
        }
Packit Service 7f3b24
    }
Packit Service 7f3b24
    if (kind->ops->mbind) {
Packit Service 7f3b24
        err = kind->ops->mbind(kind, result, size);
Packit Service 7f3b24
        if (err) {
Packit Service 7f3b24
            munmap(result, size);
Packit Service 7f3b24
            result = MAP_FAILED;
Packit Service 7f3b24
        }
Packit Service 7f3b24
    }
Packit Service 7f3b24
    if (kind->ops->madvise) {
Packit Service 7f3b24
        err = kind->ops->madvise(kind, result, size);
Packit Service 7f3b24
        if (err) {
Packit Service 7f3b24
            munmap(result, size);
Packit Service 7f3b24
            result = MAP_FAILED;
Packit Service 7f3b24
        }
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return result;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_nohugepage_madvise(struct memkind *kind, void *addr,
Packit Service 7f3b24
                                              size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    int err = madvise(addr, size, MADV_NOHUGEPAGE);
Packit Service 7f3b24
Packit Service 7f3b24
    //checking if EINVAL was returned due to lack of THP support in kernel
Packit Service 7f3b24
    if ((err == EINVAL) && (((uintptr_t) addr & (uintptr_t) 0xfff) == 0) &&
Packit Service 7f3b24
        (size > 0)) {
Packit Service 7f3b24
        return 0;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    if (MEMKIND_UNLIKELY(err)) {
Packit Service 7f3b24
        log_err("syscall madvise() returned: %d", err);
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return err;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_mbind(struct memkind *kind, void *ptr,
Packit Service 7f3b24
                                         size_t size)
Packit Service 7f3b24
{
Packit Service 7f3b24
    nodemask_t nodemask;
Packit Service 7f3b24
    int err = 0;
Packit Service 7f3b24
    int mode;
Packit Service 7f3b24
Packit Service 7f3b24
    if (MEMKIND_UNLIKELY(kind->ops->get_mbind_nodemask == NULL ||
Packit Service 7f3b24
                         kind->ops->get_mbind_mode == NULL)) {
Packit Service 7f3b24
        log_err("memkind_ops->mbind_mode or memkind_ops->bind_nodemask is NULL.");
Packit Service 7f3b24
        return MEMKIND_ERROR_BADOPS;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    err = kind->ops->get_mbind_nodemask(kind, nodemask.n, NUMA_NUM_NODES);
Packit Service 7f3b24
    if (MEMKIND_UNLIKELY(err)) {
Packit Service 7f3b24
        return err;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    err = kind->ops->get_mbind_mode(kind, &mode);
Packit Service 7f3b24
    if (MEMKIND_UNLIKELY(err)) {
Packit Service 7f3b24
        return err;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    err = mbind(ptr, size, mode, nodemask.n, NUMA_NUM_NODES, 0);
Packit Service 7f3b24
    if (MEMKIND_UNLIKELY(err)) {
Packit Service 7f3b24
        log_err("syscall mbind() returned: %d", err);
Packit Service 7f3b24
        return MEMKIND_ERROR_MBIND;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return err;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_get_mmap_flags(struct memkind *kind,
Packit Service 7f3b24
                                                  int *flags)
Packit Service 7f3b24
{
Packit Service 7f3b24
    *flags = MAP_PRIVATE | MAP_ANONYMOUS;
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_get_mbind_nodemask(struct memkind *kind,
Packit Service 7f3b24
                                                      unsigned long *nodemask,
Packit Service 7f3b24
                                                      unsigned long maxnode)
Packit Service 7f3b24
{
Packit Service 7f3b24
    struct bitmask nodemask_bm = {maxnode, nodemask};
Packit Service 7f3b24
    copy_bitmask_to_bitmask(numa_all_nodes_ptr, &nodemask_bm);
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_default_get_mbind_mode(struct memkind *kind,
Packit Service 7f3b24
                                                  int *mode)
Packit Service 7f3b24
{
Packit Service 7f3b24
    *mode = MPOL_BIND;
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_preferred_get_mbind_mode(struct memkind *kind,
Packit Service 7f3b24
                                                    int *mode)
Packit Service 7f3b24
{
Packit Service 7f3b24
    *mode = MPOL_PREFERRED;
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_interleave_get_mbind_mode(struct memkind *kind,
Packit Service 7f3b24
                                                     int *mode)
Packit Service 7f3b24
{
Packit Service 7f3b24
    *mode = MPOL_INTERLEAVE;
Packit Service 7f3b24
    return 0;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT int memkind_posix_check_alignment(struct memkind *kind,
Packit Service 7f3b24
                                                 size_t alignment)
Packit Service 7f3b24
{
Packit Service 7f3b24
    int err = 0;
Packit Service 7f3b24
    if ((alignment < sizeof(void *)) ||
Packit Service 7f3b24
        (((alignment - 1) & alignment) != 0)) {
Packit Service 7f3b24
        err = EINVAL;
Packit Service 7f3b24
    }
Packit Service 7f3b24
    return err;
Packit Service 7f3b24
}
Packit Service 7f3b24
Packit Service 7f3b24
MEMKIND_EXPORT void memkind_default_init_once(void)
Packit Service 7f3b24
{
Packit Service 7f3b24
#ifdef MEMKIND_ENABLE_HEAP_MANAGER
Packit Service 7f3b24
    heap_manager_init(MEMKIND_DEFAULT);
Packit Service 7f3b24
#endif
Packit Service 7f3b24
}