CThreadHelper.cpp 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /*
  2. * CThreadHelper.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "CThreadHelper.h"
  12. #ifdef VCMI_WINDOWS
  13. #include <windows.h>
  14. #elif defined(VCMI_HAIKU)
  15. #include <OS.h>
  16. #elif !defined(VCMI_APPLE) && !defined(VCMI_FREEBSD) && !defined(VCMI_HURD)
  17. #include <sys/prctl.h>
  18. #endif
  19. VCMI_LIB_NAMESPACE_BEGIN
  20. CThreadHelper::CThreadHelper(std::vector<std::function<void()>> * Tasks, int Threads):
  21. currentTask(0),
  22. amount(static_cast<int>(Tasks->size())),
  23. tasks(Tasks),
  24. threads(Threads)
  25. {
  26. }
  27. void CThreadHelper::run()
  28. {
  29. std::vector<boost::thread> group;
  30. for(int i=0;i<threads;i++)
  31. group.emplace_back(std::bind(&CThreadHelper::processTasks,this));
  32. for (auto & thread : group)
  33. thread.join();
  34. //thread group deletes threads, do not free manually
  35. }
  36. void CThreadHelper::processTasks()
  37. {
  38. while(true)
  39. {
  40. int pom;
  41. {
  42. boost::unique_lock<boost::mutex> lock(rtinm);
  43. if((pom = currentTask) >= amount)
  44. break;
  45. else
  46. ++currentTask;
  47. }
  48. (*tasks)[pom]();
  49. }
  50. }
  51. // set name for this thread.
  52. // NOTE: on *nix string will be trimmed to 16 symbols
  53. void setThreadName(const std::string &name)
  54. {
  55. #ifdef VCMI_WINDOWS
  56. #ifndef __GNUC__
  57. //follows http://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx
  58. const DWORD MS_VC_EXCEPTION=0x406D1388;
  59. #pragma pack(push,8)
  60. typedef struct tagTHREADNAME_INFO
  61. {
  62. DWORD dwType; // Must be 0x1000.
  63. LPCSTR szName; // Pointer to name (in user addr space).
  64. DWORD dwThreadID; // Thread ID (-1=caller thread).
  65. DWORD dwFlags; // Reserved for future use, must be zero.
  66. } THREADNAME_INFO;
  67. #pragma pack(pop)
  68. THREADNAME_INFO info;
  69. info.dwType = 0x1000;
  70. info.szName = name.c_str();
  71. info.dwThreadID = -1;
  72. info.dwFlags = 0;
  73. __try
  74. {
  75. RaiseException( MS_VC_EXCEPTION, 0, sizeof(info)/sizeof(ULONG_PTR), (ULONG_PTR*)&info );
  76. }
  77. __except(EXCEPTION_EXECUTE_HANDLER)
  78. {
  79. }
  80. #else
  81. //not supported
  82. #endif
  83. #elif defined(__linux__)
  84. prctl(PR_SET_NAME, name.c_str(), 0, 0, 0);
  85. #elif defined(VCMI_APPLE)
  86. pthread_setname_np(name.c_str());
  87. #elif defined(VCMI_HAIKU)
  88. rename_thread(find_thread(NULL), name.c_str());
  89. #endif
  90. }
  91. VCMI_LIB_NAMESPACE_END