/******************************************************************************* The content of this file includes portions of the AUDIOKINETIC Wwise Technology released in source code form as part of the SDK installer package. Commercial License Usage Licensees holding valid commercial licenses to the AUDIOKINETIC Wwise Technology may use this file in accordance with the end user license agreement provided with the software or, alternatively, in accordance with the terms contained in a written agreement between you and Audiokinetic Inc. Apache License Usage Alternatively, this file may be used under the Apache License, Version 2.0 (the "Apache License"); you may not use this file except in compliance with the Apache License. You may obtain a copy of the Apache License at http://www.apache.org/licenses/LICENSE-2.0. Unless required by applicable law or agreed to in writing, software distributed under the Apache License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the Apache License for the specific language governing permissions and limitations under the License. Copyright (c) 2023 Audiokinetic Inc. *******************************************************************************/ #pragma once #include #include #if (defined(AK_CPU_X86_64) || defined(AK_CPU_X86)) #include #endif #include #include #include #include #define AK_THREAD_INIT_CODE(_threadProperties) \ if (_threadProperties.dwAffinityMask != AK_THREAD_AFFINITY_DEFAULT) \ { \ cpu_set_t affinity; \ CPU_ZERO(&affinity); \ for (AkUInt32 i = 0; i < 32; ++i) { \ if (_threadProperties.dwAffinityMask & ( 1 << i )){\ CPU_SET(i, &affinity); \ } \ } \ sched_setaffinity(0, sizeof(cpu_set_t), &affinity); \ } namespace AKPLATFORM { // Time functions // ------------------------------------------------------------------ /// Platform Independent Helper inline void PerformanceCounter( AkInt64 * out_piLastTime ) { struct timespec clockNow; clock_gettime(CLOCK_MONOTONIC, &clockNow); *out_piLastTime = ((clockNow.tv_sec + clockNow.tv_nsec/ 1000000000.0) * CLOCKS_PER_SEC); } /// Platform Independent Helper inline void PerformanceFrequency( AkInt64 * out_piFreq ) { // TO DO ANDROID ... is there something better *out_piFreq = CLOCKS_PER_SEC; } template inline size_t AkSimpleConvertString( destType* in_pdDest, const srcType* in_pSrc, size_t in_MaxSize, size_t destStrLen(const destType *), size_t srcStrLen(const srcType *) ) { size_t i; size_t lenToCopy = srcStrLen(in_pSrc); lenToCopy = (lenToCopy > in_MaxSize-1) ? in_MaxSize-1 : lenToCopy; for(i = 0; i < lenToCopy; i++) { in_pdDest[i] = (destType) in_pSrc[i]; } in_pdDest[lenToCopy] = (destType)0; return lenToCopy; } #define CONVERT_UTF16_TO_CHAR( _astring_, _charstring_ ) \ _charstring_ = (char*)AkAlloca( (1 + AKPLATFORM::AkUtf16StrLen((const AkUtf16*)_astring_)) * sizeof(char) ); \ AK_UTF16_TO_CHAR( _charstring_, (const AkUtf16*)_astring_, AKPLATFORM::AkUtf16StrLen((const AkUtf16*)_astring_)+1 ) #define AK_UTF16_TO_CHAR( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, strlen, AKPLATFORM::AkUtf16StrLen ) #define AK_UTF8_TO_OSCHAR( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, strlen, strlen ) #define AK_UTF16_TO_OSCHAR( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, strlen, AKPLATFORM::AkUtf16StrLen ) #define AK_UTF16_TO_WCHAR( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, wcslen, AKPLATFORM::AkUtf16StrLen ) #define AK_CHAR_TO_UTF16( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, AKPLATFORM::AkUtf16StrLen, strlen ) #define AK_OSCHAR_TO_UTF16( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, AKPLATFORM::AkUtf16StrLen, strlen ) #define AK_WCHAR_TO_UTF16( in_pdDest, in_pSrc, in_MaxSize ) AKPLATFORM::AkSimpleConvertString( in_pdDest, in_pSrc, in_MaxSize, AKPLATFORM::AkUtf16StrLen, wcslen ) #define AK_FILEHANDLE_TO_UINTPTR(_h) ((AkUIntPtr)_h) #define AK_SET_FILEHANDLE_TO_UINTPTR(_h,_u) _h = (AkFileHandle)_u /// Stack allocations. #define AkAlloca( _size_ ) __builtin_alloca( _size_ ) #if (defined(AK_CPU_X86_64) || defined(AK_CPU_X86)) // Once our minimum compiler version supports __get_cpuid_count, these asm blocks can be replaced #if defined(__i386__) && defined(__PIC__) // %ebx may be the PIC register. #define __ak_cpuid_count(level, count, a, b, c, d) \ __asm__ ("xchg{l}\t{%%}ebx, %k1\n\t" \ "cpuid\n\t" \ "xchg{l}\t{%%}ebx, %k1\n\t" \ : "=a" (a), "=&r" (b), "=c" (c), "=d" (d) \ : "0" (level), "2" (count)) #elif defined(__x86_64__) && defined(__PIC__) // %rbx may be the PIC register. #define __ak_cpuid_count(level, count, a, b, c, d) \ __asm__ ("xchg{q}\t{%%}rbx, %q1\n\t" \ "cpuid\n\t" \ "xchg{q}\t{%%}rbx, %q1\n\t" \ : "=a" (a), "=&r" (b), "=c" (c), "=d" (d) \ : "0" (level), "2" (count)) #else #define __ak_cpuid_count(level, count, a, b, c, d) \ __asm__ ("cpuid\n\t" \ : "=a" (a), "=b" (b), "=c" (c), "=d" (d) \ : "0" (level), "2" (count)) #endif static __inline int __ak_get_cpuid_count(unsigned int __leaf, unsigned int __subleaf, unsigned int *__eax, unsigned int *__ebx, unsigned int *__ecx, unsigned int *__edx) { unsigned int __max_leaf = __get_cpuid_max(__leaf & 0x80000000, 0); if (__max_leaf == 0 || __max_leaf < __leaf) return 0; __ak_cpuid_count(__leaf, __subleaf, *__eax, *__ebx, *__ecx, *__edx); return 1; } /// Support to fetch the CPUID for the platform. Only valid for X86 targets /// \remark Note that IAkProcessorFeatures should be preferred to fetch this data /// as it will have already translated the feature bits into AK-relevant enums inline void CPUID(AkUInt32 in_uLeafOpcode, AkUInt32 in_uSubLeafOpcode, unsigned int out_uCPUFeatures[4]) { __ak_get_cpuid_count( in_uLeafOpcode, in_uSubLeafOpcode, &out_uCPUFeatures[0], &out_uCPUFeatures[1], &out_uCPUFeatures[2], &out_uCPUFeatures[3]); } #endif }