Blob Blame History Raw
/*
* Copyright (C) 2014 - 2018 Intel Corporation.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright notice(s),
*    this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice(s),
*    this list of conditions and the following disclaimer in the documentation
*    and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``AS IS'' AND ANY EXPRESS
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO
* EVENT SHALL THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

#pragma once

#include <memkind.h>

#include <malloc.h>
#include <cassert>
#include "jemalloc/jemalloc.h"

// Malloc, jemalloc, memkind jemalloc and memkind memory operations definitions
namespace performance_tests
{
    using std::vector;
    using std::string;
    using std::thread;

#ifdef __DEBUG
#include <mutex>
    // Write entire text at once, avoiding switching to another thread
    extern int g_msgLevel;
    extern std::mutex g_coutMutex;
#define EMIT(LEVEL, TEXT) \
    if (g_msgLevel >= LEVEL) \
    { \
        g_coutMutex.lock(); std::cout << TEXT << std::endl; g_coutMutex.unlock(); \
    }
#else
#define EMIT(LEVEL, TEXT)
#endif

    // Use jemalloc, compiled with unique prefix (--with-jemalloc-prefix= configure option)
#ifdef SYSTEM_JEMALLOC_PREFIX
#define TOKENPASTE(x, y) x ## y
#define JE(x, y) TOKENPASTE(x, y)
#define jexx_malloc JE(SYSTEM_JEMALLOC_PREFIX, malloc)
#define jexx_calloc JE(SYSTEM_JEMALLOC_PREFIX, calloc)
#define jexx_memalign JE(SYSTEM_JEMALLOC_PREFIX, memalign)
#define jexx_realloc JE(SYSTEM_JEMALLOC_PREFIX, realloc)
#define jexx_free JE(SYSTEM_JEMALLOC_PREFIX, free)
    extern "C" {
        // jemalloc function prototypes
        // full header cannot be include due to conflict with memkind jemalloc
        extern void *jexx_malloc(size_t size);
        extern void *jexx_calloc(size_t num, size_t size);
        extern void *jexx_memalign(size_t alignment, size_t size);
        extern void *jexx_realloc(void *ptr, size_t size);
        extern void  jexx_free(void *ptr);
    }
#endif

    // Available memory operations
    enum OperationName {
        Malloc,
        Calloc,
        Realloc,
        Align,
        Free,
        Invalid
    };

    // Reprents a memory operation
    class Operation
    {
    public:
        // Each operation is assigned a bucket size from range (0, MaxBucketSize)
        static const unsigned MaxBucketSize = 100;
        // For memalign operation, alignment parameter will be a random value
        // from range (sizeof(void*), sizeof(void*) * MemalignMaxMultiplier)
        static const unsigned MemalignMaxMultiplier = 4;

    protected:
        OperationName m_name;
        // If random number from range (0, MaxBucketSize) is lower than m_bucketSize, an operation will be performed
        unsigned m_bucketSize;

    public:
        Operation()
        {
        }

        // Creates an operation with given name and bucket size
        // If no bucket size is given, operation will be always performed
        Operation(
            OperationName name,
            unsigned bucketSize = MaxBucketSize)
            : m_name(name)
            , m_bucketSize(bucketSize)
        {
            assert(bucketSize <= MaxBucketSize);
        }

        virtual ~Operation() {}
        ;

        // Check if operation should be performed (currently drawn random number lower than bucket size)
        bool checkCondition(unsigned ballSize) const
        {
            return (ballSize < m_bucketSize);
        }

        OperationName getName() const
        {
            return m_name;
        }

        string getNameStr() const
        {
            switch (m_name) {
                case OperationName::Malloc:
                    return "malloc";

                case OperationName::Calloc:
                    return "calloc";

                case OperationName::Realloc:
                    return "realloc";

                case OperationName::Align:
                    return "align";

                case OperationName::Free:
                    return "free";

                default:
                    return "<unknown>";
            }
        }

        // Get operation bucket size
        unsigned getBucketSize() const
        {
            return m_bucketSize;
        }

        // perform memory operation
        virtual void perform(const memkind_t &kind,
                             void *&mem,
                             size_t size = 0,
                             size_t offset=0,
                             size_t alignment=0)
        const = 0;

    };

    // Malloc memory operations
    class MallocOperation : public Operation
    {
    public:
        MallocOperation(
            OperationName name)
            : Operation(name)
        {
        }

        MallocOperation(
            OperationName name,
            unsigned bucketSize)
            : Operation(name, bucketSize)
        {
        }

        virtual void perform(const memkind_t &kind,
                             void *&mem,
                             size_t size,
                             size_t offset,
                             size_t alignment) const override
        {
            EMIT(2, "Entering Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
            switch(m_name) {
                case Malloc: {
                    if (mem != nullptr) {
                        free(mem);
                    }
                    mem = malloc(size);
                    break;
                }
                case Calloc: {
                    if (mem != nullptr) {
                        free(mem);
                    }
                    // split allocation size randomly
                    // between number of elements and element size
                    mem = calloc((1 << offset), (size >> offset));
                    break;
                }
                case Realloc: {
                    mem = realloc(mem, size);
                    break;
                }
                case Align: {
                    if (mem != nullptr) {
                        free(mem);
                    }
                    // randomly choose alignment from (8, 8 * MemalignMaxMultiplie)
                    mem = memalign(alignment, size);
                    break;
                }
                case Free: {
                    free(mem);
                    mem = nullptr;
                    break;
                }

                default:
                    throw "Not implemented";
                    break;
            }
            EMIT(2, "Exiting Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
        }
    };

#ifdef SYSTEM_JEMALLOC_PREFIX
    // Jemalloc memory operations
    class JemallocOperation : public Operation
    {
    public:
        JemallocOperation(
            OperationName name)
            : Operation(name)
        {
        }

        JemallocOperation(
            OperationName name,
            unsigned bucketSize)
            : Operation(name, bucketSize)
        {
        }

        virtual void perform(const memkind_t &kind,
                             void *&mem,
                             size_t size,
                             size_t offset,
                             size_t alignment) const override
        {
            EMIT(2, "Entering Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
            switch(m_name) {
                case Malloc: {
                    if (mem != nullptr) {
                        jexx_free(mem);
                    }
                    mem = jexx_malloc(size);
                    break;
                }
                case Calloc: {
                    if (mem != nullptr) {
                        jexx_free(mem);
                    }
                    // split allocation size randomly
                    // between number of elements and element size
                    mem = jexx_calloc((1 <<, offset), (size >>, offset));
                    break;
                }
                case Realloc: {
                    mem = jexx_realloc(mem, size);
                    break;
                }
                case Align: {
                    if (mem != nullptr) {
                        jexx_free(mem);
                    }
                    // randomly choose alignment from (8, 8 * MemalignMaxMultiplie)
                    mem = jexx_memalign(alignment, size);
                    break;
                }
                case Free: {
                    jexx_free(mem);
                    mem = nullptr;
                    break;
                }

                default:
                    throw "Not implemented";
                    break;
            }
            EMIT(2, "Exiting Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
        }
    };
#endif
    // Jemkmalloc memory operations
    class JemkmallocOperation : public Operation
    {
    public:
        JemkmallocOperation(
            OperationName name)
            : Operation(name)
        {
        }

        JemkmallocOperation(
            OperationName name,
            unsigned bucketSize)
            : Operation(name, bucketSize)
        {
        }

        virtual void perform(const memkind_t &kind,
                             void *&mem,
                             size_t size,
                             size_t offset,
                             size_t alignment) const override
        {
#ifdef JEMK
            EMIT(2, "Entering Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
            switch(m_name) {
                case Malloc: {
                    if (mem != nullptr) {
                        jemk_free(mem);
                    }
                    mem = jemk_malloc(size);
                    break;
                }
                case Calloc: {
                    if (mem != nullptr) {
                        jemk_free(mem);
                    }
                    // split allocation size randomly
                    // between number of elements and element size
                    mem = jemk_calloc((1 << offset), (size >> offset));
                    break;
                }
                case Realloc: {
                    mem = jemk_realloc(mem, size);
                    break;
                }
                case Align: {
                    if (mem != nullptr) {
                        jemk_free(mem);
                    }
                    // randomly choose alignment from (8, 8 * MemalignMaxMultiplie)
                    mem = jemk_memalign(alignment, size);
                    break;
                }
                case Free: {
                    jemk_free(mem);
                    mem = nullptr;
                    break;
                }

                default:
                    throw "Not implemented";
                    break;
            }
            EMIT(2, "Exiting Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
#endif // JE_MK
        }
    };

    // Memkind memory operations
    class MemkindOperation : public Operation
    {
    public:
        MemkindOperation()
        {}

        MemkindOperation(
            OperationName name)
            : Operation(name)
        {
        }

        MemkindOperation(
            OperationName name,
            size_t bucketSize)
            : Operation(name, bucketSize)
        {
        }

        virtual void perform(const memkind_t &kind,
                             void *&mem,
                             size_t size,
                             size_t offset,
                             size_t alignment) const override
        {
            EMIT(2, "Entering Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
            switch (m_name) {
                case Malloc: {
                    if (mem != nullptr) {
                        memkind_free(kind, mem);
                    }
                    mem = memkind_malloc(kind, size);
                    break;
                }
                case Calloc: {
                    if (mem != nullptr) {
                        memkind_free(kind, mem);
                    }
                    // split allocation size randomly
                    // between number of elements and element size
                    mem = memkind_calloc(kind, 1 << offset, size >> offset);
                    break;
                }
                case Realloc: {
                    mem = memkind_realloc(kind, mem, size);
                    break;
                }

                case Align: {
                    if (mem != nullptr) {
                        memkind_free(kind, mem);
                        mem = nullptr;
                    }
                    // randomly choose alignment from (sizeof(void*), sizeof(void*) * MemalignMaxMultiplie)
                    if (memkind_posix_memalign(kind, &mem, alignment, size) != 0) {
                        // failure
                        mem = nullptr;
                    }
                    break;
                }
                case Free: {
                    memkind_free(kind, mem);
                    mem = nullptr;
                    break;
                }

                default:
                    break;
            }
            EMIT(2, "Exiting Operation::" << getNameStr()
                 <<  ", size=" << size
                 << ", offset=" << offset
                 << ", alignment=" << alignment
                 << ", mem=" << mem)
        }
    };
}