tcoptimizations.pas 43 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735
  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. // unit optimization: jsshortrefglobals
  47. procedure TestOptShortRefGlobals_Program;
  48. procedure TestOptShortRefGlobals_Unit_FromIntfImpl_ToIntfImpl;
  49. procedure TestOptShortRefGlobals_Property;
  50. procedure TestOptShortRefGlobals_GenericFunction;
  51. procedure TestOptShortRefGlobals_SameUnit_EnumType;
  52. procedure TestOptShortRefGlobals_SameUnit_ClassType;
  53. procedure TestOptShortRefGlobals_SameUnit_RecordType;
  54. // Whole Program Optimization
  55. procedure TestWPO_OmitLocalVar;
  56. procedure TestWPO_OmitLocalProc;
  57. procedure TestWPO_OmitLocalProcForward;
  58. procedure TestWPO_OmitProcLocalVar;
  59. procedure TestWPO_OmitProcLocalConst;
  60. procedure TestWPO_OmitProcLocalType;
  61. procedure TestWPO_OmitProcLocalProc;
  62. procedure TestWPO_OmitProcLocalForwardProc;
  63. procedure TestWPO_OmitRecordMember;
  64. procedure TestWPO_OmitNotUsedTObject;
  65. procedure TestWPO_TObject;
  66. procedure TestWPO_Class_Property;
  67. procedure TestWPO_Class_OmitField;
  68. procedure TestWPO_Class_OmitMethod;
  69. procedure TestWPO_Class_OmitClassMethod;
  70. procedure TestWPO_Class_OmitPropertyGetter1;
  71. procedure TestWPO_Class_OmitPropertyGetter2;
  72. procedure TestWPO_Class_OmitPropertySetter1;
  73. procedure TestWPO_Class_OmitPropertySetter2;
  74. procedure TestWPO_Class_KeepNewInstance;
  75. procedure TestWPO_CallInherited;
  76. procedure TestWPO_UseUnit;
  77. procedure TestWPO_ArrayOfConst_Use;
  78. procedure TestWPO_ArrayOfConst_NotUsed;
  79. procedure TestWPO_Class_PropertyInOtherUnit;
  80. procedure TestWPO_ProgramPublicDeclaration;
  81. procedure TestWPO_ConstructorDefaultValueConst;
  82. procedure TestWPO_RTTI_PublishedField;
  83. procedure TestWPO_RTTI_TypeInfo;
  84. end;
  85. implementation
  86. { TCustomTestOptimizations }
  87. function TCustomTestOptimizations.OnConverterIsElementUsed(Sender: TObject;
  88. El: TPasElement): boolean;
  89. var
  90. A: TPas2JSAnalyzer;
  91. begin
  92. if WholeProgramOptimization then
  93. A:=AnalyzerProgram
  94. else if Sender=Converter then
  95. A:=AnalyzerModule
  96. else
  97. begin
  98. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  99. writeln('TCustomTestOptimizations.OnConverterIsElementUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Sender=',GetObjName(Sender));
  100. {$ENDIF}
  101. Fail('converting other unit without WPO');
  102. end;
  103. Result:=A.IsUsed(El);
  104. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  105. writeln('TCustomTestOptimizations.OnConverterIsElementUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Result=',Result);
  106. {$ENDIF}
  107. end;
  108. function TCustomTestOptimizations.OnConverterIsTypeInfoUsed(Sender: TObject;
  109. El: TPasElement): boolean;
  110. var
  111. A: TPas2JSAnalyzer;
  112. begin
  113. if WholeProgramOptimization then
  114. A:=AnalyzerProgram
  115. else if Sender=Converter then
  116. A:=AnalyzerModule
  117. else
  118. begin
  119. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  120. writeln('TCustomTestOptimizations.OnConverterIsTypeInfoUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Sender=',GetObjName(Sender));
  121. {$ENDIF}
  122. Fail('converting other unit without WPO');
  123. end;
  124. Result:=A.IsTypeInfoUsed(El);
  125. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  126. writeln('TCustomTestOptimizations.OnConverterIsTypeInfoUsed El=',GetObjName(El),' WPO=',WholeProgramOptimization,' Result=',Result);
  127. {$ENDIF}
  128. end;
  129. procedure TCustomTestOptimizations.SetUp;
  130. begin
  131. inherited SetUp;
  132. FWholeProgramOptimization:=false;
  133. FAnalyzerModule:=TPas2JSAnalyzer.Create;
  134. FAnalyzerModule.Resolver:=Engine;
  135. FAnalyzerProgram:=TPas2JSAnalyzer.Create;
  136. FAnalyzerProgram.Resolver:=Engine;
  137. end;
  138. procedure TCustomTestOptimizations.TearDown;
  139. begin
  140. FreeAndNil(FAnalyzerProgram);
  141. FreeAndNil(FAnalyzerModule);
  142. inherited TearDown;
  143. end;
  144. procedure TCustomTestOptimizations.ParseModule;
  145. begin
  146. inherited ParseModule;
  147. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  148. writeln('TCustomTestOptimizations.ParseModule START');
  149. {$ENDIF}
  150. AnalyzerModule.AnalyzeModule(Module);
  151. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  152. writeln('TCustomTestOptimizations.ParseModule END');
  153. {$ENDIF}
  154. end;
  155. procedure TCustomTestOptimizations.ParseProgram;
  156. begin
  157. WholeProgramOptimization:=true;
  158. inherited ParseProgram;
  159. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  160. writeln('TCustomTestOptimizations.ParseProgram START');
  161. {$ENDIF}
  162. AnalyzerProgram.AnalyzeWholeProgram(Module as TPasProgram);
  163. {$IF defined(VerbosePas2JS) or defined(VerbosePasAnalyzer)}
  164. writeln('TCustomTestOptimizations.ParseProgram START');
  165. {$ENDIF}
  166. end;
  167. function TCustomTestOptimizations.CreateConverter: TPasToJSConverter;
  168. begin
  169. Result:=inherited CreateConverter;
  170. Result.OnIsElementUsed:=@OnConverterIsElementUsed;
  171. Result.OnIsTypeInfoUsed:=@OnConverterIsTypeInfoUsed;
  172. end;
  173. { TTestOptimizations }
  174. procedure TTestOptimizations.TestOptShortRefGlobals_Program;
  175. begin
  176. AddModuleWithIntfImplSrc('UnitA.pas',
  177. LinesToStr([
  178. 'type',
  179. ' TColor = (red,green,blue);',
  180. ' TColors = set of TColor;',
  181. 'const',
  182. ' cRedBlue = [red,blue];',
  183. 'type',
  184. ' TBird = class',
  185. ' public',
  186. ' class var c: word;',
  187. ' class function Run(w: word): word; virtual; abstract;',
  188. ' end;',
  189. ' TRec = record',
  190. ' x: word;',
  191. ' end;',
  192. 'var b: TBird;',
  193. '']),
  194. LinesToStr([
  195. '']));
  196. StartProgram(true,[supTObject]);
  197. Add([
  198. '{$optimization JSShortRefGlobals}',
  199. 'uses unita;',
  200. 'type',
  201. ' TEagle = class(TBird)',
  202. ' class function Run(w: word = 5): word; override;',
  203. ' end;',
  204. 'class function TEagle.Run(w: word): word;',
  205. 'begin',
  206. 'end;',
  207. 'var',
  208. ' e: TEagle;',
  209. ' r: TRec;',
  210. ' c: TColors;',
  211. 'begin',
  212. ' e:=TEagle.Create;',
  213. ' b:=TBird.Create;',
  214. ' e.c:=e.c+1;',
  215. ' r.x:=TBird.c;',
  216. ' r.x:=b.c;',
  217. ' r.x:=e.Run;',
  218. ' r.x:=e.Run();',
  219. ' r.x:=e.Run(4);',
  220. ' c:=cRedBlue;',
  221. '']);
  222. ConvertProgram;
  223. CheckSource('TestOptShortRefGlobals_Program',
  224. LinesToStr([
  225. 'var $lt = null;',
  226. 'var $lm = pas.UnitA;',
  227. 'var $lt1 = $lm.TBird;',
  228. 'var $lt2 = $lm.TRec;',
  229. 'rtl.createClass(this, "TEagle", $lt1, function () {',
  230. ' $lt = this;',
  231. ' this.Run = function (w) {',
  232. ' var Result = 0;',
  233. ' return Result;',
  234. ' };',
  235. '});',
  236. 'this.e = null;',
  237. 'this.r = $lt2.$new();',
  238. 'this.c = {};',
  239. '']),
  240. LinesToStr([
  241. '$mod.e = $lt.$create("Create");',
  242. '$lm.b = $lt1.$create("Create");',
  243. '$lt1.c = $mod.e.c + 1;',
  244. '$mod.r.x = $lt1.c;',
  245. '$mod.r.x = $lm.b.c;',
  246. '$mod.r.x = $mod.e.$class.Run(5);',
  247. '$mod.r.x = $mod.e.$class.Run(5);',
  248. '$mod.r.x = $mod.e.$class.Run(4);',
  249. '$mod.c = rtl.refSet($lm.cRedBlue);',
  250. '']));
  251. end;
  252. procedure TTestOptimizations.TestOptShortRefGlobals_Unit_FromIntfImpl_ToIntfImpl;
  253. begin
  254. AddModuleWithIntfImplSrc('UnitA.pas',
  255. LinesToStr([
  256. 'type',
  257. ' TBird = class',
  258. ' public Speed: word;',
  259. ' end;',
  260. ' TRecA = record',
  261. ' x: word;',
  262. ' end;',
  263. 'var Bird: TBird;',
  264. '']),
  265. LinesToStr([
  266. '']));
  267. AddModuleWithIntfImplSrc('UnitB.pas',
  268. LinesToStr([
  269. 'type',
  270. ' TAnt = class',
  271. ' public Size: word;',
  272. ' end;',
  273. ' TRecB = record',
  274. ' y: word;',
  275. ' end;',
  276. ' TBear = class',
  277. ' end;',
  278. ' TFrog = class',
  279. ' end;',
  280. 'var Ant: TAnt;',
  281. '']),
  282. LinesToStr([
  283. '']));
  284. StartUnit(true,[supTObject]);
  285. Add([
  286. '{$optimization JSShortRefGlobals}',
  287. 'interface',
  288. 'uses unita;',
  289. 'type',
  290. ' TEagle = class(TBird)', // intf-JS to intf-uses
  291. ' procedure Fly;',
  292. ' end;',
  293. 'implementation',
  294. 'uses unitb;',
  295. 'type',
  296. ' TRedAnt = class(TAnt)', // impl-JS to impl-uses
  297. ' procedure Run;',
  298. ' end;',
  299. 'procedure TEagle.Fly;',
  300. 'begin',
  301. ' TRedAnt.Create;', // intf-JS to impl-JS
  302. ' TAnt.Create;', // intf-JS to impl-uses
  303. ' TBird.Create;', // intf-JS to intf-uses
  304. ' TEagle.Create;', // intf-JS to intf-JS
  305. 'end;',
  306. 'procedure TRedAnt.Run;',
  307. 'begin',
  308. ' TRedAnt.Create;', // impl-JS to impl-JS
  309. ' TAnt.Create;', // impl-JS to impl-uses
  310. ' TBird.Create;', // impl-JS to intf-uses
  311. ' TEagle.Create;', // impl-JS to intf-JS
  312. ' TBear.Create', // only in impl-JS to impl-uses
  313. 'end;',
  314. 'var',
  315. ' RedAnt: TRedAnt;',
  316. ' Ant: TAnt;',
  317. ' Bird: TBird;',
  318. ' Eagle: TEagle;',
  319. 'initialization',
  320. ' RedAnt:=TRedAnt.Create;', // init to impl-JS
  321. ' Ant:=TAnt.Create;', // init to impl-uses
  322. ' Bird:=TBird.Create;', // init to intf-uses
  323. ' Eagle:=TEagle.Create;', // init to intf-JS
  324. ' TFrog.Create;', // only in init to impl-uses
  325. ' Eagle.Fly;',
  326. ' RedAnt.Run;',
  327. '']);
  328. ConvertUnit;
  329. CheckSource('TestOptShortRefGlobals_Unit_FromIntfImpl_ToIntfImpl',
  330. LinesToStr([
  331. 'var $impl = $mod.$impl;',
  332. 'var $lt = null;',
  333. 'var $lt1 = null;',
  334. 'var $lm = pas.UnitA;',
  335. 'var $lt2 = $lm.TBird;',
  336. 'var $lm1 = null;',
  337. 'var $lt3 = null;',
  338. 'var $lt4 = null;',
  339. 'var $lt5 = null;',
  340. 'rtl.createClass(this, "TEagle", $lt2, function () {',
  341. ' $lt = this;',
  342. ' this.Fly = function () {',
  343. ' $lt1.$create("Create");',
  344. ' $lt3.$create("Create");',
  345. ' $lt2.$create("Create");',
  346. ' $lt.$create("Create");',
  347. ' };',
  348. '});',
  349. '']),
  350. LinesToStr([
  351. '$impl.RedAnt = $lt1.$create("Create");',
  352. '$impl.Ant = $lt3.$create("Create");',
  353. '$impl.Bird = $lt2.$create("Create");',
  354. '$impl.Eagle = $lt.$create("Create");',
  355. '$lt5.$create("Create");',
  356. '$impl.Eagle.Fly();',
  357. '$impl.RedAnt.Run();',
  358. '']),
  359. LinesToStr([
  360. '$lm1 = pas.UnitB;',
  361. '$lt3 = $lm1.TAnt;',
  362. '$lt4 = $lm1.TBear;',
  363. '$lt5 = $lm1.TFrog;',
  364. 'rtl.createClass($impl, "TRedAnt", $lt3, function () {',
  365. ' $lt1 = this;',
  366. ' this.Run = function () {',
  367. ' $lt1.$create("Create");',
  368. ' $lt3.$create("Create");',
  369. ' $lt2.$create("Create");',
  370. ' $lt.$create("Create");',
  371. ' $lt4.$create("Create");',
  372. ' };',
  373. '});',
  374. '$impl.RedAnt = null;',
  375. '$impl.Ant = null;',
  376. '$impl.Bird = null;',
  377. '$impl.Eagle = null;',
  378. '']));
  379. end;
  380. procedure TTestOptimizations.TestOptShortRefGlobals_Property;
  381. begin
  382. AddModuleWithIntfImplSrc('UnitA.pas',
  383. LinesToStr([
  384. 'type',
  385. ' TBird = class',
  386. ' FWing: TObject;',
  387. ' class var FLeg: TObject;',
  388. ' public',
  389. ' property Wing: TObject read FWing write FWing;',
  390. ' class property Leg: TObject read FLeg write FLeg;',
  391. ' end;',
  392. '']),
  393. LinesToStr([
  394. '']));
  395. StartUnit(true,[supTObject]);
  396. Add([
  397. '{$optimization JSShortRefGlobals}',
  398. 'interface',
  399. 'uses unita;',
  400. 'type',
  401. ' TEagle = class(TBird)', // intf-JS to intf-uses
  402. ' procedure Fly(o: TObject);',
  403. ' end;',
  404. 'implementation',
  405. 'procedure TEagle.Fly(o: TObject);',
  406. 'begin',
  407. ' Fly(Wing);',
  408. ' Fly(Leg);',
  409. 'end;',
  410. '']);
  411. ConvertUnit;
  412. CheckSource('TestOptShortRefGlobals_Property',
  413. LinesToStr([
  414. 'var $lt = null;',
  415. 'var $lm = pas.UnitA;',
  416. 'var $lt1 = $lm.TBird;',
  417. 'rtl.createClass(this, "TEagle", $lt1, function () {',
  418. ' $lt = this;',
  419. ' this.Fly = function (o) {',
  420. ' this.Fly(this.FWing);',
  421. ' this.Fly(this.FLeg);',
  422. ' };',
  423. '});',
  424. '']),
  425. LinesToStr([
  426. '']),
  427. LinesToStr([
  428. '']));
  429. end;
  430. procedure TTestOptimizations.TestOptShortRefGlobals_GenericFunction;
  431. begin
  432. AddModuleWithIntfImplSrc('UnitA.pas',
  433. LinesToStr([
  434. 'generic function Run<T>(a: T): T;',
  435. '']),
  436. LinesToStr([
  437. 'generic function Run<T>(a: T): T;',
  438. 'begin',
  439. 'end;',
  440. '']));
  441. StartUnit(true,[supTObject]);
  442. Add([
  443. '{$optimization JSShortRefGlobals}',
  444. 'interface',
  445. 'uses unita;',
  446. 'type',
  447. ' TEagle = class',
  448. ' end;',
  449. 'procedure Fly;',
  450. 'implementation',
  451. 'procedure Fly;',
  452. 'begin',
  453. ' specialize Run<TEagle>(nil);',
  454. 'end;',
  455. '']);
  456. ConvertUnit;
  457. CheckSource('TestOptShortRefGlobals_GenericFunction',
  458. LinesToStr([
  459. 'var $lt = null;',
  460. 'var $lm = pas.system;',
  461. 'var $lt1 = $lm.TObject;',
  462. 'var $lm1 = pas.UnitA;',
  463. 'var $lp = $lm1.Run$G1;',
  464. 'rtl.createClass(this, "TEagle", $lt1, function () {',
  465. ' $lt = this;',
  466. '});',
  467. 'this.Fly = function () {',
  468. ' $lp(null);',
  469. '};',
  470. '']),
  471. LinesToStr([
  472. '']),
  473. LinesToStr([
  474. '']));
  475. end;
  476. procedure TTestOptimizations.TestOptShortRefGlobals_SameUnit_EnumType;
  477. begin
  478. StartUnit(true,[supTObject]);
  479. Add([
  480. '{$optimization JSShortRefGlobals}',
  481. 'interface',
  482. 'type',
  483. ' TBird = class',
  484. ' type',
  485. ' TFlag = (big,small);',
  486. ' procedure Fly;',
  487. ' end;',
  488. ' TEnum = (red,blue);',
  489. 'var',
  490. ' e: TEnum;',
  491. ' f: TBird.TFlag;',
  492. 'procedure Run;',
  493. 'implementation',
  494. 'procedure TBird.Fly;',
  495. 'begin',
  496. ' e:=blue;',
  497. ' f:=small;',
  498. 'end;',
  499. 'procedure Run;',
  500. 'type TSub = (left,right);',
  501. 'var s: TSub;',
  502. 'begin',
  503. ' e:=red;',
  504. ' s:=right;',
  505. ' f:=big;',
  506. 'end;',
  507. '']);
  508. ConvertUnit;
  509. CheckSource('TestOptShortRefGlobals_SameUnit_EnumType',
  510. LinesToStr([
  511. 'var $lt = null;',
  512. 'var $lt1 = null;',
  513. 'var $lt2 = null;',
  514. 'var $lm = pas.system;',
  515. 'var $lt3 = $lm.TObject;',
  516. 'rtl.createClass(this, "TBird", $lt3, function () {',
  517. ' $lt = this;',
  518. ' $lt1 = this.TFlag = {',
  519. ' "0": "big",',
  520. ' big: 0,',
  521. ' "1": "small",',
  522. ' small: 1',
  523. ' };',
  524. ' this.Fly = function () {',
  525. ' $mod.e = $lt2.blue;',
  526. ' $mod.f = $lt1.small;',
  527. ' };',
  528. '});',
  529. '$lt2 = this.TEnum = {',
  530. ' "0": "red",',
  531. ' red: 0,',
  532. ' "1": "blue",',
  533. ' blue: 1',
  534. '};',
  535. 'this.e = 0;',
  536. 'this.f = 0;',
  537. 'var TSub = {',
  538. ' "0": "left",',
  539. ' left: 0,',
  540. ' "1": "right",',
  541. ' right: 1',
  542. '};',
  543. 'this.Run = function () {',
  544. ' var s = 0;',
  545. ' $mod.e = $lt2.red;',
  546. ' s = TSub.right;',
  547. ' $mod.f = $lt1.big;',
  548. '};',
  549. '']),
  550. LinesToStr([
  551. '']),
  552. LinesToStr([
  553. '']));
  554. end;
  555. procedure TTestOptimizations.TestOptShortRefGlobals_SameUnit_ClassType;
  556. begin
  557. WithTypeInfo:=true;
  558. StartUnit(true,[supTObject]);
  559. Add([
  560. '{$optimization JSShortRefGlobals}',
  561. 'interface',
  562. 'type',
  563. ' TBird = class;',
  564. ' TAnt = class',
  565. ' type',
  566. ' TLeg = class',
  567. ' end;',
  568. ' procedure Run;',
  569. ' published',
  570. ' Bird: TBird;',
  571. ' end;',
  572. ' TBird = class',
  573. ' procedure Fly;',
  574. ' end;',
  575. 'implementation',
  576. 'type',
  577. ' TFrog = class',
  578. ' end;',
  579. 'procedure TAnt.Run;',
  580. 'begin',
  581. ' if typeinfo(TBird)=nil then;',
  582. ' Bird:=TBird.Create;',
  583. ' TLeg.Create;',
  584. ' TFrog.Create;',
  585. 'end;',
  586. 'procedure TBird.Fly;',
  587. 'begin',
  588. ' if typeinfo(TAnt)=nil then;',
  589. 'end;',
  590. '']);
  591. ConvertUnit;
  592. CheckSource('TestOptShortRefGlobals_SameUnit_ClassType',
  593. LinesToStr([
  594. 'var $impl = $mod.$impl;',
  595. 'var $lt = null;',
  596. 'var $lt1 = null;',
  597. 'var $lt2 = null;',
  598. 'var $lt3 = null;',
  599. 'var $lm = pas.system;',
  600. 'var $lt4 = $lm.TObject;',
  601. 'this.$rtti.$Class("TBird");',
  602. 'rtl.createClass(this, "TAnt", $lt4, function () {',
  603. ' $lt = this;',
  604. ' rtl.createClass(this, "TLeg", $lt4, function () {',
  605. ' $lt1 = this;',
  606. ' }, "TAnt.TLeg");',
  607. ' this.$init = function () {',
  608. ' $lt4.$init.call(this);',
  609. ' this.Bird = null;',
  610. ' };',
  611. ' this.$final = function () {',
  612. ' this.Bird = undefined;',
  613. ' $lt4.$final.call(this);',
  614. ' };',
  615. ' this.Run = function () {',
  616. ' if ($mod.$rtti["TBird"] === null) ;',
  617. ' this.Bird = $lt2.$create("Create");',
  618. ' $lt1.$create("Create");',
  619. ' $lt3.$create("Create");',
  620. ' };',
  621. ' var $r = this.$rtti;',
  622. ' $r.addField("Bird", $mod.$rtti["TBird"]);',
  623. '});',
  624. 'rtl.createClass(this, "TBird", $lt4, function () {',
  625. ' $lt2 = this;',
  626. ' this.Fly = function () {',
  627. ' if ($mod.$rtti["TAnt"] === null) ;',
  628. ' };',
  629. '});',
  630. '']),
  631. LinesToStr([
  632. '']),
  633. LinesToStr([
  634. 'rtl.createClass($impl, "TFrog", $lt4, function () {',
  635. ' $lt3 = this;',
  636. '});',
  637. '']));
  638. end;
  639. procedure TTestOptimizations.TestOptShortRefGlobals_SameUnit_RecordType;
  640. begin
  641. StartUnit(true,[supTObject]);
  642. Add([
  643. '{$optimization JSShortRefGlobals}',
  644. '{$modeswitch advancedrecords}',
  645. 'interface',
  646. 'type',
  647. ' TAnt = record',
  648. ' type',
  649. ' TLeg = record',
  650. ' l: word;',
  651. ' end;',
  652. ' procedure Run;',
  653. ' Leg: TLeg;',
  654. ' end;',
  655. 'implementation',
  656. 'type',
  657. ' TBird = record',
  658. ' b: word;',
  659. ' end;',
  660. 'procedure TAnt.Run;',
  661. 'type',
  662. ' TFoot = record',
  663. ' f: word;',
  664. ' end;',
  665. 'var',
  666. ' b: TBird;',
  667. ' l: TLeg;',
  668. ' a: TAnt;',
  669. ' f: TFoot;',
  670. 'begin',
  671. ' b.b:=1;',
  672. ' l.l:=2;',
  673. ' a.Leg.l:=3;',
  674. ' f.f:=4;',
  675. 'end;',
  676. '']);
  677. ConvertUnit;
  678. CheckSource('TestOptShortRefGlobals_SameUnit_RecordType',
  679. LinesToStr([
  680. 'var $impl = $mod.$impl;',
  681. 'var $lt = null;',
  682. 'var $lt1 = null;',
  683. 'var $lt2 = null;',
  684. 'rtl.recNewT(this, "TAnt", function () {',
  685. ' $lt = this;',
  686. ' rtl.recNewT($lt, "TLeg", function () {',
  687. ' $lt1 = this;',
  688. ' this.l = 0;',
  689. ' this.$eq = function (b) {',
  690. ' return this.l === b.l;',
  691. ' };',
  692. ' this.$assign = function (s) {',
  693. ' this.l = s.l;',
  694. ' return this;',
  695. ' };',
  696. ' });',
  697. ' this.$new = function () {',
  698. ' var r = Object.create(this);',
  699. ' r.Leg = $lt1.$new();',
  700. ' return r;',
  701. ' };',
  702. ' this.$eq = function (b) {',
  703. ' return this.Leg.$eq(b.Leg);',
  704. ' };',
  705. ' this.$assign = function (s) {',
  706. ' this.Leg.$assign(s.Leg);',
  707. ' return this;',
  708. ' };',
  709. ' var TFoot = rtl.recNewT(null, "", function () {',
  710. ' this.f = 0;',
  711. ' this.$eq = function (b) {',
  712. ' return this.f === b.f;',
  713. ' };',
  714. ' this.$assign = function (s) {',
  715. ' this.f = s.f;',
  716. ' return this;',
  717. ' };',
  718. ' });',
  719. ' this.Run = function () {',
  720. ' var b = $lt2.$new();',
  721. ' var l = $lt1.$new();',
  722. ' var a = $lt.$new();',
  723. ' var f = TFoot.$new();',
  724. ' b.b = 1;',
  725. ' l.l = 2;',
  726. ' a.Leg.l = 3;',
  727. ' f.f = 4;',
  728. ' };',
  729. '}, true);',
  730. '']),
  731. LinesToStr([
  732. '']),
  733. LinesToStr([
  734. 'rtl.recNewT($impl, "TBird", function () {',
  735. ' $lt2 = this;',
  736. ' this.b = 0;',
  737. ' this.$eq = function (b) {',
  738. ' return this.b === b.b;',
  739. ' };',
  740. ' this.$assign = function (s) {',
  741. ' this.b = s.b;',
  742. ' return this;',
  743. ' };',
  744. '});',
  745. '']));
  746. end;
  747. procedure TTestOptimizations.TestWPO_OmitLocalVar;
  748. begin
  749. StartProgram(false);
  750. Add('var');
  751. Add(' a: longint;');
  752. Add(' b: longint;');
  753. Add('begin');
  754. Add(' b:=3;');
  755. ConvertProgram;
  756. CheckSource('TestWPO_OmitLocalVar',
  757. 'this.b = 0;',
  758. '$mod.b = 3;');
  759. end;
  760. procedure TTestOptimizations.TestWPO_OmitLocalProc;
  761. begin
  762. StartProgram(false);
  763. Add('procedure DoIt; begin end;');
  764. Add('procedure NoIt; begin end;');
  765. Add('begin');
  766. Add(' DoIt;');
  767. ConvertProgram;
  768. CheckSource('TestWPO_OmitLocalProc',
  769. LinesToStr([
  770. 'this.DoIt = function () {',
  771. '};',
  772. '']),
  773. LinesToStr([
  774. '$mod.DoIt();',
  775. '']));
  776. end;
  777. procedure TTestOptimizations.TestWPO_OmitLocalProcForward;
  778. begin
  779. StartProgram(false);
  780. Add('procedure DoIt; forward;');
  781. Add('procedure NoIt; forward;');
  782. Add('procedure DoIt; begin end;');
  783. Add('procedure NoIt; begin end;');
  784. Add('begin');
  785. Add(' DoIt;');
  786. ConvertProgram;
  787. CheckSource('TestWPO_OmitLocalProcForward',
  788. LinesToStr([
  789. 'this.DoIt = function () {',
  790. '};',
  791. '']),
  792. LinesToStr([
  793. '$mod.DoIt();',
  794. '']));
  795. end;
  796. procedure TTestOptimizations.TestWPO_OmitProcLocalVar;
  797. begin
  798. StartProgram(false);
  799. Add('function DoIt: longint;');
  800. Add('var');
  801. Add(' a: longint;');
  802. Add(' b: longint;');
  803. Add('begin');
  804. Add(' b:=3;');
  805. Add(' Result:=b;');
  806. Add('end;');
  807. Add('begin');
  808. Add(' DoIt;');
  809. ConvertProgram;
  810. CheckSource('TestWPO_OmitProcLocalVar',
  811. LinesToStr([
  812. 'this.DoIt = function () {',
  813. ' var Result = 0;',
  814. ' var b = 0;',
  815. ' b = 3;',
  816. ' Result = b;',
  817. ' return Result;',
  818. '};',
  819. '']),
  820. LinesToStr([
  821. '$mod.DoIt();',
  822. '']));
  823. end;
  824. procedure TTestOptimizations.TestWPO_OmitProcLocalConst;
  825. begin
  826. StartProgram(false);
  827. Add('function DoIt: longint;');
  828. Add('const');
  829. Add(' a = 3;');
  830. Add(' b = 4;');
  831. Add(' c: longint = 5;');
  832. Add(' d: longint = 6;');
  833. Add('begin');
  834. Add(' Result:=b+d;');
  835. Add('end;');
  836. Add('begin');
  837. Add(' DoIt;');
  838. ConvertProgram;
  839. CheckSource('TestWPO_OmitProcLocalConst',
  840. LinesToStr([
  841. 'var b = 4;',
  842. 'var d = 6;',
  843. 'this.DoIt = function () {',
  844. ' var Result = 0;',
  845. ' Result = 4 + d;',
  846. ' return Result;',
  847. '};',
  848. '']),
  849. LinesToStr([
  850. '$mod.DoIt();',
  851. '']));
  852. end;
  853. procedure TTestOptimizations.TestWPO_OmitProcLocalType;
  854. begin
  855. StartProgram(false);
  856. Add('function DoIt: longint;');
  857. Add('type');
  858. Add(' TEnum = (red, green);');
  859. Add(' TEnums = set of TEnum;');
  860. Add('begin');
  861. Add(' Result:=3;');
  862. Add('end;');
  863. Add('begin');
  864. Add(' DoIt;');
  865. ConvertProgram;
  866. CheckSource('TestWPO_OmitProcLocalType',
  867. LinesToStr([
  868. 'this.DoIt = function () {',
  869. ' var Result = 0;',
  870. ' Result = 3;',
  871. ' return Result;',
  872. '};',
  873. '']),
  874. LinesToStr([
  875. '$mod.DoIt();',
  876. '']));
  877. end;
  878. procedure TTestOptimizations.TestWPO_OmitProcLocalProc;
  879. begin
  880. StartProgram(false);
  881. Add('procedure DoIt;');
  882. Add(' procedure SubProcA; begin end;');
  883. Add(' procedure SubProcB; begin end;');
  884. Add('begin');
  885. Add(' SubProcB;');
  886. Add('end;');
  887. Add('begin');
  888. Add(' DoIt;');
  889. ConvertProgram;
  890. CheckSource('TestWPO_OmitProcLocalProc',
  891. LinesToStr([
  892. 'this.DoIt = function () {',
  893. ' function SubProcB() {',
  894. ' };',
  895. ' SubProcB();',
  896. '};',
  897. '']),
  898. LinesToStr([
  899. '$mod.DoIt();',
  900. '']));
  901. end;
  902. procedure TTestOptimizations.TestWPO_OmitProcLocalForwardProc;
  903. begin
  904. StartProgram(false);
  905. Add('procedure DoIt;');
  906. Add(' procedure SubProcA; forward;');
  907. Add(' procedure SubProcB; forward;');
  908. Add(' procedure SubProcA; begin end;');
  909. Add(' procedure SubProcB; begin end;');
  910. Add('begin');
  911. Add(' SubProcB;');
  912. Add('end;');
  913. Add('begin');
  914. Add(' DoIt;');
  915. ConvertProgram;
  916. CheckSource('TestWPO_OmitProcLocalForwardProc',
  917. LinesToStr([
  918. 'this.DoIt = function () {',
  919. ' function SubProcB() {',
  920. ' };',
  921. ' SubProcB();',
  922. '};',
  923. '']),
  924. LinesToStr([
  925. '$mod.DoIt();',
  926. '']));
  927. end;
  928. procedure TTestOptimizations.TestWPO_OmitRecordMember;
  929. begin
  930. StartProgram(false);
  931. Add('type');
  932. Add(' TRec = record');
  933. Add(' a: longint;');
  934. Add(' b: longint;');
  935. Add(' end;');
  936. Add('var r: TRec;');
  937. Add('begin');
  938. Add(' r.a:=3;');
  939. ConvertProgram;
  940. CheckSource('TestWPO_OmitRecordMember',
  941. LinesToStr([
  942. 'rtl.recNewT(this, "TRec", function () {',
  943. ' this.a = 0;',
  944. ' this.$eq = function (b) {',
  945. ' return this.a === b.a;',
  946. ' };',
  947. ' this.$assign = function (s) {',
  948. ' this.a = s.a;',
  949. ' return this;',
  950. ' };',
  951. '});',
  952. 'this.r = this.TRec.$new();',
  953. '']),
  954. LinesToStr([
  955. '$mod.r.a = 3;',
  956. '']));
  957. end;
  958. procedure TTestOptimizations.TestWPO_OmitNotUsedTObject;
  959. begin
  960. StartProgram(false);
  961. Add('type');
  962. Add(' TObject = class end;');
  963. Add('var o: TObject;');
  964. Add('begin');
  965. ConvertProgram;
  966. CheckSource('TestWPO_OmitNotUsedTObject',
  967. LinesToStr([
  968. '']),
  969. LinesToStr([
  970. '']));
  971. end;
  972. procedure TTestOptimizations.TestWPO_TObject;
  973. begin
  974. StartProgram(false);
  975. Add('type');
  976. Add(' TObject = class');
  977. Add(' procedure AfterConstruction; virtual;');
  978. Add(' procedure BeforeDestruction; virtual;');
  979. Add(' end;');
  980. Add('procedure TObject.AfterConstruction; begin end;');
  981. Add('procedure TObject.BeforeDestruction; begin end;');
  982. Add('var o: TObject;');
  983. Add('begin');
  984. Add(' o:=nil;');
  985. ConvertProgram;
  986. CheckSource('TestWPO_TObject',
  987. LinesToStr([
  988. 'rtl.createClass(this, "TObject", null, function () {',
  989. ' this.$init = function () {',
  990. ' };',
  991. ' this.$final = function () {',
  992. ' };',
  993. ' this.AfterConstruction = function () {',
  994. ' };',
  995. ' this.BeforeDestruction = function () {',
  996. ' };',
  997. '});',
  998. 'this.o = null;',
  999. '']),
  1000. LinesToStr([
  1001. '$mod.o = null;']));
  1002. end;
  1003. procedure TTestOptimizations.TestWPO_Class_Property;
  1004. begin
  1005. StartProgram(false);
  1006. Add([
  1007. 'type',
  1008. ' TObject = class',
  1009. ' private',
  1010. ' const CA = 3;',
  1011. ' private',
  1012. ' FA: longint;',
  1013. ' function GetA: longint;',
  1014. ' procedure SetA(Value: longint);',
  1015. ' function IsStoredA: boolean;',
  1016. ' property A: longint read GetA write SetA stored IsStoredA default CA;',
  1017. ' end;',
  1018. 'function tobject.geta: longint; begin end;',
  1019. 'procedure tobject.seta(value: longint); begin end;',
  1020. 'function tobject.isstoreda: boolean; begin end;',
  1021. 'var o: TObject;',
  1022. 'begin',
  1023. ' o.A:=o.A;']);
  1024. ConvertProgram;
  1025. CheckSource('TestWPO_Class_TObject',
  1026. LinesToStr([
  1027. 'rtl.createClass(this, "TObject", null, function () {',
  1028. ' this.$init = function () {',
  1029. ' };',
  1030. ' this.$final = function () {',
  1031. ' };',
  1032. ' this.GetA = function () {',
  1033. ' var Result = 0;',
  1034. ' return Result;',
  1035. ' };',
  1036. ' this.SetA = function (Value) {',
  1037. ' };',
  1038. '});',
  1039. 'this.o = null;',
  1040. '']),
  1041. LinesToStr([
  1042. '$mod.o.SetA($mod.o.GetA());']));
  1043. end;
  1044. procedure TTestOptimizations.TestWPO_Class_OmitField;
  1045. begin
  1046. StartProgram(false);
  1047. Add('type');
  1048. Add(' TObject = class');
  1049. Add(' a: longint;');
  1050. Add(' b: longint;');
  1051. Add(' end;');
  1052. Add('var o: TObject;');
  1053. Add('begin');
  1054. Add(' o.a:=3;');
  1055. ConvertProgram;
  1056. CheckSource('TestWPO_OmitClassField',
  1057. LinesToStr([
  1058. 'rtl.createClass(this, "TObject", null, function () {',
  1059. ' this.$init = function () {',
  1060. ' this.a = 0;',
  1061. ' };',
  1062. ' this.$final = function () {',
  1063. ' };',
  1064. '});',
  1065. 'this.o = null;',
  1066. '']),
  1067. LinesToStr([
  1068. '$mod.o.a = 3;']));
  1069. end;
  1070. procedure TTestOptimizations.TestWPO_Class_OmitMethod;
  1071. begin
  1072. StartProgram(false);
  1073. Add('type');
  1074. Add(' TObject = class');
  1075. Add(' procedure ProcA;');
  1076. Add(' procedure ProcB;');
  1077. Add(' end;');
  1078. Add('procedure TObject.ProcA; begin end;');
  1079. Add('procedure TObject.ProcB; begin end;');
  1080. Add('var o: TObject;');
  1081. Add('begin');
  1082. Add(' o.ProcB;');
  1083. ConvertProgram;
  1084. CheckSource('TestWPO_OmitClassMethod',
  1085. LinesToStr([
  1086. 'rtl.createClass(this, "TObject", null, function () {',
  1087. ' this.$init = function () {',
  1088. ' };',
  1089. ' this.$final = function () {',
  1090. ' };',
  1091. ' this.ProcB = function () {',
  1092. ' };',
  1093. '});',
  1094. 'this.o = null;',
  1095. '']),
  1096. LinesToStr([
  1097. '$mod.o.ProcB();']));
  1098. end;
  1099. procedure TTestOptimizations.TestWPO_Class_OmitClassMethod;
  1100. begin
  1101. StartProgram(false);
  1102. Add('type');
  1103. Add(' TObject = class');
  1104. Add(' class procedure ProcA;');
  1105. Add(' class procedure ProcB;');
  1106. Add(' end;');
  1107. Add('class procedure TObject.ProcA; begin end;');
  1108. Add('class procedure TObject.ProcB; begin end;');
  1109. Add('var o: TObject;');
  1110. Add('begin');
  1111. Add(' o.ProcB;');
  1112. ConvertProgram;
  1113. CheckSource('TestWPO_OmitClassMethod',
  1114. LinesToStr([
  1115. 'rtl.createClass(this, "TObject", null, function () {',
  1116. ' this.$init = function () {',
  1117. ' };',
  1118. ' this.$final = function () {',
  1119. ' };',
  1120. ' this.ProcB = function () {',
  1121. ' };',
  1122. '});',
  1123. 'this.o = null;',
  1124. '']),
  1125. LinesToStr([
  1126. '$mod.o.$class.ProcB();']));
  1127. end;
  1128. procedure TTestOptimizations.TestWPO_Class_OmitPropertyGetter1;
  1129. begin
  1130. StartProgram(false);
  1131. Add('type');
  1132. Add(' TObject = class');
  1133. Add(' FFoo: boolean;');
  1134. Add(' function GetFoo: boolean;');
  1135. Add(' property Foo: boolean read FFoo;');
  1136. Add(' property Foo2: boolean read GetFoo;');
  1137. Add(' FBar: boolean;');
  1138. Add(' function GetBar: boolean;');
  1139. Add(' property Bar: boolean read FBar;');
  1140. Add(' property Bar2: boolean read GetBar;');
  1141. Add(' end;');
  1142. Add('function TObject.GetFoo: boolean; begin Result:=FFoo; end;');
  1143. Add('function TObject.GetBar: boolean; begin Result:=FBar; end;');
  1144. Add('var o: TObject;');
  1145. Add('begin');
  1146. Add(' if o.Foo then;');
  1147. ConvertProgram;
  1148. CheckSource('TestWPO_OmitClassPropertyGetter1',
  1149. LinesToStr([
  1150. 'rtl.createClass(this, "TObject", null, function () {',
  1151. ' this.$init = function () {',
  1152. ' this.FFoo = false;',
  1153. ' };',
  1154. ' this.$final = function () {',
  1155. ' };',
  1156. '});',
  1157. 'this.o = null;',
  1158. '']),
  1159. LinesToStr([
  1160. 'if ($mod.o.FFoo);',
  1161. '']));
  1162. end;
  1163. procedure TTestOptimizations.TestWPO_Class_OmitPropertyGetter2;
  1164. begin
  1165. StartProgram(false);
  1166. Add('type');
  1167. Add(' TObject = class');
  1168. Add(' FFoo: boolean;');
  1169. Add(' function GetFoo: boolean;');
  1170. Add(' property Foo: boolean read FFoo;');
  1171. Add(' property Foo2: boolean read GetFoo;');
  1172. Add(' end;');
  1173. Add('function TObject.GetFoo: boolean; begin Result:=FFoo; end;');
  1174. Add('var o: TObject;');
  1175. Add('begin');
  1176. Add(' if o.Foo2 then;');
  1177. ConvertProgram;
  1178. CheckSource('TestWPO_OmitClassPropertyGetter2',
  1179. LinesToStr([
  1180. 'rtl.createClass(this, "TObject", null, function () {',
  1181. ' this.$init = function () {',
  1182. ' this.FFoo = false;',
  1183. ' };',
  1184. ' this.$final = function () {',
  1185. ' };',
  1186. ' this.GetFoo = function () {',
  1187. ' var Result = false;',
  1188. ' Result = this.FFoo;',
  1189. ' return Result;',
  1190. ' };',
  1191. '});',
  1192. 'this.o = null;',
  1193. '']),
  1194. LinesToStr([
  1195. 'if ($mod.o.GetFoo()) ;',
  1196. '']));
  1197. end;
  1198. procedure TTestOptimizations.TestWPO_Class_OmitPropertySetter1;
  1199. begin
  1200. StartProgram(false);
  1201. Add('type');
  1202. Add(' TObject = class');
  1203. Add(' FFoo: boolean;');
  1204. Add(' procedure SetFoo(Value: boolean);');
  1205. Add(' property Foo: boolean write FFoo;');
  1206. Add(' property Foo2: boolean write SetFoo;');
  1207. Add(' FBar: boolean;');
  1208. Add(' procedure SetBar(Value: boolean);');
  1209. Add(' property Bar: boolean write FBar;');
  1210. Add(' property Bar2: boolean write SetBar;');
  1211. Add(' end;');
  1212. Add('procedure TObject.SetFoo(Value: boolean); begin FFoo:=Value; end;');
  1213. Add('procedure TObject.SetBar(Value: boolean); begin FBar:=Value; end;');
  1214. Add('var o: TObject;');
  1215. Add('begin');
  1216. Add(' o.Foo:=true;');
  1217. ConvertProgram;
  1218. CheckSource('TestWPO_OmitClassPropertySetter1',
  1219. LinesToStr([
  1220. 'rtl.createClass(this, "TObject", null, function () {',
  1221. ' this.$init = function () {',
  1222. ' this.FFoo = false;',
  1223. ' };',
  1224. ' this.$final = function () {',
  1225. ' };',
  1226. '});',
  1227. 'this.o = null;',
  1228. '']),
  1229. LinesToStr([
  1230. '$mod.o.FFoo = true;',
  1231. '']));
  1232. end;
  1233. procedure TTestOptimizations.TestWPO_Class_OmitPropertySetter2;
  1234. begin
  1235. StartProgram(false);
  1236. Add('type');
  1237. Add(' TObject = class');
  1238. Add(' FFoo: boolean;');
  1239. Add(' procedure SetFoo(Value: boolean);');
  1240. Add(' property Foo: boolean write FFoo;');
  1241. Add(' property Foo2: boolean write SetFoo;');
  1242. Add(' end;');
  1243. Add('procedure TObject.SetFoo(Value: boolean); begin FFoo:=Value; end;');
  1244. Add('var o: TObject;');
  1245. Add('begin');
  1246. Add(' o.Foo2:=true;');
  1247. ConvertProgram;
  1248. CheckSource('TestWPO_OmitClassPropertySetter2',
  1249. LinesToStr([
  1250. 'rtl.createClass(this, "TObject", null, function () {',
  1251. ' this.$init = function () {',
  1252. ' this.FFoo = false;',
  1253. ' };',
  1254. ' this.$final = function () {',
  1255. ' };',
  1256. ' this.SetFoo = function (Value) {',
  1257. ' this.FFoo = Value;',
  1258. ' };',
  1259. '});',
  1260. 'this.o = null;',
  1261. '']),
  1262. LinesToStr([
  1263. '$mod.o.SetFoo(true);',
  1264. '']));
  1265. end;
  1266. procedure TTestOptimizations.TestWPO_Class_KeepNewInstance;
  1267. begin
  1268. StartProgram(false);
  1269. Add([
  1270. '{$modeswitch externalclass}',
  1271. 'type',
  1272. ' TExt = class external name ''Object''',
  1273. ' end;',
  1274. ' TBird = class(TExt)',
  1275. ' protected',
  1276. ' class function NewInstance(fnname: string; const paramarray): TBird; virtual;',
  1277. ' public',
  1278. ' constructor Create;',
  1279. ' end;',
  1280. 'class function TBird.NewInstance(fnname: string; const paramarray): TBird;',
  1281. 'begin',
  1282. ' asm',
  1283. ' Result = Object.create();',
  1284. ' end;',
  1285. 'end;',
  1286. 'constructor TBird.Create;',
  1287. 'begin',
  1288. ' inherited;',
  1289. 'end;',
  1290. 'begin',
  1291. ' TBird.Create;',
  1292. '']);
  1293. ConvertProgram;
  1294. CheckSource('TestWPO_Class_KeepNewInstance',
  1295. LinesToStr([
  1296. 'rtl.createClassExt(this, "TBird", Object, "NewInstance", function () {',
  1297. ' this.$init = function () {',
  1298. ' };',
  1299. ' this.$final = function () {',
  1300. ' };',
  1301. ' this.NewInstance = function (fnname, paramarray) {',
  1302. ' var Result = null;',
  1303. ' Result = Object.create();',
  1304. ' return Result;',
  1305. ' };',
  1306. ' this.Create = function () {',
  1307. ' return this;',
  1308. ' };',
  1309. '});',
  1310. '']),
  1311. LinesToStr([
  1312. '$mod.TBird.$create("Create");',
  1313. '']));
  1314. end;
  1315. procedure TTestOptimizations.TestWPO_CallInherited;
  1316. begin
  1317. StartProgram(false);
  1318. Add('type');
  1319. Add(' TObject = class');
  1320. Add(' procedure DoA;');
  1321. Add(' procedure DoB;');
  1322. Add(' end;');
  1323. Add(' TMobile = class');
  1324. Add(' procedure DoA;');
  1325. Add(' procedure DoC;');
  1326. Add(' end;');
  1327. Add('procedure TObject.DoA; begin end;');
  1328. Add('procedure TObject.DoB; begin end;');
  1329. Add('procedure TMobile.DoA;');
  1330. Add('begin');
  1331. Add(' inherited;');
  1332. Add('end;');
  1333. Add('procedure TMobile.DoC;');
  1334. Add('begin');
  1335. Add(' inherited DoB;');
  1336. Add('end;');
  1337. Add('var o: TMobile;');
  1338. Add('begin');
  1339. Add(' o.DoA;');
  1340. Add(' o.DoC;');
  1341. ConvertProgram;
  1342. CheckSource('TestWPO_CallInherited',
  1343. LinesToStr([
  1344. 'rtl.createClass(this, "TObject", null, function () {',
  1345. ' this.$init = function () {',
  1346. ' };',
  1347. ' this.$final = function () {',
  1348. ' };',
  1349. ' this.DoA = function () {',
  1350. ' };',
  1351. ' this.DoB = function () {',
  1352. ' };',
  1353. '});',
  1354. ' rtl.createClass(this, "TMobile", this.TObject, function () {',
  1355. ' this.DoA$1 = function () {',
  1356. ' $mod.TObject.DoA.call(this);',
  1357. ' };',
  1358. ' this.DoC = function () {',
  1359. ' $mod.TObject.DoB.call(this);',
  1360. ' };',
  1361. '});',
  1362. 'this.o = null;',
  1363. '']),
  1364. LinesToStr([
  1365. '$mod.o.DoA$1();',
  1366. '$mod.o.DoC();',
  1367. '']));
  1368. end;
  1369. procedure TTestOptimizations.TestWPO_UseUnit;
  1370. var
  1371. ActualSrc, ExpectedSrc: String;
  1372. begin
  1373. AddModuleWithIntfImplSrc('unit1.pp',
  1374. LinesToStr([
  1375. 'var i: longint;',
  1376. 'procedure DoIt;',
  1377. '']),
  1378. LinesToStr([
  1379. 'procedure DoIt; begin end;']));
  1380. AddModuleWithIntfImplSrc('unit2.pp',
  1381. LinesToStr([
  1382. 'var j: longint;',
  1383. 'procedure DoMore;',
  1384. '']),
  1385. LinesToStr([
  1386. 'procedure DoMore; begin end;']));
  1387. StartProgram(true);
  1388. Add('uses unit2;');
  1389. Add('begin');
  1390. Add(' j:=3;');
  1391. ConvertProgram;
  1392. ActualSrc:=ConvertJSModuleToString(JSModule);
  1393. ExpectedSrc:=LinesToStr([
  1394. 'rtl.module("program", ["system", "unit2"], function () {',
  1395. ' var $mod = this;',
  1396. ' $mod.$main = function () {',
  1397. ' pas.unit2.j = 3;',
  1398. ' };',
  1399. '});',
  1400. '']);
  1401. CheckDiff('TestWPO_UseUnit',ExpectedSrc,ActualSrc);
  1402. end;
  1403. procedure TTestOptimizations.TestWPO_ArrayOfConst_Use;
  1404. begin
  1405. StartProgram(true,[supTVarRec]);
  1406. Add([
  1407. 'procedure Say(arr: array of const);',
  1408. 'begin',
  1409. 'end;',
  1410. 'begin',
  1411. ' Say([true]);']);
  1412. ConvertProgram;
  1413. CheckUnit('system.pp',
  1414. LinesToStr([
  1415. 'rtl.module("system", [], function () {',
  1416. ' var $mod = this;',
  1417. ' rtl.recNewT(this, "TVarRec", function () {',
  1418. ' this.VType = 0;',
  1419. ' this.VJSValue = undefined;',
  1420. ' this.$eq = function (b) {',
  1421. ' return (this.VType === b.VType) && (this.VJSValue === b.VJSValue);',
  1422. ' };',
  1423. ' this.$assign = function (s) {',
  1424. ' this.VType = s.VType;',
  1425. ' this.VJSValue = s.VJSValue;',
  1426. ' return this;',
  1427. ' };',
  1428. ' });',
  1429. ' this.VarRecs = function () {',
  1430. ' var Result = [];',
  1431. ' var v = null;',
  1432. ' v.VType = 1;',
  1433. ' v.VJSValue = 2;',
  1434. ' return Result;',
  1435. ' };',
  1436. '});',
  1437. '']));
  1438. end;
  1439. procedure TTestOptimizations.TestWPO_ArrayOfConst_NotUsed;
  1440. begin
  1441. StartProgram(true,[supTVarRec]);
  1442. Add([
  1443. 'procedure Say(arr: array of const);',
  1444. 'begin',
  1445. 'end;',
  1446. 'begin']);
  1447. ConvertProgram;
  1448. CheckUnit('system.pp',
  1449. LinesToStr([
  1450. 'rtl.module("system", [], function () {',
  1451. ' var $mod = this;',
  1452. '});',
  1453. '']));
  1454. end;
  1455. procedure TTestOptimizations.TestWPO_Class_PropertyInOtherUnit;
  1456. begin
  1457. AddModuleWithIntfImplSrc('unit1.pp',
  1458. LinesToStr([
  1459. 'type',
  1460. ' TObject = class',
  1461. ' private',
  1462. ' const CA = 3;',
  1463. ' private',
  1464. ' FOther: string;',
  1465. ' FA: longint;',
  1466. ' function GetA: longint;',
  1467. ' procedure SetA(Value: longint);',
  1468. ' function IsStoredA: boolean;',
  1469. ' public',
  1470. ' property A: longint read GetA write SetA stored IsStoredA default CA;',
  1471. ' end;',
  1472. '']),
  1473. LinesToStr([
  1474. 'function TObject.geta: longint;',
  1475. 'begin',
  1476. 'end;',
  1477. 'procedure TObject.seta(value: longint);',
  1478. 'begin',
  1479. ' FA:=Value;',
  1480. 'end;',
  1481. 'function TObject.isstoreda: boolean; begin end;',
  1482. '']));
  1483. StartProgram(true);
  1484. Add([
  1485. 'uses unit1;',
  1486. 'var o: TObject;',
  1487. 'begin',
  1488. ' o.A:=o.A;']);
  1489. ConvertProgram;
  1490. CheckUnit('unit1.pp',
  1491. LinesToStr([
  1492. 'rtl.module("unit1", ["system"], function () {',
  1493. ' var $mod = this;',
  1494. ' rtl.createClass(this, "TObject", null, function () {',
  1495. ' this.$init = function () {',
  1496. ' this.FA = 0;',
  1497. ' };',
  1498. ' this.$final = function () {',
  1499. ' };',
  1500. ' this.GetA = function () {',
  1501. ' var Result = 0;',
  1502. ' return Result;',
  1503. ' };',
  1504. ' this.SetA = function (Value) {',
  1505. ' this.FA = Value;',
  1506. ' };',
  1507. ' });',
  1508. '});',
  1509. '']));
  1510. end;
  1511. procedure TTestOptimizations.TestWPO_ProgramPublicDeclaration;
  1512. var
  1513. ActualSrc, ExpectedSrc: String;
  1514. begin
  1515. StartProgram(true);
  1516. Add('var');
  1517. Add(' vPublic: longint; public;');
  1518. Add(' vPrivate: longint;');
  1519. Add('procedure DoPublic; public; begin end;');
  1520. Add('procedure DoPrivate; begin end;');
  1521. Add('begin');
  1522. ConvertProgram;
  1523. ActualSrc:=ConvertJSModuleToString(JSModule);
  1524. ExpectedSrc:=LinesToStr([
  1525. 'rtl.module("program", ["system"], function () {',
  1526. ' var $mod = this;',
  1527. ' this.vPublic = 0;',
  1528. ' this.DoPublic =function(){',
  1529. ' };',
  1530. ' $mod.$main = function () {',
  1531. ' };',
  1532. '});',
  1533. '']);
  1534. CheckDiff('TestWPO_ProgramPublicDeclaration',ExpectedSrc,ActualSrc);
  1535. end;
  1536. procedure TTestOptimizations.TestWPO_ConstructorDefaultValueConst;
  1537. var
  1538. ActualSrc, ExpectedSrc: String;
  1539. begin
  1540. WithTypeInfo:=true;
  1541. StartProgram(true);
  1542. Add([
  1543. 'const gcBlack = 0;',
  1544. 'type',
  1545. ' TColor = longint;',
  1546. ' TObject = class',
  1547. ' private',
  1548. ' FColor: TColor;',
  1549. ' public',
  1550. ' property Color: TColor read FColor write FColor;',
  1551. ' constructor Create(const AColor: TColor = gcBlack);',
  1552. ' end;',
  1553. 'constructor TObject.Create(const AColor: TColor = gcBlack);',
  1554. 'begin',
  1555. ' FColor := AColor;',
  1556. 'end;',
  1557. 'var T: TObject;',
  1558. 'begin',
  1559. ' T := TObject.Create;',
  1560. '']);
  1561. ConvertProgram;
  1562. ActualSrc:=ConvertJSModuleToString(JSModule);
  1563. ExpectedSrc:=LinesToStr([
  1564. 'rtl.module("program",["system"],function () {',
  1565. ' var $mod = this;',
  1566. ' this.gcBlack = 0;',
  1567. ' rtl.createClass(this,"TObject",null,function () {',
  1568. ' this.$init = function () {',
  1569. ' this.FColor = 0;',
  1570. ' };',
  1571. ' this.$final = function () {',
  1572. ' };',
  1573. ' this.Create = function (AColor) {',
  1574. ' this.FColor = AColor;',
  1575. ' return this;',
  1576. ' };',
  1577. ' });',
  1578. ' this.T = null;',
  1579. ' $mod.$main = function () {',
  1580. ' $mod.T = $mod.TObject.$create("Create",[0]);',
  1581. ' };',
  1582. '});',
  1583. '']);
  1584. CheckDiff('TestWPO_ConstructorDefaultValueConst',ExpectedSrc,ActualSrc);
  1585. end;
  1586. procedure TTestOptimizations.TestWPO_RTTI_PublishedField;
  1587. var
  1588. ActualSrc, ExpectedSrc: String;
  1589. begin
  1590. WithTypeInfo:=true;
  1591. StartProgram(true);
  1592. Add('type');
  1593. Add(' TArrA = array of char;');
  1594. Add(' TArrB = array of string;');
  1595. Add(' TObject = class');
  1596. Add(' public');
  1597. Add(' PublicA: TArrA;');
  1598. Add(' published');
  1599. Add(' PublishedB: TArrB;');
  1600. Add(' end;');
  1601. Add('var');
  1602. Add(' C: TObject;');
  1603. Add('begin');
  1604. Add(' C.PublicA:=nil;');
  1605. ConvertProgram;
  1606. ActualSrc:=ConvertJSModuleToString(JSModule);
  1607. ExpectedSrc:=LinesToStr([
  1608. 'rtl.module("program", ["system"], function () {',
  1609. ' var $mod = this;',
  1610. ' this.$rtti.$DynArray("TArrB", {',
  1611. ' eltype: rtl.string',
  1612. ' });',
  1613. ' rtl.createClass(this, "TObject", null, function () {',
  1614. ' this.$init = function () {',
  1615. ' this.PublicA = [];',
  1616. ' this.PublishedB = [];',
  1617. ' };',
  1618. ' this.$final = function () {',
  1619. ' this.PublicA = undefined;',
  1620. ' this.PublishedB = undefined;',
  1621. ' };',
  1622. ' var $r = this.$rtti;',
  1623. ' $r.addField("PublishedB", $mod.$rtti["TArrB"]);',
  1624. ' });',
  1625. ' this.C = null;',
  1626. ' $mod.$main = function () {',
  1627. ' $mod.C.PublicA = [];',
  1628. ' };',
  1629. '});',
  1630. '']);
  1631. CheckDiff('TestWPO_RTTI_PublishedField',ExpectedSrc,ActualSrc);
  1632. end;
  1633. procedure TTestOptimizations.TestWPO_RTTI_TypeInfo;
  1634. var
  1635. ActualSrc, ExpectedSrc: String;
  1636. begin
  1637. WithTypeInfo:=true;
  1638. StartProgram(true);
  1639. Add('type');
  1640. Add(' TArrA = array of char;');
  1641. Add(' TArrB = array of string;');
  1642. Add('var');
  1643. Add(' A: TArrA;');
  1644. Add(' B: TArrB;');
  1645. Add(' p: pointer;');
  1646. Add('begin');
  1647. Add(' A:=nil;');
  1648. Add(' p:=typeinfo(B);');
  1649. ConvertProgram;
  1650. ActualSrc:=ConvertJSModuleToString(JSModule);
  1651. ExpectedSrc:=LinesToStr([
  1652. 'rtl.module("program", ["system"], function () {',
  1653. ' var $mod = this;',
  1654. ' this.$rtti.$DynArray("TArrB", {',
  1655. ' eltype: rtl.string',
  1656. ' });',
  1657. ' this.A = [];',
  1658. ' this.B = [];',
  1659. ' this.p = null;',
  1660. ' $mod.$main = function () {',
  1661. ' $mod.A = [];',
  1662. ' $mod.p = $mod.$rtti["TArrB"];',
  1663. ' };',
  1664. '});',
  1665. '']);
  1666. CheckDiff('TestWPO_RTTI_TypeInfo',ExpectedSrc,ActualSrc);
  1667. end;
  1668. Initialization
  1669. RegisterTests([TTestOptimizations]);
  1670. end.