bx.cpp 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. /*
  2. * Copyright 2010-2025 Branimir Karadzic. All rights reserved.
  3. * License: https://github.com/bkaradzic/bx/blob/master/LICENSE
  4. */
  5. #include <bx/debug.h>
  6. #include <bx/readerwriter.h>
  7. #include <bx/os.h>
  8. #if !BX_CRT_NONE
  9. # include <string.h> // memcpy, memmove, memset
  10. #endif // !BX_CRT_NONE
  11. namespace bx
  12. {
  13. Location Location::current(const char* _filePath, uint32_t _line)
  14. {
  15. return Location(_filePath, _line);
  16. }
  17. LocationFull LocationFull::current(const char* _function, const char* _filePath, uint32_t _line)
  18. {
  19. return LocationFull(_function, _filePath, _line);
  20. }
  21. static bool defaultAssertHandler(const Location& _location, uint32_t _skip, const char* _format, va_list _argList)
  22. {
  23. char temp[8192];
  24. int32_t total = 0;
  25. StaticMemoryBlockWriter smb(temp, BX_COUNTOF(temp) );
  26. ErrorIgnore err;
  27. total += write(&smb, &err, "\n--- ASSERT ---\n\n");
  28. total += write(&smb, &err, "%s(%d): "
  29. , _location.filePath
  30. , _location.line
  31. );
  32. total += write(&smb, _format, _argList, &err);
  33. total += write(&smb, "\n\n", &err);
  34. uintptr_t stack[32];
  35. const uint32_t num = getCallStack(2 /* skip self */ + _skip, BX_COUNTOF(stack), stack);
  36. total += writeCallstack(&smb, stack, num, &err);
  37. total += write(&smb, &err,
  38. "\nBuild info:\n"
  39. "\tCompiler: " BX_COMPILER_NAME
  40. ", CPU: " BX_CPU_NAME
  41. ", Arch: " BX_ARCH_NAME
  42. ", OS: " BX_PLATFORM_NAME
  43. ", CRT: " BX_CRT_NAME
  44. ", C++: " BX_CPP_NAME
  45. ", Date: " __DATE__
  46. ", Time: " __TIME__
  47. "\n"
  48. );
  49. total += write(&smb, &err, "\n--- END ---\n\n");
  50. write(getDebugOut(), temp, total, ErrorIgnore{});
  51. return true;
  52. }
  53. static AssertHandlerFn s_assertHandler = defaultAssertHandler;
  54. void setAssertHandler(AssertHandlerFn _assertHandlerFn)
  55. {
  56. BX_WARN(defaultAssertHandler == s_assertHandler, "Assert handler is already set.");
  57. if (defaultAssertHandler == s_assertHandler)
  58. {
  59. s_assertHandler = NULL == _assertHandlerFn
  60. ? defaultAssertHandler
  61. : _assertHandlerFn
  62. ;
  63. }
  64. }
  65. bool assertFunction(const Location& _location, uint32_t _skip, const char* _format, ...)
  66. {
  67. va_list argList;
  68. va_start(argList, _format);
  69. const bool result = s_assertHandler(_location, _skip, _format, argList);
  70. va_end(argList);
  71. return result;
  72. }
  73. void swap(void* _a, void* _b, size_t _numBytes)
  74. {
  75. uint8_t* lhs = (uint8_t*)_a;
  76. uint8_t* rhs = (uint8_t*)_b;
  77. const uint8_t* end = rhs + _numBytes;
  78. while (rhs != end)
  79. {
  80. swap(*lhs++, *rhs++);
  81. }
  82. }
  83. void memCopyRef(void* _dst, const void* _src, size_t _numBytes)
  84. {
  85. uint8_t* dst = (uint8_t*)_dst;
  86. const uint8_t* end = dst + _numBytes;
  87. const uint8_t* src = (const uint8_t*)_src;
  88. while (dst != end)
  89. {
  90. *dst++ = *src++;
  91. }
  92. }
  93. void memCopy(void* _dst, const void* _src, size_t _numBytes)
  94. {
  95. #if BX_CRT_NONE
  96. memCopyRef(_dst, _src, _numBytes);
  97. #else
  98. ::memcpy(_dst, _src, _numBytes);
  99. #endif // BX_CRT_NONE
  100. }
  101. void memCopy(
  102. void* _dst
  103. , uint32_t _dstStride
  104. , const void* _src
  105. , uint32_t _srcStride
  106. , uint32_t _stride
  107. , uint32_t _numStrides
  108. )
  109. {
  110. if (_stride == _srcStride
  111. && _stride == _dstStride)
  112. {
  113. memCopy(_dst, _src, _stride*_numStrides);
  114. return;
  115. }
  116. const uint8_t* src = (const uint8_t*)_src;
  117. uint8_t* dst = (uint8_t*)_dst;
  118. for (uint32_t ii = 0; ii < _numStrides; ++ii, src += _srcStride, dst += _dstStride)
  119. {
  120. memCopy(dst, src, _stride);
  121. }
  122. }
  123. void memMoveRef(void* _dst, const void* _src, size_t _numBytes)
  124. {
  125. uint8_t* dst = (uint8_t*)_dst;
  126. const uint8_t* src = (const uint8_t*)_src;
  127. if (_numBytes == 0
  128. || dst == src)
  129. {
  130. return;
  131. }
  132. // if (src+_numBytes <= dst || end <= src)
  133. if (dst < src)
  134. {
  135. memCopy(_dst, _src, _numBytes);
  136. return;
  137. }
  138. for (intptr_t ii = _numBytes-1; ii >= 0; --ii)
  139. {
  140. dst[ii] = src[ii];
  141. }
  142. }
  143. void memMove(void* _dst, const void* _src, size_t _numBytes)
  144. {
  145. #if BX_CRT_NONE
  146. memMoveRef(_dst, _src, _numBytes);
  147. #else
  148. ::memmove(_dst, _src, _numBytes);
  149. #endif // BX_CRT_NONE
  150. }
  151. void memMove(
  152. void* _dst
  153. , uint32_t _dstStride
  154. , const void* _src
  155. , uint32_t _srcStride
  156. , uint32_t _stride
  157. , uint32_t _numStrides
  158. )
  159. {
  160. if (_stride == _srcStride
  161. && _stride == _dstStride)
  162. {
  163. memMove(_dst, _src, _stride*_numStrides);
  164. return;
  165. }
  166. const uint8_t* src = (const uint8_t*)_src;
  167. uint8_t* dst = (uint8_t*)_dst;
  168. for (uint32_t ii = 0; ii < _numStrides; ++ii, src += _srcStride, dst += _dstStride)
  169. {
  170. memMove(dst, src, _stride);
  171. }
  172. }
  173. void memSetRef(void* _dst, uint8_t _ch, size_t _numBytes)
  174. {
  175. uint8_t* dst = (uint8_t*)_dst;
  176. const uint8_t* end = dst + _numBytes;
  177. while (dst != end)
  178. {
  179. *dst++ = char(_ch);
  180. }
  181. }
  182. void memSet(void* _dst, uint8_t _ch, size_t _numBytes)
  183. {
  184. #if BX_CRT_NONE
  185. memSetRef(_dst, _ch, _numBytes);
  186. #else
  187. ::memset(_dst, _ch, _numBytes);
  188. #endif // BX_CRT_NONE
  189. }
  190. void memSet(void* _dst, uint32_t _dstStride, uint8_t _ch, uint32_t _stride, uint32_t _num)
  191. {
  192. if (_stride == _dstStride)
  193. {
  194. memSet(_dst, _ch, _stride*_num);
  195. return;
  196. }
  197. uint8_t* dst = (uint8_t*)_dst;
  198. for (uint32_t ii = 0; ii < _num; ++ii, dst += _dstStride)
  199. {
  200. memSet(dst, _ch, _stride);
  201. }
  202. }
  203. int32_t memCmpRef(const void* _lhs, const void* _rhs, size_t _numBytes)
  204. {
  205. const char* lhs = (const char*)_lhs;
  206. const char* rhs = (const char*)_rhs;
  207. for (
  208. ; 0 < _numBytes && *lhs == *rhs
  209. ; ++lhs, ++rhs, --_numBytes
  210. )
  211. {
  212. }
  213. return 0 == _numBytes ? 0 : *lhs - *rhs;
  214. }
  215. int32_t memCmp(const void* _lhs, const void* _rhs, size_t _numBytes)
  216. {
  217. #if BX_CRT_NONE
  218. return memCmpRef(_lhs, _rhs, _numBytes);
  219. #else
  220. return ::memcmp(_lhs, _rhs, _numBytes);
  221. #endif // BX_CRT_NONE
  222. }
  223. ///
  224. void gather(void* _dst, const void* _src, uint32_t _srcStride, uint32_t _stride, uint32_t _numStrides)
  225. {
  226. memMove(_dst, _stride, _src, _srcStride, _stride, _numStrides);
  227. }
  228. ///
  229. void scatter(void* _dst, uint32_t _dstStride, const void* _src, uint32_t _stride, uint32_t _numStrides)
  230. {
  231. memMove(_dst, _dstStride, _src, _stride, _stride, _numStrides);
  232. }
  233. } // namespace bx