mirror of
https://github.com/yse/easy_profiler.git
synced 2025-01-15 09:38:02 +08:00
275 lines
7.8 KiB
C
275 lines
7.8 KiB
C
|
/************************************************
|
||
|
* creation time : 2017/08/20
|
||
|
* author : Blake Martin
|
||
|
* email : rationalcoder@gmail.com
|
||
|
************************************************/
|
||
|
|
||
|
/**
|
||
|
Lightweight profiler library for c++
|
||
|
Copyright(C) 2016-2018 Sergey Yagovtsev, Victor Zarubkin
|
||
|
|
||
|
Licensed under either of
|
||
|
* MIT license (LICENSE.MIT or http://opensource.org/licenses/MIT)
|
||
|
* Apache License, Version 2.0, (LICENSE.APACHE or http://www.apache.org/licenses/LICENSE-2.0)
|
||
|
at your option.
|
||
|
|
||
|
The MIT License
|
||
|
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.
|
||
|
|
||
|
|
||
|
The Apache License, Version 2.0 (the "License");
|
||
|
You may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
|
||
|
**/
|
||
|
|
||
|
#ifndef EASY_PROFILER_ALIGNMENT_HELPERS_H
|
||
|
#define EASY_PROFILER_ALIGNMENT_HELPERS_H
|
||
|
|
||
|
#include <cstddef>
|
||
|
#include <cstdint>
|
||
|
#include <easy/details/easy_compiler_support.h>
|
||
|
|
||
|
//! Checks if a pointer is aligned.
|
||
|
//! \param ptr The pointer to check.
|
||
|
//! \param alignment The alignement (must be a power of 2)
|
||
|
//! \returns true if the memory is aligned.
|
||
|
//!
|
||
|
template <uint32_t ALIGNMENT>
|
||
|
EASY_FORCE_INLINE bool is_aligned(void* ptr)
|
||
|
{
|
||
|
static_assert((ALIGNMENT & 1) == 0, "Alignment must be a power of two.");
|
||
|
return ((uintptr_t)ptr & (ALIGNMENT-1)) == 0;
|
||
|
}
|
||
|
|
||
|
EASY_FORCE_INLINE void unaligned_zero16(void* ptr)
|
||
|
{
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*(uint16_t*)ptr = 0;
|
||
|
#else
|
||
|
((char*)ptr)[0] = 0;
|
||
|
((char*)ptr)[1] = 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
EASY_FORCE_INLINE void unaligned_zero32(void* ptr)
|
||
|
{
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*(uint32_t*)ptr = 0;
|
||
|
#else
|
||
|
((char*)ptr)[0] = 0;
|
||
|
((char*)ptr)[1] = 0;
|
||
|
((char*)ptr)[2] = 0;
|
||
|
((char*)ptr)[3] = 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
EASY_FORCE_INLINE void unaligned_zero64(void* ptr)
|
||
|
{
|
||
|
#ifdef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
// Assume unaligned is more common.
|
||
|
if (!is_aligned<alignof(uint64_t)>(ptr)) {
|
||
|
((char*)ptr)[0] = 0;
|
||
|
((char*)ptr)[1] = 0;
|
||
|
((char*)ptr)[2] = 0;
|
||
|
((char*)ptr)[3] = 0;
|
||
|
((char*)ptr)[4] = 0;
|
||
|
((char*)ptr)[5] = 0;
|
||
|
((char*)ptr)[6] = 0;
|
||
|
((char*)ptr)[7] = 0;
|
||
|
}
|
||
|
else
|
||
|
#endif
|
||
|
|
||
|
*(uint64_t*)ptr = 0;
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE void unaligned_store16(void* ptr, T val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 2, "16 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*(T*)ptr = val;
|
||
|
#else
|
||
|
const char* const temp = (const char*)&val;
|
||
|
((char*)ptr)[0] = temp[0];
|
||
|
((char*)ptr)[1] = temp[1];
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE void unaligned_store32(void* ptr, T val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 4, "32 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*(T*)ptr = val;
|
||
|
#else
|
||
|
const char* const temp = (const char*)&val;
|
||
|
((char*)ptr)[0] = temp[0];
|
||
|
((char*)ptr)[1] = temp[1];
|
||
|
((char*)ptr)[2] = temp[2];
|
||
|
((char*)ptr)[3] = temp[3];
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE void unaligned_store64(void* ptr, T val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 8, "64 bit type required.");
|
||
|
|
||
|
#ifdef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
// Assume unaligned is more common.
|
||
|
if (!is_aligned<alignof(T)>(ptr)) {
|
||
|
const char* const temp = (const char*)&val;
|
||
|
((char*)ptr)[0] = temp[0];
|
||
|
((char*)ptr)[1] = temp[1];
|
||
|
((char*)ptr)[2] = temp[2];
|
||
|
((char*)ptr)[3] = temp[3];
|
||
|
((char*)ptr)[4] = temp[4];
|
||
|
((char*)ptr)[5] = temp[5];
|
||
|
((char*)ptr)[6] = temp[6];
|
||
|
((char*)ptr)[7] = temp[7];
|
||
|
}
|
||
|
else
|
||
|
#endif
|
||
|
|
||
|
*(T*)ptr = val;
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load16(const void* ptr)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 2, "16 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
return *(const T*)ptr;
|
||
|
#else
|
||
|
T value;
|
||
|
((char*)&value)[0] = ((const char*)ptr)[0];
|
||
|
((char*)&value)[1] = ((const char*)ptr)[1];
|
||
|
return value;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load16(const void* ptr, T* val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 2, "16 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*val = *(const T*)ptr;
|
||
|
#else
|
||
|
((char*)val)[0] = ((const char*)ptr)[0];
|
||
|
((char*)val)[1] = ((const char*)ptr)[1];
|
||
|
#endif
|
||
|
|
||
|
return *val;
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load32(const void* ptr)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 4, "32 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
return *(const T*)ptr;
|
||
|
#else
|
||
|
T value;
|
||
|
((char*)&value)[0] = ((const char*)ptr)[0];
|
||
|
((char*)&value)[1] = ((const char*)ptr)[1];
|
||
|
((char*)&value)[2] = ((const char*)ptr)[2];
|
||
|
((char*)&value)[3] = ((const char*)ptr)[3];
|
||
|
return value;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load32(const void* ptr, T* val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 4, "32 bit type required.");
|
||
|
|
||
|
#ifndef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
*val = *(const T*)ptr;
|
||
|
#else
|
||
|
((char*)&val)[0] = ((const char*)ptr)[0];
|
||
|
((char*)&val)[1] = ((const char*)ptr)[1];
|
||
|
((char*)&val)[2] = ((const char*)ptr)[2];
|
||
|
((char*)&val)[3] = ((const char*)ptr)[3];
|
||
|
#endif
|
||
|
|
||
|
return *val;
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load64(const void* ptr)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 8, "64 bit type required.");
|
||
|
|
||
|
#ifdef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
if (!is_aligned<alignof(T)>(ptr)) {
|
||
|
T value;
|
||
|
((char*)&value)[0] = ((const char*)ptr)[0];
|
||
|
((char*)&value)[1] = ((const char*)ptr)[1];
|
||
|
((char*)&value)[2] = ((const char*)ptr)[2];
|
||
|
((char*)&value)[3] = ((const char*)ptr)[3];
|
||
|
((char*)&value)[4] = ((const char*)ptr)[4];
|
||
|
((char*)&value)[5] = ((const char*)ptr)[5];
|
||
|
((char*)&value)[6] = ((const char*)ptr)[6];
|
||
|
((char*)&value)[7] = ((const char*)ptr)[7];
|
||
|
return value;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
return *(const T*)ptr;
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
EASY_FORCE_INLINE T unaligned_load64(const void* ptr, T* val)
|
||
|
{
|
||
|
static_assert(sizeof(T) == 8, "64 bit type required.");
|
||
|
|
||
|
#ifdef EASY_ENABLE_STRICT_ALIGNMENT
|
||
|
if (!is_aligned<alignof(T)>(ptr)) {
|
||
|
((char*)&val)[0] = ((const char*)ptr)[0];
|
||
|
((char*)&val)[1] = ((const char*)ptr)[1];
|
||
|
((char*)&val)[2] = ((const char*)ptr)[2];
|
||
|
((char*)&val)[3] = ((const char*)ptr)[3];
|
||
|
((char*)&val)[4] = ((const char*)ptr)[4];
|
||
|
((char*)&val)[5] = ((const char*)ptr)[5];
|
||
|
((char*)&val)[6] = ((const char*)ptr)[6];
|
||
|
((char*)&val)[7] = ((const char*)ptr)[7];
|
||
|
}
|
||
|
else
|
||
|
#endif
|
||
|
|
||
|
*val = *(const T*)ptr;
|
||
|
return *val;
|
||
|
}
|
||
|
|
||
|
#endif //EASY_PROFILER_ALIGNMENT_HELPERS_H
|