tcoptimizations.pas 29 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145
  1. {
  2. This file is part of the Free Component Library (FCL)
  3. Copyright (c) 2017 by Michael Van Canneyt
  4. Unit tests for Pascal-to-Javascript converter class.
  5. See the file COPYING.FPC, included in this distribution,
  6. for details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************
  11. Examples:
  12. ./testpas2js --suite=TTestOptimizations
  13. ./testpas2js --suite=TTestOptimizations.TestOmitLocalVar
  14. }
  15. unit tcoptimizations;
  16. {$mode objfpc}{$H+}
  17. interface
  18. uses
  19. Classes, SysUtils, testregistry, fppas2js, pastree,
  20. PScanner, Pas2jsUseAnalyzer, PasResolver, PasResolveEval,
  21. tcmodules;
  22. type
  23. { TCustomTestOptimizations }
  24. TCustomTestOptimizations = class(TCustomTestModule)
  25. private
  26. FAnalyzerModule: TPas2JSAnalyzer;
  27. FAnalyzerProgram: TPas2JSAnalyzer;
  28. FWholeProgramOptimization: boolean;
  29. function OnConverterIsElementUsed(Sender: TObject; El: TPasElement): boolean;
  30. function OnConverterIsTypeInfoUsed(Sender: TObject; El: TPasElement): boolean;
  31. protected
  32. procedure SetUp; override;
  33. procedure TearDown; override;
  34. procedure ParseModule; override;
  35. procedure ParseProgram; override;
  36. function CreateConverter: TPasToJSConverter; override;
  37. public
  38. property AnalyzerModule: TPas2JSAnalyzer read FAnalyzerModule;
  39. property AnalyzerProgram: TPas2JSAnalyzer read FAnalyzerProgram;
  40. property WholeProgramOptimization: boolean read FWholeProgramOptimization
  41. write FWholeProgramOptimization;
  42. end;
  43. { TTestOptimizations }
  44. TTestOptimizations = class(TCustomTestOptimizations)
  45. published
  46. // Whole Program Optimization
  47. procedure TestWPO_OmitLocalVar;
  48. procedure TestWPO_OmitLocalProc;
  49. procedure TestWPO_OmitLocalProcForward;
  50. procedure TestWPO_OmitProcLocalVar;
  51. procedure TestWPO_OmitProcLocalConst;
  52. procedure TestWPO_OmitProcLocalType;
  53. procedure TestWPO_OmitProcLocalProc;
  54. procedure TestWPO_OmitProcLocalForwardProc;
  55. procedure TestWPO_OmitRecordMember;
  56. procedure TestWPO_OmitNotUsedTObject;
  57. procedure TestWPO_TObject;
  58. procedure TestWPO_Class_Property;
  59. procedure TestWPO_Class_OmitField;
  60. procedure TestWPO_Class_OmitMethod;
  61. procedure TestWPO_Class_OmitClassMethod;
  62. procedure TestWPO_Class_OmitPropertyGetter1;
  63. procedure TestWPO_Class_OmitPropertyGetter2;
  64. procedure TestWPO_Class_OmitPropertySetter1;
  65. procedure TestWPO_Class_OmitPropertySetter2;
  66. procedure TestWPO_Class_KeepNewInstance;
  67. procedure TestWPO_CallInherited;
  68. procedure TestWPO_UseUnit;
  69. procedure TestWPO_ArrayOfConst_Use;
  70. procedure TestWPO_ArrayOfConst_NotUsed;
  71. procedure TestWPO_Class_PropertyInOtherUnit;
  72. procedure TestWPO_ProgramPublicDeclaration;
  73. procedure TestWPO_ConstructorDefaultValueConst;
  74. procedure TestWPO_RTTI_PublishedField;
  75. procedure TestWPO_RTTI_TypeInfo;
  76. end;
  77. implementation
  78. { TCustomTestOptimizations }
  79. function TCustomTestOptimizations.OnConverterIsElementUsed(Sender: TObject;
  80. El: TPasElement): boolean;
  81. var
  82. A: TPas2JSAnalyzer;
  83. begin
  84. if WholeProgramOptimization then
  85. A:=AnalyzerProgram
  86. else if Sender=Converter then
  87. A:=AnalyzerModule
  88. else
  89. begin
  90. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  91. writeln('TCustomTestOptimizations.OnConverterIsElementUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Sender=',GetObjName(Sender));
  92. {$ENDIF}
  93. Fail('converting other unit without WPO');
  94. end;
  95. Result:=A.IsUsed(El);
  96. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  97. writeln('TCustomTestOptimizations.OnConverterIsElementUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Result=',Result);
  98. {$ENDIF}
  99. end;
  100. function TCustomTestOptimizations.OnConverterIsTypeInfoUsed(Sender: TObject;
  101. El: TPasElement): boolean;
  102. var
  103. A: TPas2JSAnalyzer;
  104. begin
  105. if WholeProgramOptimization then
  106. A:=AnalyzerProgram
  107. else if Sender=Converter then
  108. A:=AnalyzerModule
  109. else
  110. begin
  111. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  112. writeln('TCustomTestOptimizations.OnConverterIsTypeInfoUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Sender=',GetObjName(Sender));
  113. {$ENDIF}
  114. Fail('converting other unit without WPO');
  115. end;
  116. Result:=A.IsTypeInfoUsed(El);
  117. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  118. writeln('TCustomTestOptimizations.OnConverterIsTypeInfoUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Result=',Result);
  119. {$ENDIF}
  120. end;
  121. procedure TCustomTestOptimizations.SetUp;
  122. begin
  123. inherited SetUp;
  124. FWholeProgramOptimization:=false;
  125. FAnalyzerModule:=TPas2JSAnalyzer.Create;
  126. FAnalyzerModule.Resolver:=Engine;
  127. FAnalyzerProgram:=TPas2JSAnalyzer.Create;
  128. FAnalyzerProgram.Resolver:=Engine;
  129. end;
  130. procedure TCustomTestOptimizations.TearDown;
  131. begin
  132. FreeAndNil(FAnalyzerProgram);
  133. FreeAndNil(FAnalyzerModule);
  134. inherited TearDown;
  135. end;
  136. procedure TCustomTestOptimizations.ParseModule;
  137. begin
  138. inherited ParseModule;
  139. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  140. writeln('TCustomTestOptimizations.ParseModule START');
  141. {$ENDIF}
  142. AnalyzerModule.AnalyzeModule(Module);
  143. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  144. writeln('TCustomTestOptimizations.ParseModule END');
  145. {$ENDIF}
  146. end;
  147. procedure TCustomTestOptimizations.ParseProgram;
  148. begin
  149. WholeProgramOptimization:=true;
  150. inherited ParseProgram;
  151. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  152. writeln('TCustomTestOptimizations.ParseProgram START');
  153. {$ENDIF}
  154. AnalyzerProgram.AnalyzeWholeProgram(Module as TPasProgram);
  155. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  156. writeln('TCustomTestOptimizations.ParseProgram START');
  157. {$ENDIF}
  158. end;
  159. function TCustomTestOptimizations.CreateConverter: TPasToJSConverter;
  160. begin
  161. Result:=inherited CreateConverter;
  162. Result.OnIsElementUsed:=@OnConverterIsElementUsed;
  163. Result.OnIsTypeInfoUsed:=@OnConverterIsTypeInfoUsed;
  164. end;
  165. { TTestOptimizations }
  166. procedure TTestOptimizations.TestWPO_OmitLocalVar;
  167. begin
  168. StartProgram(false);
  169. Add('var');
  170. Add(' a: longint;');
  171. Add(' b: longint;');
  172. Add('begin');
  173. Add(' b:=3;');
  174. ConvertProgram;
  175. CheckSource('TestWPO_OmitLocalVar',
  176. 'this.b = 0;',
  177. '$mod.b = 3;');
  178. end;
  179. procedure TTestOptimizations.TestWPO_OmitLocalProc;
  180. begin
  181. StartProgram(false);
  182. Add('procedure DoIt; begin end;');
  183. Add('procedure NoIt; begin end;');
  184. Add('begin');
  185. Add(' DoIt;');
  186. ConvertProgram;
  187. CheckSource('TestWPO_OmitLocalProc',
  188. LinesToStr([
  189. 'this.DoIt = function () {',
  190. '};',
  191. '']),
  192. LinesToStr([
  193. '$mod.DoIt();',
  194. '']));
  195. end;
  196. procedure TTestOptimizations.TestWPO_OmitLocalProcForward;
  197. begin
  198. StartProgram(false);
  199. Add('procedure DoIt; forward;');
  200. Add('procedure NoIt; forward;');
  201. Add('procedure DoIt; begin end;');
  202. Add('procedure NoIt; begin end;');
  203. Add('begin');
  204. Add(' DoIt;');
  205. ConvertProgram;
  206. CheckSource('TestWPO_OmitLocalProcForward',
  207. LinesToStr([
  208. 'this.DoIt = function () {',
  209. '};',
  210. '']),
  211. LinesToStr([
  212. '$mod.DoIt();',
  213. '']));
  214. end;
  215. procedure TTestOptimizations.TestWPO_OmitProcLocalVar;
  216. begin
  217. StartProgram(false);
  218. Add('function DoIt: longint;');
  219. Add('var');
  220. Add(' a: longint;');
  221. Add(' b: longint;');
  222. Add('begin');
  223. Add(' b:=3;');
  224. Add(' Result:=b;');
  225. Add('end;');
  226. Add('begin');
  227. Add(' DoIt;');
  228. ConvertProgram;
  229. CheckSource('TestWPO_OmitProcLocalVar',
  230. LinesToStr([
  231. 'this.DoIt = function () {',
  232. ' var Result = 0;',
  233. ' var b = 0;',
  234. ' b = 3;',
  235. ' Result = b;',
  236. ' return Result;',
  237. '};',
  238. '']),
  239. LinesToStr([
  240. '$mod.DoIt();',
  241. '']));
  242. end;
  243. procedure TTestOptimizations.TestWPO_OmitProcLocalConst;
  244. begin
  245. StartProgram(false);
  246. Add('function DoIt: longint;');
  247. Add('const');
  248. Add(' a = 3;');
  249. Add(' b = 4;');
  250. Add(' c: longint = 5;');
  251. Add(' d: longint = 6;');
  252. Add('begin');
  253. Add(' Result:=b+d;');
  254. Add('end;');
  255. Add('begin');
  256. Add(' DoIt;');
  257. ConvertProgram;
  258. CheckSource('TestWPO_OmitProcLocalConst',
  259. LinesToStr([
  260. 'var b = 4;',
  261. 'var d = 6;',
  262. 'this.DoIt = function () {',
  263. ' var Result = 0;',
  264. ' Result = 4 + d;',
  265. ' return Result;',
  266. '};',
  267. '']),
  268. LinesToStr([
  269. '$mod.DoIt();',
  270. '']));
  271. end;
  272. procedure TTestOptimizations.TestWPO_OmitProcLocalType;
  273. begin
  274. StartProgram(false);
  275. Add('function DoIt: longint;');
  276. Add('type');
  277. Add(' TEnum = (red, green);');
  278. Add(' TEnums = set of TEnum;');
  279. Add('begin');
  280. Add(' Result:=3;');
  281. Add('end;');
  282. Add('begin');
  283. Add(' DoIt;');
  284. ConvertProgram;
  285. CheckSource('TestWPO_OmitProcLocalType',
  286. LinesToStr([
  287. 'this.DoIt = function () {',
  288. ' var Result = 0;',
  289. ' Result = 3;',
  290. ' return Result;',
  291. '};',
  292. '']),
  293. LinesToStr([
  294. '$mod.DoIt();',
  295. '']));
  296. end;
  297. procedure TTestOptimizations.TestWPO_OmitProcLocalProc;
  298. begin
  299. StartProgram(false);
  300. Add('procedure DoIt;');
  301. Add(' procedure SubProcA; begin end;');
  302. Add(' procedure SubProcB; begin end;');
  303. Add('begin');
  304. Add(' SubProcB;');
  305. Add('end;');
  306. Add('begin');
  307. Add(' DoIt;');
  308. ConvertProgram;
  309. CheckSource('TestWPO_OmitProcLocalProc',
  310. LinesToStr([
  311. 'this.DoIt = function () {',
  312. ' function SubProcB() {',
  313. ' };',
  314. ' SubProcB();',
  315. '};',
  316. '']),
  317. LinesToStr([
  318. '$mod.DoIt();',
  319. '']));
  320. end;
  321. procedure TTestOptimizations.TestWPO_OmitProcLocalForwardProc;
  322. begin
  323. StartProgram(false);
  324. Add('procedure DoIt;');
  325. Add(' procedure SubProcA; forward;');
  326. Add(' procedure SubProcB; forward;');
  327. Add(' procedure SubProcA; begin end;');
  328. Add(' procedure SubProcB; begin end;');
  329. Add('begin');
  330. Add(' SubProcB;');
  331. Add('end;');
  332. Add('begin');
  333. Add(' DoIt;');
  334. ConvertProgram;
  335. CheckSource('TestWPO_OmitProcLocalForwardProc',
  336. LinesToStr([
  337. 'this.DoIt = function () {',
  338. ' function SubProcB() {',
  339. ' };',
  340. ' SubProcB();',
  341. '};',
  342. '']),
  343. LinesToStr([
  344. '$mod.DoIt();',
  345. '']));
  346. end;
  347. procedure TTestOptimizations.TestWPO_OmitRecordMember;
  348. begin
  349. StartProgram(false);
  350. Add('type');
  351. Add(' TRec = record');
  352. Add(' a: longint;');
  353. Add(' b: longint;');
  354. Add(' end;');
  355. Add('var r: TRec;');
  356. Add('begin');
  357. Add(' r.a:=3;');
  358. ConvertProgram;
  359. CheckSource('TestWPO_OmitRecordMember',
  360. LinesToStr([
  361. 'rtl.recNewT($mod, "TRec", function () {',
  362. ' this.a = 0;',
  363. ' this.$eq = function (b) {',
  364. ' return this.a === b.a;',
  365. ' };',
  366. ' this.$assign = function (s) {',
  367. ' this.a = s.a;',
  368. ' return this;',
  369. ' };',
  370. '});',
  371. 'this.r = $mod.TRec.$new();',
  372. '']),
  373. LinesToStr([
  374. '$mod.r.a = 3;',
  375. '']));
  376. end;
  377. procedure TTestOptimizations.TestWPO_OmitNotUsedTObject;
  378. begin
  379. StartProgram(false);
  380. Add('type');
  381. Add(' TObject = class end;');
  382. Add('var o: TObject;');
  383. Add('begin');
  384. ConvertProgram;
  385. CheckSource('TestWPO_OmitNotUsedTObject',
  386. LinesToStr([
  387. '']),
  388. LinesToStr([
  389. '']));
  390. end;
  391. procedure TTestOptimizations.TestWPO_TObject;
  392. begin
  393. StartProgram(false);
  394. Add('type');
  395. Add(' TObject = class');
  396. Add(' procedure AfterConstruction; virtual;');
  397. Add(' procedure BeforeDestruction; virtual;');
  398. Add(' end;');
  399. Add('procedure TObject.AfterConstruction; begin end;');
  400. Add('procedure TObject.BeforeDestruction; begin end;');
  401. Add('var o: TObject;');
  402. Add('begin');
  403. Add(' o:=nil;');
  404. ConvertProgram;
  405. CheckSource('TestWPO_TObject',
  406. LinesToStr([
  407. 'rtl.createClass($mod, "TObject", null, function () {',
  408. ' this.$init = function () {',
  409. ' };',
  410. ' this.$final = function () {',
  411. ' };',
  412. ' this.AfterConstruction = function () {',
  413. ' };',
  414. ' this.BeforeDestruction = function () {',
  415. ' };',
  416. '});',
  417. 'this.o = null;',
  418. '']),
  419. LinesToStr([
  420. '$mod.o = null;']));
  421. end;
  422. procedure TTestOptimizations.TestWPO_Class_Property;
  423. begin
  424. StartProgram(false);
  425. Add([
  426. 'type',
  427. ' TObject = class',
  428. ' private',
  429. ' const CA = 3;',
  430. ' private',
  431. ' FA: longint;',
  432. ' function GetA: longint;',
  433. ' procedure SetA(Value: longint);',
  434. ' function IsStoredA: boolean;',
  435. ' property A: longint read GetA write SetA stored IsStoredA default CA;',
  436. ' end;',
  437. 'function tobject.geta: longint; begin end;',
  438. 'procedure tobject.seta(value: longint); begin end;',
  439. 'function tobject.isstoreda: boolean; begin end;',
  440. 'var o: TObject;',
  441. 'begin',
  442. ' o.A:=o.A;']);
  443. ConvertProgram;
  444. CheckSource('TestWPO_Class_TObject',
  445. LinesToStr([
  446. 'rtl.createClass($mod, "TObject", null, function () {',
  447. ' this.$init = function () {',
  448. ' };',
  449. ' this.$final = function () {',
  450. ' };',
  451. ' this.GetA = function () {',
  452. ' var Result = 0;',
  453. ' return Result;',
  454. ' };',
  455. ' this.SetA = function (Value) {',
  456. ' };',
  457. '});',
  458. 'this.o = null;',
  459. '']),
  460. LinesToStr([
  461. '$mod.o.SetA($mod.o.GetA());']));
  462. end;
  463. procedure TTestOptimizations.TestWPO_Class_OmitField;
  464. begin
  465. StartProgram(false);
  466. Add('type');
  467. Add(' TObject = class');
  468. Add(' a: longint;');
  469. Add(' b: longint;');
  470. Add(' end;');
  471. Add('var o: TObject;');
  472. Add('begin');
  473. Add(' o.a:=3;');
  474. ConvertProgram;
  475. CheckSource('TestWPO_OmitClassField',
  476. LinesToStr([
  477. 'rtl.createClass($mod, "TObject", null, function () {',
  478. ' this.$init = function () {',
  479. ' this.a = 0;',
  480. ' };',
  481. ' this.$final = function () {',
  482. ' };',
  483. '});',
  484. 'this.o = null;',
  485. '']),
  486. LinesToStr([
  487. '$mod.o.a = 3;']));
  488. end;
  489. procedure TTestOptimizations.TestWPO_Class_OmitMethod;
  490. begin
  491. StartProgram(false);
  492. Add('type');
  493. Add(' TObject = class');
  494. Add(' procedure ProcA;');
  495. Add(' procedure ProcB;');
  496. Add(' end;');
  497. Add('procedure TObject.ProcA; begin end;');
  498. Add('procedure TObject.ProcB; begin end;');
  499. Add('var o: TObject;');
  500. Add('begin');
  501. Add(' o.ProcB;');
  502. ConvertProgram;
  503. CheckSource('TestWPO_OmitClassMethod',
  504. LinesToStr([
  505. 'rtl.createClass($mod, "TObject", null, function () {',
  506. ' this.$init = function () {',
  507. ' };',
  508. ' this.$final = function () {',
  509. ' };',
  510. ' this.ProcB = function () {',
  511. ' };',
  512. '});',
  513. 'this.o = null;',
  514. '']),
  515. LinesToStr([
  516. '$mod.o.ProcB();']));
  517. end;
  518. procedure TTestOptimizations.TestWPO_Class_OmitClassMethod;
  519. begin
  520. StartProgram(false);
  521. Add('type');
  522. Add(' TObject = class');
  523. Add(' class procedure ProcA;');
  524. Add(' class procedure ProcB;');
  525. Add(' end;');
  526. Add('class procedure TObject.ProcA; begin end;');
  527. Add('class procedure TObject.ProcB; begin end;');
  528. Add('var o: TObject;');
  529. Add('begin');
  530. Add(' o.ProcB;');
  531. ConvertProgram;
  532. CheckSource('TestWPO_OmitClassMethod',
  533. LinesToStr([
  534. 'rtl.createClass($mod, "TObject", null, function () {',
  535. ' this.$init = function () {',
  536. ' };',
  537. ' this.$final = function () {',
  538. ' };',
  539. ' this.ProcB = function () {',
  540. ' };',
  541. '});',
  542. 'this.o = null;',
  543. '']),
  544. LinesToStr([
  545. '$mod.o.$class.ProcB();']));
  546. end;
  547. procedure TTestOptimizations.TestWPO_Class_OmitPropertyGetter1;
  548. begin
  549. StartProgram(false);
  550. Add('type');
  551. Add(' TObject = class');
  552. Add(' FFoo: boolean;');
  553. Add(' function GetFoo: boolean;');
  554. Add(' property Foo: boolean read FFoo;');
  555. Add(' property Foo2: boolean read GetFoo;');
  556. Add(' FBar: boolean;');
  557. Add(' function GetBar: boolean;');
  558. Add(' property Bar: boolean read FBar;');
  559. Add(' property Bar2: boolean read GetBar;');
  560. Add(' end;');
  561. Add('function TObject.GetFoo: boolean; begin Result:=FFoo; end;');
  562. Add('function TObject.GetBar: boolean; begin Result:=FBar; end;');
  563. Add('var o: TObject;');
  564. Add('begin');
  565. Add(' if o.Foo then;');
  566. ConvertProgram;
  567. CheckSource('TestWPO_OmitClassPropertyGetter1',
  568. LinesToStr([
  569. 'rtl.createClass($mod, "TObject", null, function () {',
  570. ' this.$init = function () {',
  571. ' this.FFoo = false;',
  572. ' };',
  573. ' this.$final = function () {',
  574. ' };',
  575. '});',
  576. 'this.o = null;',
  577. '']),
  578. LinesToStr([
  579. 'if ($mod.o.FFoo);',
  580. '']));
  581. end;
  582. procedure TTestOptimizations.TestWPO_Class_OmitPropertyGetter2;
  583. begin
  584. StartProgram(false);
  585. Add('type');
  586. Add(' TObject = class');
  587. Add(' FFoo: boolean;');
  588. Add(' function GetFoo: boolean;');
  589. Add(' property Foo: boolean read FFoo;');
  590. Add(' property Foo2: boolean read GetFoo;');
  591. Add(' end;');
  592. Add('function TObject.GetFoo: boolean; begin Result:=FFoo; end;');
  593. Add('var o: TObject;');
  594. Add('begin');
  595. Add(' if o.Foo2 then;');
  596. ConvertProgram;
  597. CheckSource('TestWPO_OmitClassPropertyGetter2',
  598. LinesToStr([
  599. 'rtl.createClass($mod, "TObject", null, function () {',
  600. ' this.$init = function () {',
  601. ' this.FFoo = false;',
  602. ' };',
  603. ' this.$final = function () {',
  604. ' };',
  605. ' this.GetFoo = function () {',
  606. ' var Result = false;',
  607. ' Result = this.FFoo;',
  608. ' return Result;',
  609. ' };',
  610. '});',
  611. 'this.o = null;',
  612. '']),
  613. LinesToStr([
  614. 'if ($mod.o.GetFoo()) ;',
  615. '']));
  616. end;
  617. procedure TTestOptimizations.TestWPO_Class_OmitPropertySetter1;
  618. begin
  619. StartProgram(false);
  620. Add('type');
  621. Add(' TObject = class');
  622. Add(' FFoo: boolean;');
  623. Add(' procedure SetFoo(Value: boolean);');
  624. Add(' property Foo: boolean write FFoo;');
  625. Add(' property Foo2: boolean write SetFoo;');
  626. Add(' FBar: boolean;');
  627. Add(' procedure SetBar(Value: boolean);');
  628. Add(' property Bar: boolean write FBar;');
  629. Add(' property Bar2: boolean write SetBar;');
  630. Add(' end;');
  631. Add('procedure TObject.SetFoo(Value: boolean); begin FFoo:=Value; end;');
  632. Add('procedure TObject.SetBar(Value: boolean); begin FBar:=Value; end;');
  633. Add('var o: TObject;');
  634. Add('begin');
  635. Add(' o.Foo:=true;');
  636. ConvertProgram;
  637. CheckSource('TestWPO_OmitClassPropertySetter1',
  638. LinesToStr([
  639. 'rtl.createClass($mod, "TObject", null, function () {',
  640. ' this.$init = function () {',
  641. ' this.FFoo = false;',
  642. ' };',
  643. ' this.$final = function () {',
  644. ' };',
  645. '});',
  646. 'this.o = null;',
  647. '']),
  648. LinesToStr([
  649. '$mod.o.FFoo = true;',
  650. '']));
  651. end;
  652. procedure TTestOptimizations.TestWPO_Class_OmitPropertySetter2;
  653. begin
  654. StartProgram(false);
  655. Add('type');
  656. Add(' TObject = class');
  657. Add(' FFoo: boolean;');
  658. Add(' procedure SetFoo(Value: boolean);');
  659. Add(' property Foo: boolean write FFoo;');
  660. Add(' property Foo2: boolean write SetFoo;');
  661. Add(' end;');
  662. Add('procedure TObject.SetFoo(Value: boolean); begin FFoo:=Value; end;');
  663. Add('var o: TObject;');
  664. Add('begin');
  665. Add(' o.Foo2:=true;');
  666. ConvertProgram;
  667. CheckSource('TestWPO_OmitClassPropertySetter2',
  668. LinesToStr([
  669. 'rtl.createClass($mod, "TObject", null, function () {',
  670. ' this.$init = function () {',
  671. ' this.FFoo = false;',
  672. ' };',
  673. ' this.$final = function () {',
  674. ' };',
  675. ' this.SetFoo = function (Value) {',
  676. ' this.FFoo = Value;',
  677. ' };',
  678. '});',
  679. 'this.o = null;',
  680. '']),
  681. LinesToStr([
  682. '$mod.o.SetFoo(true);',
  683. '']));
  684. end;
  685. procedure TTestOptimizations.TestWPO_Class_KeepNewInstance;
  686. begin
  687. StartProgram(false);
  688. Add([
  689. '{$modeswitch externalclass}',
  690. 'type',
  691. ' TExt = class external name ''Object''',
  692. ' end;',
  693. ' TBird = class(TExt)',
  694. ' protected',
  695. ' class function NewInstance(fnname: string; const paramarray): TBird; virtual;',
  696. ' public',
  697. ' constructor Create;',
  698. ' end;',
  699. 'class function TBird.NewInstance(fnname: string; const paramarray): TBird;',
  700. 'begin',
  701. ' asm',
  702. ' Result = Object.create();',
  703. ' end;',
  704. 'end;',
  705. 'constructor TBird.Create;',
  706. 'begin',
  707. ' inherited;',
  708. 'end;',
  709. 'begin',
  710. ' TBird.Create;',
  711. '']);
  712. ConvertProgram;
  713. CheckSource('TestWPO_Class_KeepNewInstance',
  714. LinesToStr([
  715. 'rtl.createClassExt($mod, "TBird", Object, "NewInstance", function () {',
  716. ' this.$init = function () {',
  717. ' };',
  718. ' this.$final = function () {',
  719. ' };',
  720. ' this.NewInstance = function (fnname, paramarray) {',
  721. ' var Result = null;',
  722. ' Result = Object.create();',
  723. ' return Result;',
  724. ' };',
  725. ' this.Create = function () {',
  726. ' return this;',
  727. ' };',
  728. '});',
  729. '']),
  730. LinesToStr([
  731. '$mod.TBird.$create("Create");',
  732. '']));
  733. end;
  734. procedure TTestOptimizations.TestWPO_CallInherited;
  735. begin
  736. StartProgram(false);
  737. Add('type');
  738. Add(' TObject = class');
  739. Add(' procedure DoA;');
  740. Add(' procedure DoB;');
  741. Add(' end;');
  742. Add(' TMobile = class');
  743. Add(' procedure DoA;');
  744. Add(' procedure DoC;');
  745. Add(' end;');
  746. Add('procedure TObject.DoA; begin end;');
  747. Add('procedure TObject.DoB; begin end;');
  748. Add('procedure TMobile.DoA;');
  749. Add('begin');
  750. Add(' inherited;');
  751. Add('end;');
  752. Add('procedure TMobile.DoC;');
  753. Add('begin');
  754. Add(' inherited DoB;');
  755. Add('end;');
  756. Add('var o: TMobile;');
  757. Add('begin');
  758. Add(' o.DoA;');
  759. Add(' o.DoC;');
  760. ConvertProgram;
  761. CheckSource('TestWPO_CallInherited',
  762. LinesToStr([
  763. 'rtl.createClass($mod, "TObject", null, function () {',
  764. ' this.$init = function () {',
  765. ' };',
  766. ' this.$final = function () {',
  767. ' };',
  768. ' this.DoA = function () {',
  769. ' };',
  770. ' this.DoB = function () {',
  771. ' };',
  772. '});',
  773. ' rtl.createClass($mod, "TMobile", $mod.TObject, function () {',
  774. ' this.DoA$1 = function () {',
  775. ' $mod.TObject.DoA.call(this);',
  776. ' };',
  777. ' this.DoC = function () {',
  778. ' $mod.TObject.DoB.call(this);',
  779. ' };',
  780. '});',
  781. 'this.o = null;',
  782. '']),
  783. LinesToStr([
  784. '$mod.o.DoA$1();',
  785. '$mod.o.DoC();',
  786. '']));
  787. end;
  788. procedure TTestOptimizations.TestWPO_UseUnit;
  789. var
  790. ActualSrc, ExpectedSrc: String;
  791. begin
  792. AddModuleWithIntfImplSrc('unit1.pp',
  793. LinesToStr([
  794. 'var i: longint;',
  795. 'procedure DoIt;',
  796. '']),
  797. LinesToStr([
  798. 'procedure DoIt; begin end;']));
  799. AddModuleWithIntfImplSrc('unit2.pp',
  800. LinesToStr([
  801. 'var j: longint;',
  802. 'procedure DoMore;',
  803. '']),
  804. LinesToStr([
  805. 'procedure DoMore; begin end;']));
  806. StartProgram(true);
  807. Add('uses unit2;');
  808. Add('begin');
  809. Add(' j:=3;');
  810. ConvertProgram;
  811. ActualSrc:=ConvertJSModuleToString(JSModule);
  812. ExpectedSrc:=LinesToStr([
  813. 'rtl.module("program", ["system", "unit2"], function () {',
  814. ' var $mod = this;',
  815. ' $mod.$main = function () {',
  816. ' pas.unit2.j = 3;',
  817. ' };',
  818. '});',
  819. '']);
  820. CheckDiff('TestWPO_UseUnit',ExpectedSrc,ActualSrc);
  821. end;
  822. procedure TTestOptimizations.TestWPO_ArrayOfConst_Use;
  823. begin
  824. StartProgram(true,[supTVarRec]);
  825. Add([
  826. 'procedure Say(arr: array of const);',
  827. 'begin',
  828. 'end;',
  829. 'begin',
  830. ' Say([true]);']);
  831. ConvertProgram;
  832. CheckUnit('system.pp',
  833. LinesToStr([
  834. 'rtl.module("system", [], function () {',
  835. ' var $mod = this;',
  836. ' rtl.recNewT($mod, "TVarRec", function () {',
  837. ' this.VType = 0;',
  838. ' this.VJSValue = undefined;',
  839. ' this.$eq = function (b) {',
  840. ' return (this.VType === b.VType) && (this.VJSValue === b.VJSValue);',
  841. ' };',
  842. ' this.$assign = function (s) {',
  843. ' this.VType = s.VType;',
  844. ' this.VJSValue = s.VJSValue;',
  845. ' return this;',
  846. ' };',
  847. ' });',
  848. ' this.VarRecs = function () {',
  849. ' var Result = [];',
  850. ' var v = null;',
  851. ' v.VType = 1;',
  852. ' v.VJSValue = 2;',
  853. ' return Result;',
  854. ' };',
  855. '});',
  856. '']));
  857. end;
  858. procedure TTestOptimizations.TestWPO_ArrayOfConst_NotUsed;
  859. begin
  860. StartProgram(true,[supTVarRec]);
  861. Add([
  862. 'procedure Say(arr: array of const);',
  863. 'begin',
  864. 'end;',
  865. 'begin']);
  866. ConvertProgram;
  867. CheckUnit('system.pp',
  868. LinesToStr([
  869. 'rtl.module("system", [], function () {',
  870. ' var $mod = this;',
  871. '});',
  872. '']));
  873. end;
  874. procedure TTestOptimizations.TestWPO_Class_PropertyInOtherUnit;
  875. begin
  876. AddModuleWithIntfImplSrc('unit1.pp',
  877. LinesToStr([
  878. 'type',
  879. ' TObject = class',
  880. ' private',
  881. ' const CA = 3;',
  882. ' private',
  883. ' FOther: string;',
  884. ' FA: longint;',
  885. ' function GetA: longint;',
  886. ' procedure SetA(Value: longint);',
  887. ' function IsStoredA: boolean;',
  888. ' public',
  889. ' property A: longint read GetA write SetA stored IsStoredA default CA;',
  890. ' end;',
  891. '']),
  892. LinesToStr([
  893. 'function TObject.geta: longint;',
  894. 'begin',
  895. 'end;',
  896. 'procedure TObject.seta(value: longint);',
  897. 'begin',
  898. ' FA:=Value;',
  899. 'end;',
  900. 'function TObject.isstoreda: boolean; begin end;',
  901. '']));
  902. StartProgram(true);
  903. Add([
  904. 'uses unit1;',
  905. 'var o: TObject;',
  906. 'begin',
  907. ' o.A:=o.A;']);
  908. ConvertProgram;
  909. CheckUnit('unit1.pp',
  910. LinesToStr([
  911. 'rtl.module("unit1", ["system"], function () {',
  912. ' var $mod = this;',
  913. ' rtl.createClass($mod, "TObject", null, function () {',
  914. ' this.$init = function () {',
  915. ' this.FA = 0;',
  916. ' };',
  917. ' this.$final = function () {',
  918. ' };',
  919. ' this.GetA = function () {',
  920. ' var Result = 0;',
  921. ' return Result;',
  922. ' };',
  923. ' this.SetA = function (Value) {',
  924. ' this.FA = Value;',
  925. ' };',
  926. ' });',
  927. '});',
  928. '']));
  929. end;
  930. procedure TTestOptimizations.TestWPO_ProgramPublicDeclaration;
  931. var
  932. ActualSrc, ExpectedSrc: String;
  933. begin
  934. StartProgram(true);
  935. Add('var');
  936. Add(' vPublic: longint; public;');
  937. Add(' vPrivate: longint;');
  938. Add('procedure DoPublic; public; begin end;');
  939. Add('procedure DoPrivate; begin end;');
  940. Add('begin');
  941. ConvertProgram;
  942. ActualSrc:=ConvertJSModuleToString(JSModule);
  943. ExpectedSrc:=LinesToStr([
  944. 'rtl.module("program", ["system"], function () {',
  945. ' var $mod = this;',
  946. ' this.vPublic = 0;',
  947. ' this.DoPublic =function(){',
  948. ' };',
  949. ' $mod.$main = function () {',
  950. ' };',
  951. '});',
  952. '']);
  953. CheckDiff('TestWPO_ProgramPublicDeclaration',ExpectedSrc,ActualSrc);
  954. end;
  955. procedure TTestOptimizations.TestWPO_ConstructorDefaultValueConst;
  956. var
  957. ActualSrc, ExpectedSrc: String;
  958. begin
  959. Converter.Options:=Converter.Options-[coNoTypeInfo];
  960. StartProgram(true);
  961. Add([
  962. 'const gcBlack = 0;',
  963. 'type',
  964. ' TColor = longint;',
  965. ' TObject = class',
  966. ' private',
  967. ' FColor: TColor;',
  968. ' public',
  969. ' property Color: TColor read FColor write FColor;',
  970. ' constructor Create(const AColor: TColor = gcBlack);',
  971. ' end;',
  972. 'constructor TObject.Create(const AColor: TColor = gcBlack);',
  973. 'begin',
  974. ' FColor := AColor;',
  975. 'end;',
  976. 'var T: TObject;',
  977. 'begin',
  978. ' T := TObject.Create;',
  979. '']);
  980. ConvertProgram;
  981. ActualSrc:=ConvertJSModuleToString(JSModule);
  982. ExpectedSrc:=LinesToStr([
  983. 'rtl.module("program",["system"],function () {',
  984. ' var $mod = this;',
  985. ' this.gcBlack = 0;',
  986. ' rtl.createClass($mod,"TObject",null,function () {',
  987. ' this.$init = function () {',
  988. ' this.FColor = 0;',
  989. ' };',
  990. ' this.$final = function () {',
  991. ' };',
  992. ' this.Create = function (AColor) {',
  993. ' this.FColor = AColor;',
  994. ' return this;',
  995. ' };',
  996. ' });',
  997. ' this.T = null;',
  998. ' $mod.$main = function () {',
  999. ' $mod.T = $mod.TObject.$create("Create",[0]);',
  1000. ' };',
  1001. '});',
  1002. '']);
  1003. CheckDiff('TestWPO_ConstructorDefaultValueConst',ExpectedSrc,ActualSrc);
  1004. end;
  1005. procedure TTestOptimizations.TestWPO_RTTI_PublishedField;
  1006. var
  1007. ActualSrc, ExpectedSrc: String;
  1008. begin
  1009. Converter.Options:=Converter.Options-[coNoTypeInfo];
  1010. StartProgram(true);
  1011. Add('type');
  1012. Add(' TArrA = array of char;');
  1013. Add(' TArrB = array of string;');
  1014. Add(' TObject = class');
  1015. Add(' public');
  1016. Add(' PublicA: TArrA;');
  1017. Add(' published');
  1018. Add(' PublishedB: TArrB;');
  1019. Add(' end;');
  1020. Add('var');
  1021. Add(' C: TObject;');
  1022. Add('begin');
  1023. Add(' C.PublicA:=nil;');
  1024. ConvertProgram;
  1025. ActualSrc:=ConvertJSModuleToString(JSModule);
  1026. ExpectedSrc:=LinesToStr([
  1027. 'rtl.module("program", ["system"], function () {',
  1028. ' var $mod = this;',
  1029. ' $mod.$rtti.$DynArray("TArrB", {',
  1030. ' eltype: rtl.string',
  1031. ' });',
  1032. ' rtl.createClass($mod, "TObject", null, function () {',
  1033. ' this.$init = function () {',
  1034. ' this.PublicA = [];',
  1035. ' this.PublishedB = [];',
  1036. ' };',
  1037. ' this.$final = function () {',
  1038. ' this.PublicA = undefined;',
  1039. ' this.PublishedB = undefined;',
  1040. ' };',
  1041. ' var $r = this.$rtti;',
  1042. ' $r.addField("PublishedB", $mod.$rtti["TArrB"]);',
  1043. ' });',
  1044. ' this.C = null;',
  1045. ' $mod.$main = function () {',
  1046. ' $mod.C.PublicA = [];',
  1047. ' };',
  1048. '});',
  1049. '']);
  1050. CheckDiff('TestWPO_RTTI_PublishedField',ExpectedSrc,ActualSrc);
  1051. end;
  1052. procedure TTestOptimizations.TestWPO_RTTI_TypeInfo;
  1053. var
  1054. ActualSrc, ExpectedSrc: String;
  1055. begin
  1056. Converter.Options:=Converter.Options-[coNoTypeInfo];
  1057. StartProgram(true);
  1058. Add('type');
  1059. Add(' TArrA = array of char;');
  1060. Add(' TArrB = array of string;');
  1061. Add('var');
  1062. Add(' A: TArrA;');
  1063. Add(' B: TArrB;');
  1064. Add(' p: pointer;');
  1065. Add('begin');
  1066. Add(' A:=nil;');
  1067. Add(' p:=typeinfo(B);');
  1068. ConvertProgram;
  1069. ActualSrc:=ConvertJSModuleToString(JSModule);
  1070. ExpectedSrc:=LinesToStr([
  1071. 'rtl.module("program", ["system"], function () {',
  1072. ' var $mod = this;',
  1073. ' $mod.$rtti.$DynArray("TArrB", {',
  1074. ' eltype: rtl.string',
  1075. ' });',
  1076. ' this.A = [];',
  1077. ' this.B = [];',
  1078. ' this.p = null;',
  1079. ' $mod.$main = function () {',
  1080. ' $mod.A = [];',
  1081. ' $mod.p = $mod.$rtti["TArrB"];',
  1082. ' };',
  1083. '});',
  1084. '']);
  1085. CheckDiff('TestWPO_RTTI_TypeInfo',ExpectedSrc,ActualSrc);
  1086. end;
  1087. Initialization
  1088. RegisterTests([TTestOptimizations]);
  1089. end.