// Copyright (c) 2012-2024 Wojciech Figat. All rights reserved. #pragma once #include "Types/BaseTypes.h" #include "Types/String.h" #include "Types/Span.h" #if _MSC_VER && PLATFORM_SIMD_SSE4_2 #include #endif namespace Utilities { struct Private { static FLAXENGINE_API Span BytesSizes; static FLAXENGINE_API Span HertzSizes; }; // Round floating point value up to 1 decimal place template FORCE_INLINE T RoundTo1DecimalPlace(T value) { return (T)round((double)value * 10) / (T)10; } // Round floating point value up to 2 decimal places template FORCE_INLINE T RoundTo2DecimalPlaces(T value) { return (T)round((double)value * 100.0) / (T)100; } // Round floating point value up to 3 decimal places template FORCE_INLINE T RoundTo3DecimalPlaces(T value) { return (T)round((double)value * 1000.0) / (T)1000; } /// /// Converts units to the best fitting human-readable denominator. /// /// Value type. /// Units count /// Amount of units required for the next size. /// Array with human-readable sizes to convert from. /// The best fitting string of the units. template String UnitsToText(T units, int32 divider, const Span sizes) { if (sizes.Length() == 0) return String::Format(TEXT("{0}"), units); int32 i = 0; double dblSUnits = static_cast(units); for (; static_cast(units / static_cast(divider)) > 0; i++, units /= divider) dblSUnits = (double)units / (double)divider; if (i >= sizes.Length()) i = 0; String text = String::Format(TEXT("{}"), RoundTo2DecimalPlaces(dblSUnits)); const int32 dot = text.FindLast('.'); if (dot != -1) text = text.Left(dot + 3); return String::Format(TEXT("{0} {1}"), text, sizes[i]); } /// /// Converts size of the data (in bytes) to the best fitting string. /// /// Value type. /// Size of the data in bytes. /// The best fitting string of the data size. template String BytesToText(T bytes) { return UnitsToText(bytes, 1024, Private::BytesSizes); } /// /// Converts hertz to the best fitting string. /// /// Value type. /// Value in hertz for conversion. /// The best fitting string. template String HertzToText(T hertz) { return UnitsToText(hertz, 1000, Private::HertzSizes); } // Returns the amount of set bits in 32-bit integer. inline int32 CountBits(uint32 x) { // [Reference: https://stackoverflow.com/questions/109023/how-to-count-the-number-of-set-bits-in-a-32-bit-integer] #ifdef __GNUC_ return __builtin_popcount(x); #elif _MSC_VER && PLATFORM_SIMD_SSE4_2 return __popcnt(x); #else x = x - ((x >> 1) & 0x55555555); x = (x & 0x33333333) + ((x >> 2) & 0x33333333); x = (x + (x >> 4)) & 0x0F0F0F0F; return (x * 0x01010101) >> 24; #endif } // Copy memory region but ignoring address sanatizer checks for memory regions. NO_SANITIZE_ADDRESS static void UnsafeMemoryCopy(void* dst, const void* src, uint64 size) { #if BUILD_RELEASE memcpy(dst, src, static_cast(size)); #else for (uint64 i = 0; i < size; i++) ((byte*)dst)[i] = ((byte*)src)[i]; #endif } }