ASClassBinder.cpp 50 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403
  1. //
  2. // Copyright (c) 2008-2020 the Urho3D project.
  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. //
  22. #include "ASResult.h"
  23. #include "ASUtils.h"
  24. #include "Tuning.h"
  25. #include "Utils.h"
  26. #include "XmlAnalyzer.h"
  27. #include "XmlSourceData.h"
  28. #include <cassert>
  29. #include <iostream>
  30. #include <regex>
  31. #include <fstream>
  32. #include <vector>
  33. namespace ASBindingGenerator
  34. {
  35. static string _outputBasePath;
  36. static shared_ptr<ASGeneratedFile_Classes> _result_Classes;
  37. static shared_ptr<ASGeneratedFile_Members_HighPriority> _result_Members_HighPriority;
  38. static shared_ptr<ASGeneratedFile_Members> _result_Members_A;
  39. static shared_ptr<ASGeneratedFile_Members> _result_Members_B;
  40. static shared_ptr<ASGeneratedFile_Members> _result_Members_Constraint;
  41. static shared_ptr<ASGeneratedFile_Members> _result_Members_Ca_Cm;
  42. static shared_ptr<ASGeneratedFile_Members> _result_Members_Cn_Cz;
  43. static shared_ptr<ASGeneratedFile_Members> _result_Members_D;
  44. static shared_ptr<ASGeneratedFile_Members> _result_Members_E;
  45. static shared_ptr<ASGeneratedFile_Members> _result_Members_F;
  46. static shared_ptr<ASGeneratedFile_Members> _result_Members_G;
  47. static shared_ptr<ASGeneratedFile_Members> _result_Members_H;
  48. static shared_ptr<ASGeneratedFile_Members> _result_Members_I;
  49. static shared_ptr<ASGeneratedFile_Members> _result_Members_J;
  50. static shared_ptr<ASGeneratedFile_Members> _result_Members_K;
  51. static shared_ptr<ASGeneratedFile_Members> _result_Members_L;
  52. static shared_ptr<ASGeneratedFile_Members> _result_Members_M;
  53. static shared_ptr<ASGeneratedFile_Members> _result_Members_N;
  54. static shared_ptr<ASGeneratedFile_Members> _result_Members_O;
  55. static shared_ptr<ASGeneratedFile_Members> _result_Members_P;
  56. static shared_ptr<ASGeneratedFile_Members> _result_Members_Q;
  57. static shared_ptr<ASGeneratedFile_Members> _result_Members_R;
  58. static shared_ptr<ASGeneratedFile_Members> _result_Members_Sa_Sm;
  59. static shared_ptr<ASGeneratedFile_Members> _result_Members_Sn_Sz;
  60. static shared_ptr<ASGeneratedFile_Members> _result_Members_T;
  61. static shared_ptr<ASGeneratedFile_Members> _result_Members_U;
  62. static shared_ptr<ASGeneratedFile_Members> _result_Members_V;
  63. static shared_ptr<ASGeneratedFile_Members> _result_Members_W;
  64. static shared_ptr<ASGeneratedFile_Members> _result_Members_X;
  65. static shared_ptr<ASGeneratedFile_Members> _result_Members_Y;
  66. static shared_ptr<ASGeneratedFile_Members> _result_Members_Z;
  67. static shared_ptr<ASGeneratedFile_Members> _result_Members_Other;
  68. static shared_ptr<ASGeneratedFile_Templates> _result_Templates;
  69. static shared_ptr<ASGeneratedFile_Members> GetGeneratedFile(const string& className)
  70. {
  71. if (StartsWith(className, "A"))
  72. return _result_Members_A;
  73. if (StartsWith(className, "B"))
  74. return _result_Members_B;
  75. if (StartsWith(className, "Constraint"))
  76. return _result_Members_Constraint;
  77. if (StartsWith(className, "C"))
  78. {
  79. if (className.length() < 2)
  80. return _result_Members_Ca_Cm;
  81. char secondChar = className[1];
  82. if (secondChar >= 'n')
  83. return _result_Members_Cn_Cz;
  84. return _result_Members_Ca_Cm;
  85. }
  86. if (StartsWith(className, "D"))
  87. return _result_Members_D;
  88. if (StartsWith(className, "E"))
  89. return _result_Members_E;
  90. if (StartsWith(className, "F"))
  91. return _result_Members_F;
  92. if (StartsWith(className, "G"))
  93. return _result_Members_G;
  94. if (StartsWith(className, "H"))
  95. return _result_Members_H;
  96. if (StartsWith(className, "I"))
  97. return _result_Members_I;
  98. if (StartsWith(className, "J"))
  99. return _result_Members_J;
  100. if (StartsWith(className, "K"))
  101. return _result_Members_K;
  102. if (StartsWith(className, "L"))
  103. return _result_Members_L;
  104. if (StartsWith(className, "M"))
  105. return _result_Members_M;
  106. if (StartsWith(className, "N"))
  107. return _result_Members_N;
  108. if (StartsWith(className, "O"))
  109. return _result_Members_O;
  110. if (StartsWith(className, "P"))
  111. return _result_Members_P;
  112. if (StartsWith(className, "Q"))
  113. return _result_Members_Q;
  114. if (StartsWith(className, "R"))
  115. return _result_Members_R;
  116. if (StartsWith(className, "S"))
  117. {
  118. if (className.length() < 2)
  119. return _result_Members_Sa_Sm;
  120. char secondChar = className[1];
  121. if (secondChar >= 'n')
  122. return _result_Members_Sn_Sz;
  123. return _result_Members_Sa_Sm;
  124. }
  125. if (StartsWith(className, "T"))
  126. return _result_Members_T;
  127. if (StartsWith(className, "U"))
  128. return _result_Members_U;
  129. if (StartsWith(className, "V"))
  130. return _result_Members_V;
  131. if (StartsWith(className, "W"))
  132. return _result_Members_W;
  133. if (StartsWith(className, "X"))
  134. return _result_Members_X;
  135. if (StartsWith(className, "Y"))
  136. return _result_Members_Y;
  137. if (StartsWith(className, "Z"))
  138. return _result_Members_Z;
  139. if (StartsWith(className, "W"))
  140. return _result_Members_W;
  141. return _result_Members_Other;
  142. }
  143. //static void RegisterStaticFunction(const ClassStaticFunctionAnalyzer& functionAnalyzer, shared_ptr<ASGeneratedFile_Members> result)
  144. static void RegisterStaticFunction(const ClassStaticFunctionAnalyzer& functionAnalyzer, ASGeneratedFile_Base* result)
  145. {
  146. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  147. vector<ParamAnalyzer> params = functionAnalyzer.GetParams();
  148. vector<shared_ptr<FuncParamConv> > convertedParams;
  149. string outGlue;
  150. bool needWrapper = false;
  151. for (size_t i = 0; i < params.size(); i++)
  152. {
  153. ParamAnalyzer param = params[i];
  154. shared_ptr<FuncParamConv> conv = CppFunctionParamToAS(i, param);
  155. if (!conv->success_)
  156. {
  157. result->reg_ << " // " << conv->errorMessage_ << "\n";
  158. return;
  159. }
  160. if (conv->NeedWrapper())
  161. needWrapper = true;
  162. convertedParams.push_back(conv);
  163. }
  164. shared_ptr<FuncReturnTypeConv> convertedReturn = CppFunctionReturnTypeToAS(functionAnalyzer.GetReturnType());
  165. if (!convertedReturn->success_)
  166. {
  167. result->reg_ << " // " << GetLastErrorMessage() << "\n";
  168. return;
  169. }
  170. if (convertedReturn->needWrapper_)
  171. needWrapper = true;
  172. string declParams;
  173. for (shared_ptr<FuncParamConv> conv : convertedParams)
  174. {
  175. if (declParams.length() > 0)
  176. declParams += ", ";
  177. declParams += conv->asDecl_;
  178. }
  179. string asFunctionName = functionAnalyzer.GetName();
  180. string className = functionAnalyzer.GetClassName();
  181. if (needWrapper)
  182. result->glue_ << GenerateWrapper(functionAnalyzer, convertedParams, convertedReturn);
  183. string decl = convertedReturn->asReturnType_ + " " + asFunctionName + "(" + declParams + ")";
  184. result->reg_ << " engine->SetDefaultNamespace(\"" << className << "\");\n";
  185. result->reg_ << " engine->RegisterGlobalFunction(\"" << decl << "\", ";
  186. if (needWrapper)
  187. result->reg_ << "asFUNCTION(" << GenerateWrapperName(functionAnalyzer) << "), asCALL_CDECL);\n";
  188. else
  189. result->reg_ << Generate_asFUNCTIONPR(functionAnalyzer) << ", asCALL_CDECL);\n";
  190. result->reg_ << " engine->SetDefaultNamespace(\"\");\n";
  191. }
  192. static void RegisterRefCountedConstructor(const ClassFunctionAnalyzer& functionAnalyzer, bool templateVersion)
  193. {
  194. string className = functionAnalyzer.GetClassName();
  195. string wrapperName = GenerateWrapperName(functionAnalyzer, templateVersion);
  196. string header = functionAnalyzer.GetClass().GetHeaderFile();
  197. string insideDefine = InsideDefine(header);
  198. string args = ExtractArgsstring(functionAnalyzer.GetMemberdef());
  199. assert(!args.empty());
  200. assert(StartsWith(args, "("));
  201. size_t endPos = args.find_last_of(')'); // After ')' can be " override"
  202. assert(endPos != string::npos);
  203. args = args.substr(1, endPos - 1);
  204. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(functionAnalyzer.GetClass().GetClassName());
  205. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(functionAnalyzer.GetClass().GetClassName()).get();
  206. bool firstContext = StartsWith(args, "Context *context");
  207. if (firstContext)
  208. {
  209. args = CutStart(args, "Context *context");
  210. args = CutStart(args, ", ");
  211. }
  212. else if (args.find("Context") != string::npos)
  213. {
  214. result->reg_ <<
  215. " // " << functionAnalyzer.GetLocation() << "\n"
  216. " // Error: context can be only first\n";
  217. return;
  218. }
  219. string cppParamsNames = functionAnalyzer.JoinParamsNames(firstContext);
  220. if (firstContext)
  221. {
  222. if (cppParamsNames.length() > 0)
  223. cppParamsNames = ", " + cppParamsNames;
  224. cppParamsNames = "GetScriptContext()" + cppParamsNames;
  225. }
  226. string decl = "";
  227. vector<ParamAnalyzer> params = functionAnalyzer.GetParams();
  228. for (unsigned i = 0; i < params.size(); i++)
  229. {
  230. ParamAnalyzer param = params[i];
  231. if (i == 0 && firstContext)
  232. {
  233. assert(param.GetType().ToString() == "Context*");
  234. continue;
  235. }
  236. if (decl.length() > 0)
  237. decl += ", ";
  238. bool outSuccess;
  239. decl += CppTypeToAS(param.GetType(), false, outSuccess);
  240. string defval = param.GetDefval();
  241. if (!defval.empty())
  242. {
  243. defval = CppValueToAS(defval);
  244. defval = ReplaceAll(defval, "\"", "\\\"");
  245. decl += /*" " + param.GetDeclname() +*/ " = " + defval;
  246. }
  247. if (!outSuccess)
  248. {
  249. result->reg_ <<
  250. " // " << functionAnalyzer.GetLocation() << "\n"
  251. " // " << GetLastErrorMessage() << "\n";
  252. return;
  253. }
  254. }
  255. decl = className + "@+ f(" + decl + ")";
  256. if (templateVersion)
  257. {
  258. result->reg_ <<
  259. " // " << functionAnalyzer.GetLocation() << "\n"
  260. " {\n"
  261. " String declFactory(String(className) + \"@ f()\");\n"
  262. " engine->RegisterObjectBehaviour(className, "
  263. "asBEHAVE_FACTORY, "
  264. "declFactory.CString(), "
  265. "asFUNCTION(" << wrapperName << "), "
  266. "asCALL_CDECL);\n"
  267. " }\n";
  268. }
  269. else
  270. {
  271. result->reg_ <<
  272. " // " << functionAnalyzer.GetLocation() << "\n"
  273. " engine->RegisterObjectBehaviour("
  274. "\"" << className << "\", "
  275. "asBEHAVE_FACTORY, "
  276. "\"" << decl << "\", "
  277. "asFUNCTION(" << wrapperName << "), "
  278. "asCALL_CDECL);\n";
  279. }
  280. if (!insideDefine.empty())
  281. result->glue_ << "#ifdef " << insideDefine << "\n";
  282. result->glue_ <<
  283. "// " << functionAnalyzer.GetLocation() << "\n"
  284. "static " << className << "* " << wrapperName << "(" << args << ")\n"
  285. "{\n"
  286. " return new " << className << "(" << cppParamsNames << ");\n"
  287. "}\n";
  288. if (!insideDefine.empty())
  289. result->glue_ << "#endif\n";
  290. result->glue_ << "\n";
  291. }
  292. static void RegisterValueConstructor(const ClassFunctionAnalyzer& functionAnalyzer, bool templateVersion)
  293. {
  294. string header = functionAnalyzer.GetClass().GetHeaderFile();
  295. string insideDefine = InsideDefine(header);
  296. string className = functionAnalyzer.GetClassName();
  297. string wrapperName = GenerateWrapperName(functionAnalyzer, templateVersion);
  298. string args = ExtractArgsstring(functionAnalyzer.GetMemberdef());
  299. assert(!args.empty());
  300. assert(StartsWith(args, "("));
  301. size_t endPos = args.find_last_of(')'); // After ')' can be " override"
  302. assert(endPos != string::npos);
  303. args = args.substr(1, endPos - 1);
  304. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  305. bool isDefaultConstructor = args.empty();
  306. stringstream& reg = isDefaultConstructor ? _result_Members_HighPriority->reg_ : result->reg_;
  307. stringstream& glue = isDefaultConstructor ? _result_Members_HighPriority->glue_ : result->glue_;
  308. string decl = "";
  309. vector<ParamAnalyzer> params = functionAnalyzer.GetParams();
  310. for (ParamAnalyzer param : params)
  311. {
  312. if (decl.length() > 0)
  313. decl += ", ";
  314. bool outSuccess;
  315. decl += CppTypeToAS(param.GetType(), false, outSuccess);
  316. string defval = param.GetDefval();
  317. if (!defval.empty())
  318. {
  319. defval = CppValueToAS(defval);
  320. defval = ReplaceAll(defval, "\"", "\\\"");
  321. decl += /*" " + param.GetDeclname() +*/ " = " + defval;
  322. }
  323. if (!outSuccess)
  324. {
  325. if (isDefaultConstructor && !insideDefine.empty())
  326. reg << "#ifdef " << insideDefine << "\n";
  327. reg <<
  328. " // " << functionAnalyzer.GetLocation() << "\n"
  329. " // " << GetLastErrorMessage() << "\n";
  330. if (isDefaultConstructor && !insideDefine.empty())
  331. reg << "#endif\n";
  332. return;
  333. }
  334. }
  335. decl = "void f(" + decl + ")";
  336. if (isDefaultConstructor && !insideDefine.empty())
  337. reg << "#ifdef " << insideDefine << "\n";
  338. reg <<
  339. " // " << functionAnalyzer.GetLocation() << "\n"
  340. " engine->RegisterObjectBehaviour("
  341. "\"" << className << "\", "
  342. "asBEHAVE_CONSTRUCT, "
  343. "\"" << decl << "\", "
  344. "asFUNCTION(" << wrapperName << "), "
  345. "asCALL_CDECL_OBJFIRST);\n";
  346. if (isDefaultConstructor && !insideDefine.empty())
  347. reg << "#endif\n";
  348. if (isDefaultConstructor && !insideDefine.empty())
  349. glue << "#ifdef " << insideDefine << "\n";
  350. glue <<
  351. "// " << functionAnalyzer.GetLocation() << "\n"
  352. "static void " << wrapperName << "(" << className << "* ptr";
  353. if (args.length() > 0)
  354. glue << ", " << args;
  355. glue << ")\n"
  356. "{\n"
  357. " new(ptr) " << className << "(" << functionAnalyzer.JoinParamsNames() << ");\n"
  358. "}\n";
  359. if (isDefaultConstructor && !insideDefine.empty())
  360. glue << "#endif\n";
  361. glue << "\n";
  362. if (isDefaultConstructor)
  363. _result_Members_HighPriority->AddHeader(header);
  364. }
  365. static void RegisterValueDestructor(const ClassFunctionAnalyzer& functionAnalyzer, bool templateVersion)
  366. {
  367. string className = functionAnalyzer.GetClassName();
  368. string wrapperName = GenerateWrapperName(functionAnalyzer, templateVersion);
  369. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  370. result->glue_ <<
  371. "// " << functionAnalyzer.GetLocation() << "\n"
  372. "static void " << wrapperName << "(" << className << "* ptr)\n"
  373. "{\n"
  374. " ptr->~" << className << "();\n"
  375. "}\n\n";
  376. result->reg_ << " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_DESTRUCT, \"void f()\", asFUNCTION(" << wrapperName << "), asCALL_CDECL_OBJFIRST);\n";
  377. }
  378. static void RegisterImplicitlyDeclaredDestructor(const ClassAnalyzer& classAnalyzer)
  379. {
  380. string className = classAnalyzer.GetClassName();
  381. string wrapperName = className + "_Destructor";
  382. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  383. string header = classAnalyzer.GetHeaderFile();
  384. string insideDefine = InsideDefine(header);
  385. if (!insideDefine.empty())
  386. result->glue_ << "#ifdef " << insideDefine << "\n";
  387. result->glue_ <<
  388. "// " << className << "::~" << className << "() | Implicitly-declared \n"
  389. "static void " << wrapperName << "(" << className << "* ptr)\n"
  390. "{\n"
  391. " ptr->~" << className << "();\n"
  392. "}\n";
  393. if (!insideDefine.empty())
  394. result->glue_ << "#endif\n";
  395. result->glue_ << "\n";
  396. result->reg_ <<
  397. " // " << className << "::~" << className << "() | Implicitly-declared\n"
  398. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_DESTRUCT, \"void f()\", asFUNCTION(" << wrapperName << "), asCALL_CDECL_OBJFIRST);\n";
  399. }
  400. static void RegisterImplicitlyDeclaredConstructor(const ClassAnalyzer& classAnalyzer)
  401. {
  402. string header = classAnalyzer.GetHeaderFile();
  403. string insideDefine = InsideDefine(header);
  404. string className = classAnalyzer.GetClassName();
  405. string wrapperName = className + "_Constructor";
  406. _result_Members_HighPriority->AddHeader(header);
  407. if (!insideDefine.empty())
  408. {
  409. _result_Members_HighPriority->glue_ << "#ifdef " << insideDefine << "\n";
  410. _result_Members_HighPriority->reg_ << "#ifdef " << insideDefine << "\n";
  411. }
  412. _result_Members_HighPriority->glue_ <<
  413. "// " << className << "::" << className << "() | Implicitly-declared\n"
  414. "static void " << wrapperName << "(" << className << "* ptr)\n"
  415. "{\n"
  416. " new(ptr) " << className << "();\n"
  417. "}\n";
  418. _result_Members_HighPriority->reg_ <<
  419. " // " << className << "::" << className << "() | Implicitly-declared\n"
  420. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_CONSTRUCT, \"void f()\", asFUNCTION(" << wrapperName << "), asCALL_CDECL_OBJFIRST);\n";
  421. if (!insideDefine.empty())
  422. {
  423. _result_Members_HighPriority->glue_ << "#endif\n";
  424. _result_Members_HighPriority->reg_ << "#endif\n";
  425. }
  426. _result_Members_HighPriority->glue_ << "\n";
  427. }
  428. static void TryRegisterImplicitlyDeclaredAssignOperator(const ClassAnalyzer& classAnalyzer)
  429. {
  430. string className = classAnalyzer.GetClassName();
  431. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  432. result->reg_ <<
  433. " // " << className << "& " << className << "::operator=(const " << className << "&) | Possible implicitly-declared\n"
  434. " RegisterImplicitlyDeclaredAssignOperatorIfPossible<" << className << ">(engine, \"" << className << "\");\n";
  435. }
  436. static bool IsDestructorRequired(const ClassAnalyzer& classAnalyzer)
  437. {
  438. if (classAnalyzer.IsRefCounted())
  439. return false;
  440. if (Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  441. return false;
  442. if (classAnalyzer.IsPod())
  443. return false;
  444. return true;
  445. }
  446. static bool IsConstructorRequired(const ClassAnalyzer& classAnalyzer)
  447. {
  448. if (classAnalyzer.IsRefCounted())
  449. return false;
  450. if (Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  451. return false;
  452. if (classAnalyzer.IsPod())
  453. return false;
  454. return true;
  455. }
  456. // Some required methods can not be bound automatically when processing class because implicitly-declared
  457. static void RegisterImplicitlyDeclaredMethods(const ClassAnalyzer& classAnalyzer)
  458. {
  459. if (!classAnalyzer.HasThisConstructor() && IsConstructorRequired(classAnalyzer))
  460. RegisterImplicitlyDeclaredConstructor(classAnalyzer);
  461. if (!classAnalyzer.HasDestructor() && IsDestructorRequired(classAnalyzer))
  462. RegisterImplicitlyDeclaredDestructor(classAnalyzer);
  463. if (!classAnalyzer.ContainsFunction("operator="))
  464. TryRegisterImplicitlyDeclaredAssignOperator(classAnalyzer);
  465. // No need register impicitly declared copy constructor when registered opAssign operator
  466. }
  467. static void RegisterComparisonOperator(const ClassAnalyzer& classAnalyzer)
  468. {
  469. string className = classAnalyzer.GetClassName();
  470. string wrapperName = className + "_Comparison";
  471. string operatorLessLocation = classAnalyzer.GetFunction("operator<").GetLocation();
  472. string operatorGreaterLocation = classAnalyzer.GetFunction("operator>").GetLocation();
  473. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  474. result->glue_ <<
  475. "// " << operatorLessLocation << "\n"
  476. "// " << operatorGreaterLocation << "\n"
  477. "static int " << wrapperName << "(const " << className << "& lhs, const " << className << "& rhs)\n"
  478. "{\n"
  479. " if (lhs < rhs)\n"
  480. " return -1;\n\n"
  481. " if (lhs > rhs)\n"
  482. " return 1;\n\n"
  483. " return 0;\n"
  484. "}\n\n";
  485. result->reg_ <<
  486. " // " << operatorLessLocation << "\n"
  487. " // " << operatorGreaterLocation << "\n"
  488. " engine->RegisterObjectMethod(\"" << className << "\", \"int opCmp(const " << className << "&in) const\", asFUNCTION(" << wrapperName << "), asCALL_CDECL_OBJFIRST);\n";
  489. }
  490. static void RegisterAddReleaseRef(const ClassFunctionAnalyzer& functionAnalyzer, bool templateVersion)
  491. {
  492. string className = functionAnalyzer.GetClassName();
  493. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  494. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(functionAnalyzer.GetClass().GetClassName()).get();
  495. string functionName = functionAnalyzer.GetName();
  496. string behaviour = (functionName == "AddRef") ? "asBEHAVE_ADDREF" : "asBEHAVE_RELEASE";
  497. string methodpr = Generate_asMETHODPR(functionAnalyzer, templateVersion);
  498. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  499. if (templateVersion)
  500. result->reg_ << " engine->RegisterObjectBehaviour(className, " << behaviour << ", \"void f()\", " << methodpr << ", asCALL_THISCALL);\n";
  501. else
  502. result->reg_ << " engine->RegisterObjectBehaviour(\"" << className << "\", " << behaviour << ", \"void f()\", " << methodpr << ", asCALL_THISCALL);\n";
  503. }
  504. static void RegisterFakeAddReleaseRef(const ClassAnalyzer& classAnalyzer)
  505. {
  506. string className = classAnalyzer.GetClassName();
  507. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  508. result->reg_ <<
  509. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_ADDREF, \"void f()\", asFUNCTION(FakeAddRef), asCALL_CDECL_OBJLAST);\n"
  510. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_RELEASE, \"void f()\", asFUNCTION(FakeReleaseRef), asCALL_CDECL_OBJLAST);\n";
  511. }
  512. // https://www.angelcode.com/angelscript/sdk/docs/manual/doc_script_class_ops.html
  513. string CppMethodNameToAS(const ClassFunctionAnalyzer& functionAnalyzer, bool& outSuccess)
  514. {
  515. string name = functionAnalyzer.GetName();
  516. if (name == "operator=")
  517. {
  518. outSuccess = true;
  519. return "opAssign";
  520. }
  521. if (name == "operator+")
  522. {
  523. outSuccess = true;
  524. return "opAdd";
  525. }
  526. if (name == "operator-")
  527. {
  528. outSuccess = true;
  529. if (!functionAnalyzer.GetParams().size()) // If no params
  530. return "opNeg"; // then unary minus
  531. else
  532. return "opSub";
  533. }
  534. if (name == "operator*")
  535. {
  536. outSuccess = true;
  537. return "opMul";
  538. }
  539. if (name == "operator/")
  540. {
  541. outSuccess = true;
  542. return "opDiv";
  543. }
  544. if (name == "operator+=")
  545. {
  546. outSuccess = true;
  547. return "opAddAssign";
  548. }
  549. if (name == "operator-=")
  550. {
  551. outSuccess = true;
  552. return "opSubAssign";
  553. }
  554. if (name == "operator*=")
  555. {
  556. outSuccess = true;
  557. return "opMulAssign";
  558. }
  559. if (name == "operator/=")
  560. {
  561. outSuccess = true;
  562. return "opDivAssign";
  563. }
  564. if (name == "operator==")
  565. {
  566. outSuccess = true;
  567. return "opEquals";
  568. }
  569. if (name == "operator[]")
  570. {
  571. outSuccess = true;
  572. return "opIndex";
  573. }
  574. // Conversion to another type operator
  575. if (StartsWith(name, "operator "))
  576. {
  577. outSuccess = true;
  578. if (functionAnalyzer.IsExplicit())
  579. return "opConv";
  580. else
  581. return "opImplConv";
  582. }
  583. if (name == "operator!=")
  584. {
  585. outSuccess = false;
  586. SetLastErrorMessage("Only operator== is needed");
  587. return "IGNORED";
  588. }
  589. if (name == "operator<")
  590. {
  591. outSuccess = false;
  592. SetLastErrorMessage("Registerd as opCmp separately");
  593. return "IGNORED";
  594. }
  595. if (name == "operator>")
  596. {
  597. outSuccess = false;
  598. SetLastErrorMessage("Registerd as opCmp separately");
  599. return "IGNORED";
  600. }
  601. outSuccess = true;
  602. return name;
  603. }
  604. // Iterate over overrided funcions
  605. static bool HaveMark(const ClassFunctionAnalyzer& functionAnalyzer, const string& mark)
  606. {
  607. if (Contains(functionAnalyzer.GetComment(), mark))
  608. return true;
  609. shared_ptr<ClassFunctionAnalyzer> reimplements = functionAnalyzer.Reimplements();
  610. if (!reimplements)
  611. return false;
  612. return HaveMark(*reimplements, mark);
  613. }
  614. // Can return BIND_AS_ALIAS_xxxx or BIND_AS_PROPERTY
  615. // Return "" if no this marks
  616. static string GetPropertyMark(ClassFunctionAnalyzer function)
  617. {
  618. string comment = function.GetComment();
  619. smatch match;
  620. regex_match(comment, match, regex(".*\\b(BIND_AS_ALIAS_.+?)\\b.*"));
  621. if (match.size() == 2)
  622. return match[1].str();
  623. regex_match(comment, match, regex(".*\\bBIND_AS_PROPERTY\\b.*"));
  624. if (match.size() == 1)
  625. return "BIND_AS_PROPERTY";
  626. shared_ptr<ClassFunctionAnalyzer> reimplements = function.Reimplements();
  627. if (!reimplements)
  628. return "";
  629. return GetPropertyMark(*reimplements);
  630. }
  631. // https://www.angelcode.com/angelscript/sdk/docs/manual/doc_reg_objprop.html
  632. static string CppMethodNameToASProperty(const ClassFunctionAnalyzer& functionAnalyzer, bool& outSuccess)
  633. {
  634. string name = functionAnalyzer.GetName();
  635. if (StartsWith(name, "Is") || StartsWith(name, "Get"))
  636. {
  637. outSuccess = true;
  638. string result = CutStart(name, "Is");
  639. result = CutStart(result, "Get");
  640. result = "get_" + FirstCharToLower(result);
  641. return result;
  642. }
  643. if (StartsWith(name, "Set"))
  644. {
  645. outSuccess = true;
  646. string result = CutStart(name, "Set");
  647. result = "set_" + FirstCharToLower(result);
  648. return result;
  649. }
  650. if (functionAnalyzer.CanBeGetProperty())
  651. {
  652. outSuccess = true;
  653. string result = name;
  654. result = "get_" + FirstCharToLower(result);
  655. return result;
  656. }
  657. if (functionAnalyzer.CanBeSetProperty())
  658. {
  659. outSuccess = true;
  660. string result = name;
  661. result = "set_" + FirstCharToLower(result);
  662. return result;
  663. }
  664. outSuccess = false;
  665. SetLastErrorMessage("ERROR: Can not be property");
  666. return "ERROR";
  667. }
  668. static void RegisterMethod(const ClassFunctionAnalyzer& functionAnalyzer, bool templateVersion)
  669. {
  670. if (functionAnalyzer.IsDefine()) // URHO3D_OBJECT
  671. return;
  672. if (!functionAnalyzer.IsPublic())
  673. return;
  674. if (functionAnalyzer.IsParentDestructor())
  675. return;
  676. if (functionAnalyzer.IsParentConstructor())
  677. return;
  678. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(functionAnalyzer.GetClass().GetClassName());
  679. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(functionAnalyzer.GetClass().GetClassName()).get();
  680. if (HaveMark(functionAnalyzer, "NO_BIND"))
  681. {
  682. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  683. result->reg_ << " // Not registered because have @nobind mark\n";
  684. return;
  685. }
  686. if (HaveMark(functionAnalyzer, "MANUAL_BIND"))
  687. {
  688. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  689. result->reg_ << " // Not registered because have @manualbind mark\n";
  690. return;
  691. }
  692. if (functionAnalyzer.IsTemplate())
  693. {
  694. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  695. result->reg_ << " // Not registered because template\n";
  696. return;
  697. }
  698. if (functionAnalyzer.IsStatic())
  699. {
  700. ClassStaticFunctionAnalyzer classStaticFunctionAnalyzer(functionAnalyzer.GetClass(), functionAnalyzer.GetMemberdef());
  701. RegisterStaticFunction(classStaticFunctionAnalyzer, result);
  702. return;
  703. }
  704. if (functionAnalyzer.IsDeleted())
  705. return;
  706. /*if (function.IsPureVirtual()) // Register pure virtual function but not register constructor for this class
  707. return;*/
  708. // Do not register destructor for refcounted because object is deleted by himself
  709. if (functionAnalyzer.IsThisDestructor()
  710. && (functionAnalyzer.GetClass().IsRefCounted() || Contains(functionAnalyzer.GetClass().GetComment(), "FAKE_REF")))
  711. {
  712. return;
  713. }
  714. string functionName = functionAnalyzer.GetName();
  715. if (functionName == "operator!=")
  716. return;
  717. // Already registered as sigle function opCmp
  718. if (functionName == "operator<" || functionName == "operator>")
  719. return;
  720. if (functionName == "AddRef" || functionName == "ReleaseRef")
  721. {
  722. RegisterAddReleaseRef(functionAnalyzer, templateVersion);
  723. return;
  724. }
  725. if (functionAnalyzer.IsThisConstructor())
  726. {
  727. if (functionAnalyzer.GetClass().IsRefCounted() || Contains(functionAnalyzer.GetClass().GetComment(), "FAKE_REF"))
  728. {
  729. // Do not register construnctor for abstract class
  730. if (functionAnalyzer.GetClass().IsAbstract())
  731. return;
  732. RegisterRefCountedConstructor(functionAnalyzer, templateVersion);
  733. }
  734. else
  735. {
  736. RegisterValueConstructor(functionAnalyzer, templateVersion);
  737. }
  738. return;
  739. }
  740. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  741. if (functionAnalyzer.IsThisDestructor())
  742. {
  743. // Do not register destructor for reference type
  744. if (!functionAnalyzer.GetClass().IsRefCounted())
  745. RegisterValueDestructor(functionAnalyzer, templateVersion);
  746. return;
  747. }
  748. vector<ParamAnalyzer> params = functionAnalyzer.GetParams();
  749. vector<shared_ptr<FuncParamConv> > convertedParams;
  750. bool needWrapper = false;
  751. for (size_t i = 0; i < params.size(); i++)
  752. {
  753. ParamAnalyzer param = params[i];
  754. shared_ptr<FuncParamConv> conv = CppFunctionParamToAS(i, param);
  755. if (!conv->success_)
  756. {
  757. result->reg_ << " // " << conv->errorMessage_ << "\n";
  758. return;
  759. }
  760. if (conv->NeedWrapper())
  761. needWrapper = true;
  762. convertedParams.push_back(conv);
  763. }
  764. shared_ptr<FuncReturnTypeConv> retConv = CppFunctionReturnTypeToAS(functionAnalyzer.GetReturnType());
  765. if (!retConv->success_)
  766. {
  767. result->reg_ << " // " << retConv->errorMessage_ << "\n";
  768. return;
  769. }
  770. if (retConv->needWrapper_)
  771. needWrapper = true;
  772. string declParams = "";
  773. for (shared_ptr<FuncParamConv> conv : convertedParams)
  774. {
  775. if (declParams.length() > 0)
  776. declParams += ", ";
  777. declParams += conv->asDecl_;
  778. }
  779. string asReturnType = retConv->asReturnType_;
  780. string asFunctionName = functionAnalyzer.GetName();
  781. if (functionAnalyzer.IsConsversionOperator())
  782. asReturnType = CutStart(asFunctionName, "operator ");
  783. bool outSuccess;
  784. asFunctionName = CppMethodNameToAS(functionAnalyzer, outSuccess);
  785. if (!outSuccess)
  786. {
  787. result->reg_ << " // " << GetLastErrorMessage() << "\n";
  788. return;
  789. }
  790. if (needWrapper)
  791. result->glue_ << GenerateWrapper(functionAnalyzer, templateVersion, convertedParams, retConv);
  792. string decl = asReturnType + " " + asFunctionName + "(" + declParams + ")";
  793. if (functionAnalyzer.IsConst())
  794. decl += " const";
  795. if (templateVersion)
  796. result->reg_ << " engine->RegisterObjectMethod(className, \"" << decl << "\", ";
  797. else
  798. result->reg_ << " engine->RegisterObjectMethod(\"" << functionAnalyzer.GetClassName() << "\", \"" << decl << "\", ";
  799. if (needWrapper)
  800. result->reg_ << "asFUNCTION(" << GenerateWrapperName(functionAnalyzer, templateVersion) << "), asCALL_CDECL_OBJFIRST);\n";
  801. else
  802. result->reg_ << Generate_asMETHODPR(functionAnalyzer, templateVersion) << ", asCALL_THISCALL);\n";
  803. // Also register as property if needed
  804. string propertyMark = GetPropertyMark(functionAnalyzer);
  805. if (!propertyMark.empty())
  806. {
  807. if (StartsWith(propertyMark, "BIND_AS_ALIAS_"))
  808. {
  809. asFunctionName = CutStart(propertyMark, "BIND_AS_ALIAS_");
  810. }
  811. else
  812. {
  813. asFunctionName = CppMethodNameToASProperty(functionAnalyzer, outSuccess);
  814. if (!outSuccess)
  815. {
  816. result->reg_ << " // " << GetLastErrorMessage() << "\n";
  817. return;
  818. }
  819. }
  820. decl = asReturnType + " " + asFunctionName + "(" + declParams + ")";
  821. if (functionAnalyzer.IsConst())
  822. decl += " const";
  823. if (templateVersion)
  824. result->reg_ << " engine->RegisterObjectMethod(className, \"" << decl << "\", ";
  825. else
  826. result->reg_ << " engine->RegisterObjectMethod(\"" << functionAnalyzer.GetClassName() << "\", \"" << decl << "\", ";
  827. if (needWrapper)
  828. result->reg_ << "asFUNCTION(" << GenerateWrapperName(functionAnalyzer, templateVersion) << "), asCALL_CDECL_OBJFIRST);\n";
  829. else
  830. result->reg_ << Generate_asMETHODPR(functionAnalyzer, templateVersion) << ", asCALL_THISCALL);\n";
  831. }
  832. }
  833. static void RegisterClassVarAsProperty(ClassVariableAnalyzer& variable)
  834. {
  835. if (!variable.IsPublic())
  836. return;
  837. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(variable.GetClassName());
  838. if (Contains(variable.GetComment(), "NO_BIND"))
  839. {
  840. result->reg_ << " // " << variable.GetLocation() << "\n";
  841. result->reg_ << " // Not registered because have @nobind mark\n";
  842. return;
  843. }
  844. if (Contains(variable.GetComment(), "MANUAL_BIND"))
  845. {
  846. result->reg_ << " // " << variable.GetLocation() << "\n";
  847. result->reg_ << " // Not registered because have @manualbind mark\n";
  848. return;
  849. }
  850. if (variable.IsStatic())
  851. {
  852. result->reg_ << " // " << variable.GetLocation() << "\n";
  853. bool outSuccess;
  854. string asType = CppTypeToAS(variable.GetType(), false, outSuccess);
  855. if (variable.GetType().IsConst())
  856. asType = "const " + asType;
  857. asType = ReplaceAll(asType, "struct ", "");
  858. if (!outSuccess)
  859. {
  860. result->reg_ << " // " << GetLastErrorMessage() << "\n";
  861. return;
  862. }
  863. string className = variable.GetClassName();
  864. result->reg_ <<
  865. " engine->SetDefaultNamespace(\"" << className << "\");\n";
  866. result->reg_ <<
  867. " engine->RegisterGlobalProperty("
  868. "\"" << asType << " " << variable.GetName() << "\", "
  869. "(void*)&" << className << "::" << variable.GetName() << ");\n";
  870. result->reg_ << " engine->SetDefaultNamespace(\"\");\n";
  871. }
  872. else
  873. {
  874. result->reg_ << " // " << variable.GetLocation() << "\n";
  875. if (variable.IsArray())
  876. {
  877. result->reg_ << " // Not registered because array\n";
  878. return;
  879. }
  880. if (variable.GetType().IsPointer())
  881. {
  882. result->reg_ << " // " << variable.GetType().ToString() << " can not be registered\n";
  883. return;
  884. }
  885. bool outSuccess;
  886. string propType = CppTypeToAS(variable.GetType(), false, outSuccess);
  887. if (!outSuccess)
  888. {
  889. result->reg_ << " // " << GetLastErrorMessage() << "\n";
  890. return;
  891. }
  892. string varName = variable.GetName();
  893. assert(!varName.empty());
  894. string propName = CutEnd(varName, "_");
  895. string className = variable.GetClassName();
  896. result->reg_ <<
  897. " engine->RegisterObjectProperty("
  898. "\"" << className << "\", "
  899. "\"" << propType << " " << propName << "\", "
  900. "offsetof(" << className << ", " << varName << "));\n";
  901. }
  902. }
  903. static void RegisterObjectMembers(const ClassAnalyzer& classAnalyzer, bool templateVersion)
  904. {
  905. string header = classAnalyzer.GetHeaderFile();
  906. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(classAnalyzer.GetClassName()).get();
  907. result->AddHeader(header);
  908. string insideDefine = InsideDefine(header);
  909. if (!insideDefine.empty())
  910. result->reg_ << "#ifdef " << insideDefine << "\n";
  911. vector<ClassVariableAnalyzer> variables = classAnalyzer.GetVariables();
  912. for (ClassVariableAnalyzer variable : variables)
  913. RegisterClassVarAsProperty(variable);
  914. vector<ClassFunctionAnalyzer> functions = classAnalyzer.GetFunctions();
  915. for (ClassFunctionAnalyzer function : functions)
  916. RegisterMethod(function, templateVersion);
  917. RegisterImplicitlyDeclaredMethods(classAnalyzer);
  918. // 2 operators is replaced by single function opCmp
  919. if (classAnalyzer.ContainsFunction("operator>") || classAnalyzer.ContainsFunction("operator>"))
  920. RegisterComparisonOperator(classAnalyzer);
  921. if (Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  922. RegisterFakeAddReleaseRef(classAnalyzer);
  923. vector<ClassAnalyzer> classList = classAnalyzer.GetAllBaseClasses();
  924. classList.push_back(classAnalyzer); // Also add current class to list
  925. string className = classAnalyzer.GetClassName();
  926. for (ClassAnalyzer& cl : classList)
  927. {
  928. string clName = cl.GetClassName();
  929. result->reg_ << "#ifdef REGISTER_MANUAL_PART_" << clName << "\n";
  930. if (templateVersion)
  931. result->reg_ << " REGISTER_MANUAL_PART_" << clName << "(T, className)\n";
  932. else
  933. result->reg_ << " REGISTER_MANUAL_PART_" << clName << "(" << className << ", \"" << className << "\")\n";
  934. result->reg_ << "#endif\n";
  935. }
  936. if (classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  937. {
  938. vector<ClassAnalyzer> baseClasses = classAnalyzer.GetAllBaseClasses();
  939. for (ClassAnalyzer baseClass : baseClasses)
  940. {
  941. if (baseClass.IsRefCounted() || Contains(baseClass.GetComment(), "FAKE_REF"))
  942. {
  943. string baseClassName = baseClass.GetClassName();
  944. if (templateVersion)
  945. result->reg_ << " RegisterSubclass<" << baseClassName << ", T>(engine, \"" << baseClassName << "\", className);\n";
  946. else
  947. result->reg_ << " RegisterSubclass<" << baseClassName << ", " << className << ">(engine, \"" << baseClassName << "\", \"" << className << "\");\n";
  948. }
  949. }
  950. // In template also add current class
  951. if (templateVersion)
  952. result->reg_ << " RegisterSubclass<" << className << ", T>(engine, \"" << className << "\", className);\n";
  953. }
  954. /*if (!analyzer.IsRefCounted()) // Executed also for FAKE_REF
  955. // TODO refcounted <=> value conversion if refcounted is chield value
  956. {
  957. vector<ClassAnalyzer> baseClasses = analyzer.GetAllBaseClasses();
  958. for (ClassAnalyzer baseClass : baseClasses)
  959. {
  960. if (!baseClass.IsRefCounted()) // Executed also for FAKE_REF
  961. {
  962. ASResult::reg_ << " RegisterSubclassValue<" << baseClass.GetClassName()
  963. << ", " << analyzer.GetClassName() << ">(engine, "
  964. << "\"" << baseClass.GetClassName() << "\", "
  965. << "\"" << analyzer.GetClassName() << "\");\n";
  966. }
  967. }
  968. }
  969. */
  970. if (!insideDefine.empty())
  971. result->reg_ << "#endif\n";
  972. if (!templateVersion)
  973. result->reg_ << "\n";
  974. }
  975. static void RegisterObjectType(const ClassAnalyzer& classAnalyzer, bool templateVersion)
  976. {
  977. string header = classAnalyzer.GetHeaderFile();
  978. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)_result_Classes.get();
  979. result->AddHeader(header);
  980. string insideDefine = InsideDefine(header);
  981. if (!insideDefine.empty())
  982. result->reg_ << "#ifdef " << insideDefine << "\n";
  983. result->reg_ << " // " + classAnalyzer.GetLocation() + "\n";
  984. string className = classAnalyzer.GetClassName();
  985. if (classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  986. {
  987. if (templateVersion)
  988. result->reg_ << " engine->RegisterObjectType(className, 0, asOBJ_REF);\n";
  989. else
  990. result->reg_ << " engine->RegisterObjectType(\"" << className << "\", 0, asOBJ_REF);\n";
  991. }
  992. else // Value type
  993. {
  994. // TODO templateversion
  995. string flags = "asOBJ_VALUE | asGetTypeTraits<" + className + ">()";
  996. if (classAnalyzer.IsPod())
  997. {
  998. flags += " | asOBJ_POD";
  999. if (classAnalyzer.AllFloats())
  1000. flags += " | asOBJ_APP_CLASS_ALLFLOATS";
  1001. else if (classAnalyzer.AllInts())
  1002. flags += " | asOBJ_APP_CLASS_ALLINTS";
  1003. }
  1004. result->reg_ << " engine->RegisterObjectType(\"" << className << "\", sizeof(" << className << "), " << flags << ");\n";
  1005. }
  1006. if (!insideDefine.empty())
  1007. result->reg_ << "#endif\n";
  1008. }
  1009. static void ProcessClass(const ClassAnalyzer& classAnalyzer, bool templateVersion)
  1010. {
  1011. if (classAnalyzer.IsInternal())
  1012. return;
  1013. if (classAnalyzer.IsAbstract() && !(classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF")))
  1014. return;
  1015. if (Contains(classAnalyzer.GetComment(), "NO_BIND"))
  1016. {
  1017. _result_Classes->reg_ << " // " << classAnalyzer.GetLocation() << "\n";
  1018. _result_Classes->reg_ << " // Not registered because have @nobind mark\n";
  1019. return;
  1020. }
  1021. if (Contains(classAnalyzer.GetComment(), "MANUAL_BIND"))
  1022. {
  1023. _result_Classes->reg_ << " // " << classAnalyzer.GetLocation() << "\n";
  1024. _result_Classes->reg_ << " // Not registered because have @manualbind mark\n";
  1025. return;
  1026. }
  1027. string header = classAnalyzer.GetHeaderFile();
  1028. if (IsIgnoredHeader(header))
  1029. {
  1030. _result_Classes->AddIgnoredHeader(header);
  1031. return;
  1032. }
  1033. if (classAnalyzer.IsTemplate())
  1034. return;
  1035. if (templateVersion)
  1036. _result_Templates->reg_ << "template <class T> void Register" << classAnalyzer.GetClassName() << "(asIScriptEngine* engine, const char* className)\n{\n";
  1037. RegisterObjectType(classAnalyzer, templateVersion);
  1038. RegisterObjectMembers(classAnalyzer, templateVersion);
  1039. if (templateVersion)
  1040. _result_Templates->reg_ << "}\n\n";
  1041. }
  1042. void ProcessAllClasses(const string& outputBasePath)
  1043. {
  1044. _outputBasePath = outputBasePath;
  1045. _result_Members_A = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_A.cpp", "ASRegisterGenerated_Members_A");
  1046. _result_Members_B = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_B.cpp", "ASRegisterGenerated_Members_B");
  1047. _result_Members_Constraint = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Constraint.cpp", "ASRegisterGenerated_Members_Constraint");
  1048. _result_Members_Ca_Cm = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Ca_Cm.cpp", "ASRegisterGenerated_Members_Ca_Cm");
  1049. _result_Members_Cn_Cz = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Cn_Cz.cpp", "ASRegisterGenerated_Members_Cn_Cz");
  1050. _result_Members_D = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_D.cpp", "ASRegisterGenerated_Members_D");
  1051. _result_Members_E = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_E.cpp", "ASRegisterGenerated_Members_E");
  1052. _result_Members_F = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_F.cpp", "ASRegisterGenerated_Members_F");
  1053. _result_Members_G = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_G.cpp", "ASRegisterGenerated_Members_G");
  1054. _result_Members_H = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_H.cpp", "ASRegisterGenerated_Members_H");
  1055. _result_Members_I = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_I.cpp", "ASRegisterGenerated_Members_I");
  1056. _result_Members_J = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_J.cpp", "ASRegisterGenerated_Members_J");
  1057. _result_Members_K = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_K.cpp", "ASRegisterGenerated_Members_K");
  1058. _result_Members_L = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_L.cpp", "ASRegisterGenerated_Members_L");
  1059. _result_Members_M = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_M.cpp", "ASRegisterGenerated_Members_M");
  1060. _result_Members_N = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_N.cpp", "ASRegisterGenerated_Members_N");
  1061. _result_Members_O = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_O.cpp", "ASRegisterGenerated_Members_O");
  1062. _result_Members_P = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_P.cpp", "ASRegisterGenerated_Members_P");
  1063. _result_Members_Q = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Q.cpp", "ASRegisterGenerated_Members_Q");
  1064. _result_Members_R = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_R.cpp", "ASRegisterGenerated_Members_R");
  1065. _result_Members_Sa_Sm = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Sa_Sm.cpp", "ASRegisterGenerated_Members_Sa_Sm");
  1066. _result_Members_Sn_Sz = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Sn_Sz.cpp", "ASRegisterGenerated_Members_Sn_Sz");
  1067. _result_Members_T = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_T.cpp", "ASRegisterGenerated_Members_T");
  1068. _result_Members_U = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_U.cpp", "ASRegisterGenerated_Members_U");
  1069. _result_Members_V = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_V.cpp", "ASRegisterGenerated_Members_V");
  1070. _result_Members_W = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_W.cpp", "ASRegisterGenerated_Members_W");
  1071. _result_Members_X = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_X.cpp", "ASRegisterGenerated_Members_X");
  1072. _result_Members_Y = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Y.cpp", "ASRegisterGenerated_Members_Y");
  1073. _result_Members_Z = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Z.cpp", "ASRegisterGenerated_Members_Z");
  1074. _result_Classes = make_shared<ASGeneratedFile_Classes>(outputBasePath + "/Source/Urho3D/AngelScript/Generated_Classes.cpp", "ASRegisterGenerated_Classes");
  1075. _result_Members_HighPriority = make_shared<ASGeneratedFile_Members_HighPriority>(outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_HighPriority.cpp", "ASRegisterGenerated_Members_HighPriority");
  1076. _result_Members_Other = make_shared<ASGeneratedFile_Members>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Members_Other.cpp", "ASRegisterGenerated_Members_Other");
  1077. _result_Templates = make_shared<ASGeneratedFile_Templates>(_outputBasePath + "/Source/Urho3D/AngelScript/Generated_Templates.h");
  1078. for (auto element : SourceData::classesByID_)
  1079. {
  1080. xml_node compounddef = element.second;
  1081. ClassAnalyzer analyzer(compounddef);
  1082. ProcessClass(analyzer, false);
  1083. // Generate also template version of bindings
  1084. if (Contains(analyzer.GetComment(), "TEMPLATE_VERSION"))
  1085. ProcessClass(analyzer, true);
  1086. }
  1087. _result_Classes->Save();
  1088. _result_Members_HighPriority->Save();
  1089. _result_Members_A->Save();
  1090. _result_Members_B->Save();
  1091. _result_Members_Constraint->Save();
  1092. _result_Members_Ca_Cm->Save();
  1093. _result_Members_Cn_Cz->Save();
  1094. _result_Members_D->Save();
  1095. _result_Members_E->Save();
  1096. _result_Members_F->Save();
  1097. _result_Members_G->Save();
  1098. _result_Members_H->Save();
  1099. _result_Members_I->Save();
  1100. _result_Members_J->Save();
  1101. _result_Members_K->Save();
  1102. _result_Members_L->Save();
  1103. _result_Members_M->Save();
  1104. _result_Members_N->Save();
  1105. _result_Members_O->Save();
  1106. _result_Members_P->Save();
  1107. _result_Members_Q->Save();
  1108. _result_Members_R->Save();
  1109. _result_Members_Sa_Sm->Save();
  1110. _result_Members_Sn_Sz->Save();
  1111. _result_Members_T->Save();
  1112. _result_Members_U->Save();
  1113. _result_Members_V->Save();
  1114. _result_Members_W->Save();
  1115. _result_Members_X->Save();
  1116. _result_Members_Y->Save();
  1117. _result_Members_Z->Save();
  1118. _result_Members_Other->Save();
  1119. _result_Templates->Save();
  1120. }
  1121. }