WinBase.cpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854
  1. /*
  2. * Copyright (c) Contributors to the Open 3D Engine Project.
  3. * For complete copyright and license terms please see the LICENSE at the root of this distribution.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0 OR MIT
  6. *
  7. */
  8. #include <AzCore/AzCore_Traits_Platform.h>
  9. // Description : Linux/Mac port support for Win32API calls
  10. #if AZ_TRAIT_LEGACY_CRYCOMMON_USE_WINDOWS_STUBS
  11. #include "platform.h" // Note: This should be first to get consistent debugging definitions
  12. #include <CryCommon/ISystem.h>
  13. #include <CryAssert.h>
  14. #if defined(AZ_RESTRICTED_PLATFORM)
  15. #undef AZ_RESTRICTED_SECTION
  16. #define WINBASE_CPP_SECTION_1 1
  17. #define WINBASE_CPP_SECTION_2 2
  18. #define WINBASE_CPP_SECTION_3 3
  19. #define WINBASE_CPP_SECTION_4 4
  20. #define WINBASE_CPP_SECTION_5 5
  21. #define WINBASE_CPP_SECTION_6 6
  22. #endif
  23. #if defined(AZ_RESTRICTED_PLATFORM)
  24. #define AZ_RESTRICTED_SECTION WINBASE_CPP_SECTION_1
  25. #include AZ_RESTRICTED_FILE(WinBase_cpp)
  26. #endif
  27. #if defined(AZ_RESTRICTED_SECTION_IMPLEMENTED)
  28. #undef AZ_RESTRICTED_SECTION_IMPLEMENTED
  29. #else
  30. #include <signal.h>
  31. #endif
  32. #include <pthread.h>
  33. #include <sys/types.h>
  34. #include <fcntl.h>
  35. #include <AzCore/IO/SystemFile.h>
  36. #include <AzCore/std/string/conversions.h>
  37. #ifdef APPLE
  38. #include <mach/mach.h>
  39. #include <mach/mach_time.h>
  40. #include <sys/sysctl.h> // for total physical memory on Mac
  41. #include <CoreFoundation/CoreFoundation.h> // for CryMessageBox
  42. #include <mach/vm_statistics.h> // host_statistics
  43. #include <mach/mach_types.h>
  44. #include <mach/mach_init.h>
  45. #include <mach/mach_host.h>
  46. #endif
  47. #include <sys/time.h>
  48. #if defined(LINUX) || defined(APPLE)
  49. #include <sys/types.h>
  50. #include <unistd.h>
  51. #include <utime.h>
  52. #include <dirent.h>
  53. #endif
  54. #if AZ_TRAIT_COMPILER_DEFINE_FS_ERRNO_TYPE
  55. typedef int FS_ERRNO_TYPE;
  56. #include <mutex>
  57. #endif
  58. bool IsBadReadPtr(void* ptr, unsigned int size)
  59. {
  60. //too complicated to really support it
  61. return ptr ? false : true;
  62. }
  63. //////////////////////////////////////////////////////////////////////////
  64. char* _strtime(char* date)
  65. {
  66. azstrcpy(date, AZ_ARRAY_SIZE(date), "0:0:0");
  67. return date;
  68. }
  69. //////////////////////////////////////////////////////////////////////////
  70. char* _strdate(char* date)
  71. {
  72. azstrcpy(date, AZ_ARRAY_SIZE(date), "0");
  73. return date;
  74. }
  75. //////////////////////////////////////////////////////////////////////////
  76. char* strlwr (char* str)
  77. {
  78. char* cp; /* traverses string for C locale conversion */
  79. for (cp = str; *cp; ++cp)
  80. {
  81. if ('A' <= *cp && *cp <= 'Z')
  82. {
  83. *cp += 'a' - 'A';
  84. }
  85. }
  86. return str;
  87. }
  88. char* strupr (char* str)
  89. {
  90. char* cp; /* traverses string for C locale conversion */
  91. for (cp = str; *cp; ++cp)
  92. {
  93. if ('a' <= *cp && *cp <= 'z')
  94. {
  95. *cp += 'A' - 'a';
  96. }
  97. }
  98. return str;
  99. }
  100. char* ltoa (long i, char* a, int radix)
  101. {
  102. if (a == nullptr)
  103. {
  104. return nullptr;
  105. }
  106. strcpy (a, "0");
  107. if (i && radix > 1 && radix < 37)
  108. {
  109. char buf[35];
  110. unsigned long u = i, p = 34;
  111. buf[p] = 0;
  112. if (i < 0 && radix == 10)
  113. {
  114. u = -i;
  115. }
  116. while (u)
  117. {
  118. unsigned int d = u % radix;
  119. buf[--p] = d < 10 ? '0' + d : 'a' + d - 10;
  120. u /= radix;
  121. }
  122. if (i < 0 && radix == 10)
  123. {
  124. buf[--p] = '-';
  125. }
  126. strcpy (a, buf + p);
  127. }
  128. return a;
  129. }
  130. #if AZ_TRAIT_COMPILER_DEFINE_WCSICMP
  131. // For Linux it's redefined to wcscasecmp and wcsncasecmp'
  132. int wcsicmp (const wchar_t* s1, const wchar_t* s2)
  133. {
  134. wint_t c1, c2;
  135. if (s1 == s2)
  136. {
  137. return 0;
  138. }
  139. do
  140. {
  141. c1 = towlower(*s1++);
  142. c2 = towlower(*s2++);
  143. }
  144. while (c1 && c1 == c2);
  145. return (int) (c1 - c2);
  146. }
  147. int wcsnicmp (const wchar_t* s1, const wchar_t* s2, size_t count)
  148. {
  149. wint_t c1, c2;
  150. if (s1 == s2 || count == 0)
  151. {
  152. return 0;
  153. }
  154. do
  155. {
  156. c1 = towlower(*s1++);
  157. c2 = towlower(*s2++);
  158. }
  159. while ((--count) && c1 && (c1 == c2));
  160. return (int) (c1 - c2);
  161. }
  162. #endif
  163. #if defined(ANDROID)
  164. // not defined in android-19 or prior
  165. size_t wcsnlen(const wchar_t* str, size_t maxLen)
  166. {
  167. size_t length;
  168. for (length = 0; length < maxLen; ++length, ++str)
  169. {
  170. if (!*str)
  171. {
  172. break;
  173. }
  174. }
  175. return length;
  176. }
  177. char* stpcpy(char* dest, const char* str)
  178. {
  179. while (*str != '\0')
  180. {
  181. *dest++ = *str++;
  182. }
  183. *dest = '\0';
  184. return dest;
  185. }
  186. #endif
  187. void _makepath(char* path, const char* drive, const char* dir, const char* filename, const char* ext)
  188. {
  189. char ch;
  190. char tmp[MAX_PATH];
  191. if (!path)
  192. {
  193. return;
  194. }
  195. tmp[0] = '\0';
  196. if (drive && drive[0])
  197. {
  198. tmp[0] = drive[0];
  199. tmp[1] = ':';
  200. tmp[2] = 0;
  201. }
  202. if (dir && dir[0])
  203. {
  204. azstrcat(tmp, MAX_PATH, dir);
  205. ch = tmp[strlen(tmp) - 1];
  206. if (ch != '/' && ch != '\\')
  207. {
  208. azstrcat(tmp, MAX_PATH, "\\");
  209. }
  210. }
  211. if (filename && filename[0])
  212. {
  213. azstrcat(tmp, MAX_PATH, filename);
  214. if (ext && ext[0])
  215. {
  216. if (ext[0] != '.')
  217. {
  218. azstrcat(tmp, MAX_PATH, ".");
  219. }
  220. azstrcat(tmp, MAX_PATH, ext);
  221. }
  222. }
  223. azstrcpy(path, strlen(tmp) + 1, tmp);
  224. }
  225. char* _ui64toa(unsigned long long value, char* str, int radix)
  226. {
  227. if (str == nullptr)
  228. {
  229. return nullptr;
  230. }
  231. char buffer[65];
  232. char* pos;
  233. int digit;
  234. pos = &buffer[64];
  235. *pos = '\0';
  236. do
  237. {
  238. digit = value % radix;
  239. value = value / radix;
  240. if (digit < 10)
  241. {
  242. *--pos = '0' + digit;
  243. }
  244. else
  245. {
  246. *--pos = 'a' + digit - 10;
  247. } /* if */
  248. } while (value != 0L);
  249. memcpy(str, pos, &buffer[64] - pos + 1);
  250. return str;
  251. }
  252. long long _atoi64(const char* str)
  253. {
  254. if (str == nullptr)
  255. {
  256. return -1;
  257. }
  258. unsigned long long RunningTotal = 0;
  259. char bMinus = 0;
  260. while (*str == ' ' || (*str >= '\011' && *str <= '\015'))
  261. {
  262. str++;
  263. } /* while */
  264. if (*str == '+')
  265. {
  266. str++;
  267. }
  268. else if (*str == '-')
  269. {
  270. bMinus = 1;
  271. str++;
  272. } /* if */
  273. while (*str >= '0' && *str <= '9')
  274. {
  275. RunningTotal = RunningTotal * 10 + *str - '0';
  276. str++;
  277. } /* while */
  278. return bMinus ? ((long long)-RunningTotal) : (long long)RunningTotal;
  279. }
  280. #if defined(AZ_RESTRICTED_PLATFORM)
  281. #define AZ_RESTRICTED_SECTION WINBASE_CPP_SECTION_2
  282. #include AZ_RESTRICTED_FILE(WinBase_cpp)
  283. #endif
  284. #if defined(AZ_RESTRICTED_SECTION_IMPLEMENTED)
  285. #undef AZ_RESTRICTED_SECTION_IMPLEMENTED
  286. #else
  287. bool QueryPerformanceCounter(LARGE_INTEGER* counter)
  288. {
  289. #if defined(LINUX)
  290. // replaced gettimeofday
  291. // http://fixunix.com/kernel/378888-gettimeofday-resolution-linux.html
  292. timespec tv;
  293. clock_gettime(CLOCK_MONOTONIC, &tv);
  294. counter->QuadPart = (uint64)tv.tv_sec * 1000000 + tv.tv_nsec / 1000;
  295. return true;
  296. #elif defined(APPLE)
  297. counter->QuadPart = mach_absolute_time();
  298. return true;
  299. #else
  300. return false;
  301. #endif
  302. }
  303. bool QueryPerformanceFrequency(LARGE_INTEGER* frequency)
  304. {
  305. #if defined(LINUX)
  306. // On Linux we'll use gettimeofday(). The API resolution is microseconds,
  307. // so we'll report that to the caller.
  308. frequency->u.LowPart = 1000000;
  309. frequency->u.HighPart = 0;
  310. return true;
  311. #elif defined(APPLE)
  312. static mach_timebase_info_data_t s_kTimeBaseInfoData;
  313. if (s_kTimeBaseInfoData.denom == 0)
  314. {
  315. mach_timebase_info(&s_kTimeBaseInfoData);
  316. }
  317. // mach_timebase_info_data_t expresses the tick period in nanoseconds
  318. frequency->QuadPart = 1e+9 * (uint64_t)s_kTimeBaseInfoData.denom / (uint64_t)s_kTimeBaseInfoData.numer;
  319. return true;
  320. #else
  321. return false;
  322. #endif
  323. }
  324. #endif
  325. void _splitpath(const char* inpath, char* drv, char* dir, char* fname, char* ext)
  326. {
  327. if (drv)
  328. {
  329. drv[0] = 0;
  330. }
  331. typedef AZStd::fixed_string<AZ_MAX_PATH_LEN> path_stack_string;
  332. const path_stack_string inPath(inpath);
  333. AZStd::string::size_type s = inPath.rfind('/', inPath.size());//position of last /
  334. path_stack_string fName;
  335. if (s == AZStd::string::npos)
  336. {
  337. if (dir)
  338. {
  339. dir[0] = 0;
  340. }
  341. fName = inpath; //assign complete string as rest
  342. }
  343. else
  344. {
  345. if (dir)
  346. {
  347. azstrcpy(dir, AZ_MAX_PATH_LEN, (inPath.substr((AZStd::string::size_type)0, (AZStd::string::size_type)(s + 1))).c_str()); //assign directory
  348. }
  349. fName = inPath.substr((AZStd::string::size_type)(s + 1)); //assign remaining string as rest
  350. }
  351. if (fName.size() == 0)
  352. {
  353. if (ext)
  354. {
  355. ext[0] = 0;
  356. }
  357. if (fname)
  358. {
  359. fname[0] = 0;
  360. }
  361. }
  362. else
  363. {
  364. //dir and drive are now set
  365. s = fName.find(".", (AZStd::string::size_type)0);//position of first .
  366. if (s == AZStd::string::npos)
  367. {
  368. if (ext)
  369. {
  370. ext[0] = 0;
  371. }
  372. if (fname)
  373. {
  374. azstrcpy(fname, fName.size() + 1, fName.c_str()); //assign filename
  375. }
  376. }
  377. else
  378. {
  379. if (ext)
  380. {
  381. azstrcpy(ext, AZ_MAX_PATH_LEN, (fName.substr(s)).c_str()); //assign extension including .
  382. }
  383. if (fname)
  384. {
  385. if (s == 0)
  386. {
  387. fname[0] = 0;
  388. }
  389. else
  390. {
  391. azstrcpy(fname, AZ_MAX_PATH_LEN, (fName.substr((AZStd::string::size_type)0, s)).c_str()); //assign filename
  392. }
  393. }
  394. }
  395. }
  396. }
  397. //-----------------------------------------other stuff-------------------------------------------------------------------
  398. void GlobalMemoryStatus(LPMEMORYSTATUS lpmem)
  399. {
  400. //not complete implementation
  401. #if defined(AZ_RESTRICTED_PLATFORM)
  402. #define AZ_RESTRICTED_SECTION WINBASE_CPP_SECTION_3
  403. #include AZ_RESTRICTED_FILE(WinBase_cpp)
  404. #endif
  405. #if defined(AZ_RESTRICTED_SECTION_IMPLEMENTED)
  406. #undef AZ_RESTRICTED_SECTION_IMPLEMENTED
  407. #elif defined(APPLE)
  408. // Retrieve dwTotalPhys
  409. int kMIB[] = {CTL_HW, HW_MEMSIZE};
  410. uint64_t dwTotalPhys;
  411. size_t ulength = sizeof(dwTotalPhys);
  412. if (sysctl(kMIB, 2, &dwTotalPhys, &ulength, NULL, 0) != 0)
  413. {
  414. gEnv->pLog->LogError("sysctl failed\n");
  415. }
  416. else
  417. {
  418. lpmem->dwTotalPhys = static_cast<SIZE_T>(dwTotalPhys);
  419. }
  420. // Get the page size
  421. mach_port_t kHost(mach_host_self());
  422. vm_size_t uPageSize;
  423. if (host_page_size(kHost, &uPageSize) != 0)
  424. {
  425. gEnv->pLog->LogError("host_page_size failed\n");
  426. }
  427. else
  428. {
  429. // Get memory statistics
  430. vm_statistics_data_t kVMStats;
  431. mach_msg_type_number_t uCount(sizeof(kVMStats) / sizeof(natural_t));
  432. if (host_statistics(kHost, HOST_VM_INFO, (host_info_t)&kVMStats, &uCount) != 0)
  433. {
  434. gEnv->pLog->LogError("host_statistics failed\n");
  435. }
  436. else
  437. {
  438. // Calculate dwAvailPhys
  439. lpmem->dwAvailPhys = uPageSize * kVMStats.free_count;
  440. }
  441. }
  442. #else
  443. FILE* f;
  444. lpmem->dwMemoryLoad = 0;
  445. lpmem->dwTotalPhys = 16 * 1024 * 1024;
  446. lpmem->dwAvailPhys = 16 * 1024 * 1024;
  447. lpmem->dwTotalPageFile = 16 * 1024 * 1024;
  448. lpmem->dwAvailPageFile = 16 * 1024 * 1024;
  449. azfopen(&f, "/proc/meminfo", "r");
  450. if (f)
  451. {
  452. char buffer[256];
  453. memset(buffer, '0', 256);
  454. int total, used, free, shared, buffers, cached;
  455. lpmem->dwLength = sizeof(MEMORYSTATUS);
  456. lpmem->dwTotalPhys = lpmem->dwAvailPhys = 0;
  457. lpmem->dwTotalPageFile = lpmem->dwAvailPageFile = 0;
  458. while (fgets(buffer, sizeof(buffer), f))
  459. {
  460. if (azsscanf(buffer, "Mem: %d %d %d %d %d %d", &total, &used, &free, &shared, &buffers, &cached))
  461. {
  462. lpmem->dwTotalPhys += total;
  463. lpmem->dwAvailPhys += free + buffers + cached;
  464. }
  465. if (azsscanf(buffer, "Swap: %d %d %d", &total, &used, &free))
  466. {
  467. lpmem->dwTotalPageFile += total;
  468. lpmem->dwAvailPageFile += free;
  469. }
  470. if (azsscanf(buffer, "MemTotal: %d", &total))
  471. {
  472. lpmem->dwTotalPhys = total * 1024;
  473. }
  474. if (azsscanf(buffer, "MemFree: %d", &free))
  475. {
  476. lpmem->dwAvailPhys = free * 1024;
  477. }
  478. if (azsscanf(buffer, "SwapTotal: %d", &total))
  479. {
  480. lpmem->dwTotalPageFile = total * 1024;
  481. }
  482. if (azsscanf(buffer, "SwapFree: %d", &free))
  483. {
  484. lpmem->dwAvailPageFile = free * 1024;
  485. }
  486. if (azsscanf(buffer, "Buffers: %d", &buffers))
  487. {
  488. lpmem->dwAvailPhys += buffers * 1024;
  489. }
  490. if (azsscanf(buffer, "Cached: %d", &cached))
  491. {
  492. lpmem->dwAvailPhys += cached * 1024;
  493. }
  494. }
  495. fclose(f);
  496. if (lpmem->dwTotalPhys)
  497. {
  498. DWORD TotalPhysical = lpmem->dwTotalPhys + lpmem->dwTotalPageFile;
  499. DWORD AvailPhysical = lpmem->dwAvailPhys + lpmem->dwAvailPageFile;
  500. lpmem->dwMemoryLoad = (TotalPhysical - AvailPhysical) / (TotalPhysical / 100);
  501. }
  502. }
  503. #endif
  504. }
  505. static const int YearLengths[2] = {DAYSPERNORMALYEAR, DAYSPERLEAPYEAR};
  506. static const int MonthLengths[2][MONSPERYEAR] =
  507. {
  508. { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
  509. { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
  510. };
  511. static int IsLeapYear(int Year)
  512. {
  513. return Year % 4 == 0 && (Year % 100 != 0 || Year % 400 == 0) ? 1 : 0;
  514. }
  515. static void NormalizeTimeFields(short* FieldToNormalize, short* CarryField, int Modulus)
  516. {
  517. *FieldToNormalize = (short) (*FieldToNormalize - Modulus);
  518. *CarryField = (short) (*CarryField + 1);
  519. }
  520. static bool TimeFieldsToTime(PTIME_FIELDS tfTimeFields, PLARGE_INTEGER Time)
  521. {
  522. #define SECSPERMIN 60
  523. #define MINSPERHOUR 60
  524. #define HOURSPERDAY 24
  525. #define MONSPERYEAR 12
  526. #define EPOCHYEAR 1601
  527. #define SECSPERDAY 86400
  528. #define TICKSPERMSEC 10000
  529. #define TICKSPERSEC 10000000
  530. #define SECSPERHOUR 3600
  531. int CurYear, CurMonth;
  532. LONGLONG rcTime;
  533. TIME_FIELDS TimeFields = *tfTimeFields;
  534. rcTime = 0;
  535. while (TimeFields.Second >= SECSPERMIN)
  536. {
  537. NormalizeTimeFields(&TimeFields.Second, &TimeFields.Minute, SECSPERMIN);
  538. }
  539. while (TimeFields.Minute >= MINSPERHOUR)
  540. {
  541. NormalizeTimeFields(&TimeFields.Minute, &TimeFields.Hour, MINSPERHOUR);
  542. }
  543. while (TimeFields.Hour >= HOURSPERDAY)
  544. {
  545. NormalizeTimeFields(&TimeFields.Hour, &TimeFields.Day, HOURSPERDAY);
  546. }
  547. while (TimeFields.Day > MonthLengths[IsLeapYear(TimeFields.Year)][TimeFields.Month - 1])
  548. {
  549. NormalizeTimeFields(&TimeFields.Day, &TimeFields.Month, SECSPERMIN);
  550. }
  551. while (TimeFields.Month > MONSPERYEAR)
  552. {
  553. NormalizeTimeFields(&TimeFields.Month, &TimeFields.Year, MONSPERYEAR);
  554. }
  555. for (CurYear = EPOCHYEAR; CurYear < TimeFields.Year; CurYear++)
  556. {
  557. rcTime += YearLengths[IsLeapYear(CurYear)];
  558. }
  559. for (CurMonth = 1; CurMonth < TimeFields.Month; CurMonth++)
  560. {
  561. rcTime += MonthLengths[IsLeapYear(CurYear)][CurMonth - 1];
  562. }
  563. rcTime += TimeFields.Day - 1;
  564. rcTime *= SECSPERDAY;
  565. rcTime += TimeFields.Hour * SECSPERHOUR + TimeFields.Minute * SECSPERMIN + TimeFields.Second;
  566. rcTime *= TICKSPERSEC;
  567. rcTime += TimeFields.Milliseconds * TICKSPERMSEC;
  568. Time->QuadPart = rcTime;
  569. return true;
  570. }
  571. BOOL SystemTimeToFileTime(const SYSTEMTIME* syst, LPFILETIME ft)
  572. {
  573. TIME_FIELDS tf;
  574. LARGE_INTEGER t;
  575. tf.Year = syst->wYear;
  576. tf.Month = syst->wMonth;
  577. tf.Day = syst->wDay;
  578. tf.Hour = syst->wHour;
  579. tf.Minute = syst->wMinute;
  580. tf.Second = syst->wSecond;
  581. tf.Milliseconds = syst->wMilliseconds;
  582. TimeFieldsToTime(&tf, &t);
  583. ft->dwLowDateTime = t.u.LowPart;
  584. ft->dwHighDateTime = t.u.HighPart;
  585. return TRUE;
  586. }
  587. //////////////////////////////////////////////////////////////////////////
  588. #if defined(AZ_RESTRICTED_PLATFORM)
  589. #define AZ_RESTRICTED_SECTION WINBASE_CPP_SECTION_4
  590. #include AZ_RESTRICTED_FILE(WinBase_cpp)
  591. #endif
  592. #if defined(AZ_RESTRICTED_SECTION_IMPLEMENTED)
  593. #undef AZ_RESTRICTED_SECTION_IMPLEMENTED
  594. #else
  595. threadID GetCurrentThreadId()
  596. {
  597. return threadID(pthread_self());
  598. }
  599. #endif
  600. #include <chrono>
  601. #include <thread>
  602. //////////////////////////////////////////////////////////////////////////
  603. DWORD Sleep(DWORD dwMilliseconds)
  604. {
  605. #if defined(LINUX) || defined(APPLE)
  606. std::this_thread::sleep_for(std::chrono::milliseconds(dwMilliseconds));
  607. return 0;
  608. #define AZ_RESTRICTED_SECTION_IMPLEMENTED
  609. #elif defined(AZ_RESTRICTED_PLATFORM)
  610. #define AZ_RESTRICTED_SECTION WINBASE_CPP_SECTION_5
  611. #include AZ_RESTRICTED_FILE(WinBase_cpp)
  612. #endif
  613. #if defined(AZ_RESTRICTED_SECTION_IMPLEMENTED)
  614. #undef AZ_RESTRICTED_SECTION_IMPLEMENTED
  615. #else
  616. timeval tv, start, now;
  617. uint64 tStart;
  618. memset(&tv, 0, sizeof tv);
  619. memset(&start, 0, sizeof start);
  620. memset(&now, 0, sizeof now);
  621. gettimeofday(&now, NULL);
  622. start = now;
  623. tStart = (uint64)start.tv_sec * 1000000 + start.tv_usec;
  624. while (true)
  625. {
  626. uint64 tNow, timePassed, timeRemaining;
  627. tNow = (uint64)now.tv_sec * 1000000 + now.tv_usec;
  628. timePassed = tNow - tStart;
  629. if (timePassed >= dwMilliseconds)
  630. {
  631. break;
  632. }
  633. timeRemaining = dwMilliseconds * 1000 - timePassed;
  634. tv.tv_sec = timeRemaining / 1000000;
  635. tv.tv_usec = timeRemaining % 1000000;
  636. select(1, NULL, NULL, NULL, &tv);
  637. gettimeofday(&now, NULL);
  638. }
  639. return 0;
  640. #endif
  641. }
  642. #if defined(LINUX) || defined(APPLE)
  643. BOOL GetComputerName(LPSTR lpBuffer, LPDWORD lpnSize)
  644. {
  645. if (!lpBuffer || !lpnSize)
  646. {
  647. return FALSE;
  648. }
  649. int err = gethostname(lpBuffer, *lpnSize);
  650. if (-1 == err)
  651. {
  652. CryLog("GetComputerName falied [%d]\n", errno);
  653. return FALSE;
  654. }
  655. return TRUE;
  656. }
  657. #endif
  658. #if AZ_TRAIT_COMPILER_DEFINE_GETCURRENTPROCESSID
  659. DWORD GetCurrentProcessId(void)
  660. {
  661. return (DWORD)getpid();
  662. }
  663. #endif
  664. //////////////////////////////////////////////////////////////////////////
  665. //////////////////////////////////////////////////////////////////////////
  666. //////////////////////////////////////////////////////////////////////////
  667. void CrySleep(unsigned int dwMilliseconds)
  668. {
  669. Sleep(dwMilliseconds);
  670. }
  671. //////////////////////////////////////////////////////////////////////////
  672. //////////////////////////////////////////////////////////////////////////
  673. void CryMessageBox(const char* lpText, const char* lpCaption, [[maybe_unused]] unsigned int uType)
  674. {
  675. #ifdef WIN32
  676. # error WIN32 is defined in WinBase.cpp (it is a non-Windows file)
  677. #elif defined(MAC)
  678. CFStringRef strText = CFStringCreateWithCString(NULL, lpText, kCFStringEncodingMacRoman);
  679. CFStringRef strCaption = CFStringCreateWithCString(NULL, lpCaption, kCFStringEncodingMacRoman);
  680. CFStringRef strOk = CFSTR("OK");
  681. CFStringRef strCancel = CFSTR("Cancel");
  682. CFStringRef strRetry = CFSTR("Retry");
  683. CFStringRef strYes = CFSTR("Yes");
  684. CFStringRef strNo = CFSTR("No");
  685. CFStringRef strAbort = CFSTR("Abort");
  686. CFStringRef strIgnore = CFSTR("Ignore");
  687. CFStringRef strTryAgain = CFSTR("Try Again");
  688. CFStringRef strContinue = CFSTR("Continue");
  689. CFStringRef defaultButton = nullptr;
  690. CFStringRef alternativeButton = nullptr;
  691. CFStringRef otherButton = nullptr;
  692. switch (uType & 0xf)
  693. {
  694. case MB_OKCANCEL:
  695. defaultButton = strOk;
  696. alternativeButton = strCancel;
  697. break;
  698. case MB_ABORTRETRYIGNORE:
  699. defaultButton = strAbort;
  700. alternativeButton = strRetry;
  701. otherButton = strIgnore;
  702. break;
  703. case MB_YESNOCANCEL:
  704. defaultButton = strYes;
  705. alternativeButton = strNo;
  706. otherButton = strCancel;
  707. break;
  708. case MB_YESNO:
  709. defaultButton = strYes;
  710. alternativeButton = strNo;
  711. break;
  712. case MB_RETRYCANCEL:
  713. defaultButton = strRetry;
  714. alternativeButton = strCancel;
  715. break;
  716. case MB_CANCELTRYCONTINUE:
  717. defaultButton = strCancel;
  718. alternativeButton = strTryAgain;
  719. otherButton = strContinue;
  720. break;
  721. case MB_OK:
  722. default:
  723. defaultButton = strOk;
  724. break;
  725. }
  726. CFOptionFlags kResult;
  727. CFUserNotificationDisplayAlert(
  728. 0, // no timeout
  729. kCFUserNotificationNoteAlertLevel, //change it depending message_type flags ( MB_ICONASTERISK.... etc.)
  730. NULL, //icon url, use default, you can change it depending message_type flags
  731. NULL, //not used
  732. NULL, //localization of strings
  733. strText, //header text
  734. strCaption, //message text
  735. defaultButton, //default "ok" text in button
  736. alternativeButton, //alternate button title
  737. otherButton, //other button title, null--> no other button
  738. &kResult //response flags
  739. );
  740. if (strCaption)
  741. {
  742. CFRelease(strCaption);
  743. }
  744. if (strText)
  745. {
  746. CFRelease(strText);
  747. }
  748. #else
  749. printf("Messagebox: cap: %s text:%s\n", lpCaption ? lpCaption : " ", lpText ? lpText : " ");
  750. #endif
  751. }
  752. #if defined(LINUX) || defined(APPLE)
  753. threadID CryGetCurrentThreadId()
  754. {
  755. return GetCurrentThreadId();
  756. }
  757. #endif//LINUX APPLE
  758. #if defined(APPLE) || defined(LINUX)
  759. // WinAPI debug functions.
  760. AZ_DLL_EXPORT void OutputDebugString(const char* outputString)
  761. {
  762. #if !defined(_RELEASE)
  763. // Emulates dev tools output in Xcode and cmd line launch with idevicedebug.
  764. fprintf(stdout, "%s", outputString);
  765. #endif
  766. }
  767. #endif
  768. #endif // AZ_TRAIT_LEGACY_CRYCOMMON_USE_WINDOWS_STUBS