IPCPlayerApp.cpp 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. //
  2. // Copyright (c) 2014-2016 THUNDERBEAST GAMES LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to deal
  6. // in the Software without restriction, including without limitation the rights
  7. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. // copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  20. // THE SOFTWARE.
  21. #include <Atomic/Core/CoreEvents.h>//
  22. #include <Atomic/IO/IOEvents.h>
  23. #include <Atomic/Input/InputEvents.h>
  24. #include <Atomic/Graphics/Graphics.h>
  25. #include <Atomic/Graphics/GraphicsEvents.h>
  26. #include <Atomic/IPC/IPCEvents.h>
  27. #include <AtomicJS/Javascript/JSIPCEvents.h>
  28. #include "IPCPlayerAppEvents.h"
  29. #include <Atomic/Engine/Engine.h>
  30. #include <Atomic/IPC/IPC.h>
  31. #include <AtomicJS/Javascript/Javascript.h>
  32. #include <Atomic/UI/SystemUI/DebugHud.h>
  33. #include "IPCPlayerApp.h"
  34. namespace Atomic
  35. {
  36. IPCPlayerApp::IPCPlayerApp(Context* context) :
  37. PlayerApp(context),
  38. subprocess_(false),
  39. debugPlayer_(false),
  40. brokerActive_(false)
  41. {
  42. fd_[0] = INVALID_IPCHANDLE_VALUE;
  43. fd_[1] = INVALID_IPCHANDLE_VALUE;
  44. }
  45. IPCPlayerApp::~IPCPlayerApp()
  46. {
  47. }
  48. void IPCPlayerApp::Setup()
  49. {
  50. PlayerApp::Setup();
  51. // This should be configurable
  52. engineParameters_.InsertNew("LogLevel", LOG_DEBUG);
  53. ipc_ = new IPC(context_);
  54. context_->RegisterSubsystem(ipc_);
  55. }
  56. void IPCPlayerApp::ProcessArguments()
  57. {
  58. PlayerApp::ProcessArguments();
  59. FileSystem* fileSystem = GetSubsystem<FileSystem>();
  60. if (!fileSystem)
  61. {
  62. ErrorExit("IPCPlayerApp::ProcessArguments FileSystem subsystem does not exist");
  63. }
  64. String resourcePrefix;
  65. engineParameters_["ResourcePrefixPath"] = "";
  66. for (unsigned i = 0; i < arguments_.Size(); ++i)
  67. {
  68. if (arguments_[i].Length() > 1)
  69. {
  70. String argument = arguments_[i].ToLower();
  71. String value = i + 1 < arguments_.Size() ? arguments_[i + 1] : String::EMPTY;
  72. if (argument.StartsWith("--ipc-server=") || argument.StartsWith("--ipc-client="))
  73. {
  74. subprocess_ = true;
  75. }
  76. else if (argument == "--debug")
  77. {
  78. debugPlayer_ = true;
  79. }
  80. else if (argument == "--resourceprefix" && value.Length())
  81. {
  82. resourcePrefix = value;
  83. engineParameters_["ResourcePrefixPath"] = resourcePrefix;
  84. }
  85. else if (argument == "--project" && value.Length())
  86. {
  87. value = AddTrailingSlash(value);
  88. AddEngineConfigSearchPath(value + "Settings/");
  89. // check that cache exists
  90. if (!fileSystem->DirExists(value + "Cache"))
  91. {
  92. ErrorExit("Project cache folder does not exist, projects must be loaded into the Atomic Editor at least once before using the --player command line mode");
  93. return;
  94. }
  95. #ifdef ATOMIC_DEV_BUILD
  96. String resourcePaths = ToString("%s/Resources/CoreData;%s/Resources/PlayerData;%sResources;%s;%sCache",
  97. ATOMIC_ROOT_SOURCE_DIR, ATOMIC_ROOT_SOURCE_DIR, value.CString(), value.CString(), value.CString());
  98. #else
  99. #ifdef __APPLE__
  100. engineParameters_["ResourcePrefixPath"] = "../Resources";
  101. #else
  102. if (!resourcePrefix.Length())
  103. {
  104. engineParameters_["ResourcePrefixPath"] = fileSystem->GetProgramDir() + "Resources";
  105. }
  106. #endif
  107. String resourcePaths = ToString("CoreData;PlayerData;%s/;%s/Resources;%s;%sCache",
  108. value.CString(), value.CString(), value.CString(), value.CString());
  109. #endif
  110. LOGINFOF("Adding ResourcePaths: %s", resourcePaths.CString());
  111. engineParameters_["ResourcePaths"] = resourcePaths;
  112. }
  113. }
  114. }
  115. // IPC client player should not auto exit if a subprocess
  116. if (subprocess_)
  117. engine_->SetAutoExit(false);
  118. }
  119. void IPCPlayerApp::Start()
  120. {
  121. if (subprocess_)
  122. {
  123. // do not execute main in the player app
  124. executeJSMain_ = false;
  125. }
  126. PlayerApp::Start();
  127. int id = -1;
  128. if (IPC::ProcessArguments(arguments_, id, fd_[0], fd_[1]))
  129. {
  130. SubscribeToEvent(E_IPCINITIALIZE, HANDLER(IPCPlayerApp, HandleIPCInitialize));
  131. SubscribeToEvent(E_LOGMESSAGE, HANDLER(IPCPlayerApp, HandleLogMessage));
  132. SubscribeToEvent(E_JSERROR, HANDLER(IPCPlayerApp, HandleJSError));
  133. SubscribeToEvent(E_EXITREQUESTED, HANDLER(IPCPlayerApp, HandleExitRequest));
  134. SubscribeToEvent(E_SCREENMODE, HANDLER(IPCPlayerApp, HandlePlayerWindowChanged));
  135. SubscribeToEvent(E_WINDOWPOS, HANDLER(IPCPlayerApp, HandlePlayerWindowChanged));
  136. SubscribeToEvent(E_UPDATESPAUSEDRESUMED, HANDLER(IPCPlayerApp, HandleUpdatesPausedResumed));
  137. if (ipc_->InitWorker((unsigned)id, fd_[0], fd_[1]))
  138. {
  139. brokerActive_ = true;
  140. }
  141. else if (subprocess_)
  142. {
  143. LOGERROR("IPCPlayerApp::Start() - Unable to initialize IPC Worker");
  144. }
  145. }
  146. if (subprocess_)
  147. {
  148. JSVM* vm = JSVM::GetJSVM(0);
  149. if (!vm->ExecuteMain())
  150. {
  151. SendEvent(E_EXITREQUESTED);
  152. }
  153. SubscribeToEvent(E_PLAYERQUIT, HANDLER(IPCPlayerApp, HandleQuit));
  154. }
  155. }
  156. void IPCPlayerApp::HandleQuit(StringHash eventType, VariantMap& eventData)
  157. {
  158. engine_->Exit();
  159. }
  160. void IPCPlayerApp::Stop()
  161. {
  162. PlayerApp::Stop();
  163. }
  164. void IPCPlayerApp::HandleIPCInitialize(StringHash eventType, VariantMap& eventData)
  165. {
  166. brokerActive_ = true;
  167. SystemUI::DebugHud* debugHud = GetSubsystem<SystemUI::DebugHud>();
  168. if (debugHud)
  169. debugHud->SetMode(eventData["debugHudMode"].GetUInt());
  170. }
  171. void IPCPlayerApp::HandleJSError(StringHash eventType, VariantMap& eventData)
  172. {
  173. if (brokerActive_)
  174. {
  175. if (ipc_.Null())
  176. return;
  177. String errName = eventData[JSError::P_ERRORNAME].GetString();
  178. String errStack = eventData[JSError::P_ERRORSTACK].GetString();
  179. String errMessage = eventData[JSError::P_ERRORMESSAGE].GetString();
  180. String errFilename = eventData[JSError::P_ERRORFILENAME].GetString();
  181. int errLineNumber = eventData[JSError::P_ERRORLINENUMBER].GetInt();
  182. VariantMap ipcErrorData;
  183. ipcErrorData[IPCJSError::P_ERRORNAME] = errName;
  184. ipcErrorData[IPCJSError::P_ERRORSTACK] = errStack;
  185. ipcErrorData[IPCJSError::P_ERRORMESSAGE] = errMessage;
  186. ipcErrorData[IPCJSError::P_ERRORFILENAME] = errFilename;
  187. ipcErrorData[IPCJSError::P_ERRORLINENUMBER] = errLineNumber;
  188. ipc_->SendEventToBroker(E_IPCJSERROR, ipcErrorData);
  189. LOGERROR("SENDING E_IPCJSERROR");
  190. }
  191. }
  192. void IPCPlayerApp::HandlePlayerWindowChanged(StringHash eventType, VariantMap& eventData)
  193. {
  194. Graphics* graphics = GetSubsystem<Graphics>();
  195. using namespace IPCPlayerWindowChanged;
  196. VariantMap data;
  197. data[P_POSX] = graphics->GetWindowPosition().x_;
  198. data[P_POSY] = graphics->GetWindowPosition().y_;
  199. data[P_WIDTH] = graphics->GetWidth();
  200. data[P_HEIGHT] = graphics->GetHeight();
  201. data[P_MONITOR] = graphics->GetCurrentMonitor();
  202. data[P_MAXIMIZED] = graphics->GetMaximized();
  203. ipc_->SendEventToBroker(E_IPCPLAYERWINDOWCHANGED, data);
  204. }
  205. void IPCPlayerApp::HandleUpdatesPausedResumed(StringHash eventType, VariantMap& eventData)
  206. {
  207. ipc_->SendEventToBroker(E_IPCPLAYERUPDATESPAUSEDRESUMED, eventData);
  208. }
  209. void IPCPlayerApp::HandleExitRequest(StringHash eventType, VariantMap& eventData)
  210. {
  211. UnsubscribeFromEvent(E_LOGMESSAGE);
  212. SendEvent(E_PLAYERQUIT);
  213. }
  214. void IPCPlayerApp::HandleLogMessage(StringHash eventType, VariantMap& eventData)
  215. {
  216. using namespace LogMessage;
  217. if (brokerActive_)
  218. {
  219. if (ipc_.Null())
  220. return;
  221. VariantMap logEvent;
  222. logEvent[IPCWorkerLog::P_LEVEL] = eventData[P_LEVEL].GetInt();
  223. logEvent[IPCWorkerLog::P_MESSAGE] = eventData[P_MESSAGE].GetString();
  224. ipc_->SendEventToBroker(E_IPCWORKERLOG, logEvent);
  225. }
  226. }
  227. }