2
0

simDatablock.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2012 GarageGames, 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
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell 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
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  23. // Arcane-FX for MIT Licensed Open Source version of Torque 3D from GarageGames
  24. // Copyright (C) 2015 Faust Logic, Inc.
  25. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  26. #include "platform/platform.h"
  27. #include "console/simDatablock.h"
  28. #include "script.h"
  29. #include "console/console.h"
  30. #include "console/consoleInternal.h"
  31. #include "console/engineAPI.h"
  32. #include "T3D/gameBase/gameConnectionEvents.h"
  33. #include "T3D/gameBase/gameConnection.h"
  34. #include "core/stream/bitStream.h"
  35. IMPLEMENT_CO_DATABLOCK_V1(SimDataBlock);
  36. SimObjectId SimDataBlock::sNextObjectId = DataBlockObjectIdFirst;
  37. S32 SimDataBlock::sNextModifiedKey = 0;
  38. ConsoleDocClass( SimDataBlock,
  39. "@brief \n"
  40. "@ingroup \n"
  41. "@section Datablock_Networking Datablocks and Networking\n"
  42. "@section Datablock_ClientSide Client-Side Datablocks\n"
  43. );
  44. //-----------------------------------------------------------------------------
  45. SimDataBlock::SimDataBlock()
  46. {
  47. modifiedKey = 0;
  48. setModDynamicFields(true);
  49. setModStaticFields(true);
  50. }
  51. // this implements a simple structure for managing substitution statements.
  52. SimDataBlock::SubstitutionStatement::SubstitutionStatement(StringTableEntry slot, S32 idx, const char* value)
  53. {
  54. this->mSlot = slot;
  55. this->mIdx = idx;
  56. this->mValue = dStrdup(value);
  57. }
  58. SimDataBlock::SubstitutionStatement::~SubstitutionStatement()
  59. {
  60. dFree(mValue);
  61. }
  62. void SimDataBlock::SubstitutionStatement::replaceValue(const char* value)
  63. {
  64. dFree(this->mValue);
  65. this->mValue = dStrdup(value);
  66. }
  67. // this is the copy-constructor for creating temp-clones.
  68. SimDataBlock::SimDataBlock(const SimDataBlock& other, bool temp_clone) : SimObject(other, temp_clone)
  69. {
  70. modifiedKey = other.modifiedKey;
  71. }
  72. // a destructor is added to SimDataBlock so that we can delete any substitutions.
  73. SimDataBlock::~SimDataBlock()
  74. {
  75. clear_substitutions();
  76. }
  77. void SimDataBlock::clear_substitutions()
  78. {
  79. for (S32 i = 0; i < substitutions.size(); i++)
  80. delete substitutions[i];
  81. substitutions.clear();
  82. }
  83. void SimDataBlock::addSubstitution(StringTableEntry slot, S32 idx, const char* subst)
  84. {
  85. AssertFatal(subst != 0 && subst[0] == '$' && subst[1] == '$', "Bad substition statement string added");
  86. subst += 2;
  87. while (dIsspace(*subst))
  88. subst++;
  89. bool empty_subs = (*subst == '\0');
  90. for (S32 i = 0; i < substitutions.size(); i++)
  91. {
  92. if (substitutions[i] && substitutions[i]->mSlot == slot && substitutions[i]->mIdx == idx)
  93. {
  94. if (empty_subs)
  95. {
  96. delete substitutions[i];
  97. substitutions[i] = 0;
  98. onRemoveSubstitution(slot, idx);
  99. }
  100. else
  101. {
  102. substitutions[i]->replaceValue(subst);
  103. onAddSubstitution(slot, idx, subst);
  104. }
  105. return;
  106. }
  107. }
  108. if (!empty_subs)
  109. {
  110. substitutions.push_back(new SubstitutionStatement(slot, idx, subst));
  111. onAddSubstitution(slot, idx, subst);
  112. }
  113. }
  114. const char* SimDataBlock::getSubstitution(StringTableEntry slot, S32 idx)
  115. {
  116. for (S32 i = 0; i < substitutions.size(); i++)
  117. {
  118. if (substitutions[i] && substitutions[i]->mSlot == slot && substitutions[i]->mIdx == idx)
  119. return substitutions[i]->mValue;
  120. }
  121. return 0;
  122. }
  123. bool SimDataBlock::fieldHasSubstitution(StringTableEntry slot)
  124. {
  125. for (S32 i = 0; i < substitutions.size(); i++)
  126. if (substitutions[i] && substitutions[i]->mSlot == slot)
  127. return true;
  128. return false;
  129. }
  130. void SimDataBlock::printSubstitutions()
  131. {
  132. for (S32 i = 0; i < substitutions.size(); i++)
  133. if (substitutions[i])
  134. Con::errorf("SubstitutionStatement[%s] = \"%s\" -- %d", substitutions[i]->mSlot, substitutions[i]->mValue, i);
  135. }
  136. void SimDataBlock::copySubstitutionsFrom(SimDataBlock* other)
  137. {
  138. clear_substitutions();
  139. if (!other)
  140. return;
  141. for (S32 i = 0; i < other->substitutions.size(); i++)
  142. {
  143. if (other->substitutions[i])
  144. {
  145. SubstitutionStatement* subs = other->substitutions[i];
  146. substitutions.push_back(new SubstitutionStatement(subs->mSlot, subs->mIdx, subs->mValue));
  147. }
  148. }
  149. }
  150. // This is the method that evaluates any substitution statements on a datablock and does the
  151. // actual replacement of substituted datablock fields.
  152. //
  153. // Much of the work is done by passing the statement to Con::evaluate() but first there are
  154. // some key operations performed on the statement.
  155. // -- Instances of "%%" in the statement are replaced with the id of the <obj> object.
  156. // -- Instances of "##" are replaced with the value of <index>.
  157. //
  158. // There are also some return values that get special treatment.
  159. // -- An empty result will produce a realtime error message.
  160. // -- A result of "~~" will leave the original field value unchanged.
  161. // -- A result of "~0" will clear the original field to "" without producing an error message.
  162. //
  163. void SimDataBlock::performSubstitutions(SimDataBlock* dblock, const SimObject* obj, S32 index)
  164. {
  165. if (!dblock || !dblock->getClassRep())
  166. {
  167. // error message
  168. return;
  169. }
  170. char obj_str[32];
  171. dStrcpy(obj_str, Con::getIntArg(obj->getId()), 32);
  172. char index_str[32];
  173. dStrcpy(index_str, Con::getIntArg(index), 32);
  174. for (S32 i = 0; i < substitutions.size(); i++)
  175. {
  176. if (substitutions[i])
  177. {
  178. static char buffer[1024];
  179. static char* b_oob = &buffer[1024];
  180. char* b = buffer;
  181. // perform special token expansion (%% and ##)
  182. const char* v = substitutions[i]->mValue;
  183. while (*v != '\0')
  184. {
  185. // identify "%%" tokens and replace with <obj> id
  186. if (v[0] == '%' && v[1] == '%')
  187. {
  188. const char* s = obj_str;
  189. while (*s != '\0')
  190. {
  191. b[0] = s[0];
  192. b++;
  193. s++;
  194. }
  195. v += 2;
  196. }
  197. // identify "##" tokens and replace with <index> value
  198. else if (v[0] == '#' && v[1] == '#')
  199. {
  200. const char* s = index_str;
  201. while (*s != '\0')
  202. {
  203. b[0] = s[0];
  204. b++;
  205. s++;
  206. }
  207. v += 2;
  208. }
  209. else
  210. {
  211. b[0] = v[0];
  212. b++;
  213. v++;
  214. }
  215. }
  216. AssertFatal((uintptr_t)b < (uintptr_t)b_oob, "Substitution buffer overflowed");
  217. b[0] = '\0';
  218. Con::EvalResult evalResult = Con::evaluate(avar("return %s;", buffer), false, 0);
  219. if (!evalResult.valid)
  220. {
  221. Con::errorf("Field Substitution Failed: field=\"%s\" substitution=\"%s\" -- syntax error",
  222. substitutions[i]->mSlot, substitutions[i]->mValue);
  223. return;
  224. }
  225. const char* result = evalResult.value;
  226. // output a runtime console error when a substitution produces and empty result.
  227. if (result == 0 || result[0] == '\0')
  228. {
  229. Con::errorf("Field Substitution Failed: field=\"%s\" substitution=\"%s\" -- empty result",
  230. substitutions[i]->mSlot, substitutions[i]->mValue);
  231. return;
  232. }
  233. // handle special return values
  234. if (result[0] == '~')
  235. {
  236. // if value is "~~" then keep the existing value
  237. if (result[1] == '~' && result[2] == '\0')
  238. continue;
  239. // if "~0" then clear it
  240. if (result[1] == '0' && result[2] == '\0')
  241. result = "";
  242. }
  243. const AbstractClassRep::Field* field = dblock->getClassRep()->findField(substitutions[i]->mSlot);
  244. if (!field)
  245. {
  246. // this should be very unlikely...
  247. Con::errorf("Field Substitution Failed: unknown field, \"%s\".", substitutions[i]->mSlot);
  248. continue;
  249. }
  250. if (field->keepClearSubsOnly && result[0] != '\0')
  251. {
  252. Con::errorf("Field Substitution Failed: field \"%s\" of datablock %s only allows \"$$ ~~\" (keep) and \"$$ ~0\" (clear) field substitutions. [%s]",
  253. substitutions[i]->mSlot, this->getClassName(), this->getName());
  254. continue;
  255. }
  256. // substitute the field value with its replacement
  257. Con::setData(field->type, (void*)(((const char*)(dblock)) + field->offset), substitutions[i]->mIdx, 1, &result, field->table, field->flag);
  258. //dStrncpy(buffer, result, 255);
  259. //Con::errorf("SUBSTITUTION %s.%s[%d] = %s idx=%s", Con::getIntArg(getId()), substitutions[i]->slot, substitutions[i]->idx, buffer, index_str);
  260. // notify subclasses of a field modification
  261. dblock->onStaticModified(substitutions[i]->mSlot);
  262. }
  263. }
  264. // notify subclasses of substitution operation
  265. if (substitutions.size() > 0)
  266. dblock->onPerformSubstitutions();
  267. }
  268. //-----------------------------------------------------------------------------
  269. bool SimDataBlock::onAdd()
  270. {
  271. Parent::onAdd();
  272. // This initialization is done here, and not in the constructor,
  273. // because some jokers like to construct and destruct objects
  274. // (without adding them to the manager) to check what class
  275. // they are.
  276. modifiedKey = ++sNextModifiedKey;
  277. AssertFatal(sNextObjectId <= DataBlockObjectIdLast,
  278. "Exceeded maximum number of data blocks");
  279. // add DataBlock to the DataBlockGroup unless it is client side ONLY DataBlock
  280. if ( !isClientOnly() )
  281. if (SimGroup* grp = Sim::getDataBlockGroup())
  282. grp->addObject(this);
  283. Sim::getDataBlockSet()->addObject( this );
  284. return true;
  285. }
  286. //-----------------------------------------------------------------------------
  287. void SimDataBlock::assignId()
  288. {
  289. // We don't want the id assigned by the manager, but it may have
  290. // already been assigned a correct data block id.
  291. if ( isClientOnly() )
  292. setId(sNextObjectId++);
  293. }
  294. //-----------------------------------------------------------------------------
  295. void SimDataBlock::onStaticModified(const char* slotName, const char* newValue)
  296. {
  297. modifiedKey = sNextModifiedKey++;
  298. }
  299. //-----------------------------------------------------------------------------
  300. // packData() and unpackData() do nothing in the stock implementation, but here
  301. // they've been modified to pack and unpack any substitution statements.
  302. //
  303. void SimDataBlock::packData(BitStream* stream)
  304. {
  305. for (S32 i = 0; i < substitutions.size(); i++)
  306. {
  307. if (substitutions[i])
  308. {
  309. stream->writeFlag(true);
  310. stream->writeString(substitutions[i]->mSlot);
  311. stream->write(substitutions[i]->mIdx);
  312. stream->writeString(substitutions[i]->mValue);
  313. }
  314. }
  315. stream->writeFlag(false);
  316. }
  317. void SimDataBlock::unpackData(BitStream* stream)
  318. {
  319. clear_substitutions();
  320. while(stream->readFlag())
  321. {
  322. char slotName[256];
  323. S32 idx;
  324. char value[256];
  325. stream->readString(slotName);
  326. stream->read(&idx);
  327. stream->readString(value);
  328. substitutions.push_back(new SubstitutionStatement(StringTable->insert(slotName), idx, value));
  329. }
  330. }
  331. //-----------------------------------------------------------------------------
  332. bool SimDataBlock::preload(bool, String&)
  333. {
  334. return true;
  335. }
  336. //-----------------------------------------------------------------------------
  337. void SimDataBlock::write(Stream &stream, U32 tabStop, U32 flags)
  338. {
  339. // Only output selected objects if they want that.
  340. if((flags & SelectedOnly) && !isSelected())
  341. return;
  342. stream.writeTabs(tabStop);
  343. char buffer[1024];
  344. // Client side datablocks are created with 'new' while
  345. // regular server datablocks use the 'datablock' keyword.
  346. if ( isClientOnly() )
  347. dSprintf(buffer, sizeof(buffer), "new %s(%s) {\r\n", getClassName(), getName() ? getName() : "");
  348. else
  349. dSprintf(buffer, sizeof(buffer), "datablock %s(%s) {\r\n", getClassName(), getName() ? getName() : "");
  350. stream.write(dStrlen(buffer), buffer);
  351. writeFields(stream, tabStop + 1);
  352. stream.writeTabs(tabStop);
  353. stream.write(4, "};\r\n");
  354. }
  355. //=============================================================================
  356. // API.
  357. //=============================================================================
  358. // MARK: ---- API ----
  359. //-----------------------------------------------------------------------------
  360. DefineEngineMethod( SimDataBlock, reloadOnLocalClient, void, (),,
  361. "Reload the datablock. This can only be used with a local client configuration." )
  362. {
  363. // Make sure we're running a local client.
  364. GameConnection* localClient = GameConnection::getLocalClientConnection();
  365. if( !localClient )
  366. return;
  367. // Do an in-place pack/unpack/preload.
  368. if( !object->preload( true, NetConnection::getErrorBuffer() ) )
  369. {
  370. Con::errorf( NetConnection::getErrorBuffer() );
  371. return;
  372. }
  373. U8 buffer[ 16384 ];
  374. BitStream stream( buffer, 16384 );
  375. object->packData( &stream );
  376. stream.setPosition(0);
  377. object->unpackData( &stream );
  378. if( !object->preload( false, NetConnection::getErrorBuffer() ) )
  379. {
  380. Con::errorf( NetConnection::getErrorBuffer() );
  381. return;
  382. }
  383. // Trigger a post-apply so that change notifications respond.
  384. object->inspectPostApply();
  385. }
  386. //-----------------------------------------------------------------------------
  387. DefineEngineFunction( preloadClientDataBlocks, void, (),,
  388. "Preload all datablocks in client mode.\n\n"
  389. "(Server parameter is set to false). This will take some time to complete.")
  390. {
  391. // we go from last to first because we cut 'n pasted the loop from deleteDataBlocks
  392. SimGroup *grp = Sim::getDataBlockGroup();
  393. String errorStr;
  394. for(S32 i = grp->size() - 1; i >= 0; i--)
  395. {
  396. AssertFatal(dynamic_cast<SimDataBlock*>((*grp)[i]), "Doh! non-datablock in datablock group!");
  397. SimDataBlock *obj = (SimDataBlock*)(*grp)[i];
  398. if (!obj->preload(false, errorStr))
  399. Con::errorf("Failed to preload client datablock, %s: %s", obj->getName(), errorStr.c_str());
  400. }
  401. }
  402. //-----------------------------------------------------------------------------
  403. DefineEngineFunction( deleteDataBlocks, void, (),,
  404. "Delete all the datablocks we've downloaded.\n\n"
  405. "This is usually done in preparation of downloading a new set of datablocks, "
  406. "such as occurs on a mission change, but it's also good post-mission cleanup." )
  407. {
  408. // delete from last to first:
  409. SimGroup *grp = Sim::getDataBlockGroup();
  410. grp->deleteAllObjects();
  411. SimDataBlock::sNextObjectId = DataBlockObjectIdFirst;
  412. SimDataBlock::sNextModifiedKey = 0;
  413. }