ASClassBinder.cpp 51 KB

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