tcvarparser.pas 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. unit tcvarparser;
  2. {$mode objfpc}{$H+}
  3. interface
  4. uses
  5. Classes, SysUtils, fpcunit, pastree, pscanner, pparser,
  6. tcbaseparser, testregistry;
  7. Type
  8. { TTestVarParser }
  9. TTestVarParser = Class(TTestParser)
  10. private
  11. FHint: string;
  12. FIsThreadVar: Boolean;
  13. FVar: TPasVariable;
  14. Protected
  15. Function ParseVar(ASource : String; Const AHint : String = '') : TPasVariable; virtual; overload;
  16. Procedure AssertVariableType(Const ATypeName : String);
  17. Procedure AssertVariableType(Const AClass : TClass);
  18. Procedure AssertParseVarError(ASource : String);
  19. Property IsThreadVar : Boolean Read FIsThreadVar Write FIsThreadVar;
  20. Property TheVar : TPasVariable Read FVar;
  21. Property Hint : string Read FHint Write FHint;
  22. procedure SetUp; override;
  23. Procedure TearDown; override;
  24. Published
  25. Procedure TestSimpleVar;
  26. Procedure TestSimpleThreadVar;
  27. Procedure TestSimpleVarAbsoluteName;
  28. Procedure TestSimpleVarHelperName;
  29. procedure TestSimpleVarHelperType;
  30. Procedure TestSimpleVarDeprecated;
  31. Procedure TestSimpleVarPlatform;
  32. Procedure TestSimpleVarInitialized;
  33. procedure TestSimpleVarInitializedDeprecated;
  34. procedure TestSimpleVarInitializedPlatform;
  35. Procedure TestSimpleVarAbsolute;
  36. Procedure TestSimpleVarAbsoluteAddress;
  37. Procedure TestSimpleVarAbsoluteDot;
  38. Procedure TestSimpleVarAbsolute2Dots;
  39. Procedure TestVarProcedure;
  40. procedure TestVarProcedureCdecl;
  41. procedure TestVarFunctionFar;
  42. Procedure TestVarFunctionINitialized;
  43. Procedure TestVarProcedureDeprecated;
  44. Procedure TestVarRecord;
  45. Procedure TestVarRecordDeprecated;
  46. Procedure TestVarRecordPlatform;
  47. Procedure TestVarArray;
  48. Procedure TestVarArrayDeprecated;
  49. Procedure TestVarDynArray;
  50. Procedure TestVarExternal;
  51. Procedure TestVarExternalLib;
  52. Procedure TestVarExternalLibName;
  53. procedure TestVarExternalNoSemiColon;
  54. procedure TestVarExternalLibNoName;
  55. Procedure TestVarCVar;
  56. Procedure TestVarCVarExternal;
  57. Procedure TestVarCVarExport;
  58. Procedure TestVarPublic;
  59. Procedure TestVarPublicName;
  60. Procedure TestVarDeprecatedExternalName;
  61. Procedure TestVarHintPriorToInit;
  62. Procedure TestErrorRecovery;
  63. end;
  64. implementation
  65. uses typinfo;
  66. { TTestVarParser }
  67. function TTestVarParser.ParseVar(ASource: String; const AHint: String
  68. ): TPasVariable;
  69. Var
  70. D : String;
  71. begin
  72. Hint:=AHint;
  73. if not IsThreadVar then
  74. Add('Var')
  75. else
  76. Add('Threadvar');
  77. D:='A : '+ASource;
  78. If Hint<>'' then
  79. D:=D+' '+Hint;
  80. Add(' '+D+';');
  81. // Writeln(source.text);
  82. ParseDeclarations;
  83. AssertEquals('One variable definition',1,Declarations.Variables.Count);
  84. AssertEquals('First declaration is type definition.',TPasVariable,TObject(Declarations.Variables[0]).ClassType);
  85. Result:=TPasVariable(Declarations.Variables[0]);
  86. AssertEquals('First declaration has correct name.','A',Result.Name);
  87. FVar:=Result;
  88. Definition:=Result;
  89. if (Hint<>'') then
  90. CheckHint(TPasMemberHint(Getenumvalue(typeinfo(TPasMemberHint),'h'+Hint)));
  91. end;
  92. procedure TTestVarParser.AssertVariableType(const ATypeName: String);
  93. begin
  94. AssertVariableType(TPasUnresolvedTypeRef);
  95. AssertEquals('Correct unresolved type name',ATypeName,theVar.VarType.Name);
  96. end;
  97. procedure TTestVarParser.AssertVariableType(const AClass: TClass);
  98. begin
  99. AssertNotNull('Have variable type',theVar.VarType);
  100. AssertEquals('Correct type class',AClass,theVar.VarType.ClassType);
  101. end;
  102. procedure TTestVarParser.AssertParseVarError(ASource: String);
  103. begin
  104. try
  105. ParseVar(ASource,'');
  106. Fail('Expected parser error');
  107. except
  108. // all OK.
  109. end;
  110. end;
  111. procedure TTestVarParser.SetUp;
  112. begin
  113. inherited SetUp;
  114. FHint:='';
  115. FVar:=Nil;
  116. end;
  117. procedure TTestVarParser.TearDown;
  118. begin
  119. FVar:=Nil;
  120. inherited TearDown;
  121. end;
  122. procedure TTestVarParser.TestSimpleVar;
  123. begin
  124. ParseVar('b','');
  125. AssertVariableType('b');
  126. end;
  127. procedure TTestVarParser.TestSimpleThreadVar;
  128. begin
  129. IsThreadVar:=True;
  130. ParseVar('b','');
  131. AssertVariableType('b');
  132. end;
  133. procedure TTestVarParser.TestSimpleVarAbsoluteName;
  134. Var
  135. R : TPasVariable;
  136. begin
  137. Add('Var');
  138. Add(' Absolute : integer;');
  139. // Writeln(source.text);
  140. ParseDeclarations;
  141. AssertEquals('One variable definition',1,Declarations.Variables.Count);
  142. AssertEquals('First declaration is type definition.',TPasVariable,TObject(Declarations.Variables[0]).ClassType);
  143. R:=TPasVariable(Declarations.Variables[0]);
  144. AssertEquals('First declaration has correct name.','Absolute',R.Name);
  145. end;
  146. procedure TTestVarParser.TestSimpleVarHelperName;
  147. Var
  148. R : TPasVariable;
  149. begin
  150. Add('Var');
  151. Add(' Helper : integer;');
  152. // Writeln(source.text);
  153. ParseDeclarations;
  154. AssertEquals('One variable definition',1,Declarations.Variables.Count);
  155. AssertEquals('First declaration is type definition.',TPasVariable,TObject(Declarations.Variables[0]).ClassType);
  156. R:=TPasVariable(Declarations.Variables[0]);
  157. AssertEquals('First declaration has correct name.','Helper',R.Name);
  158. end;
  159. procedure TTestVarParser.TestSimpleVarHelperType;
  160. begin
  161. ParseVar('helper','');
  162. AssertVariableType('helper');
  163. end;
  164. procedure TTestVarParser.TestSimpleVarDeprecated;
  165. begin
  166. ParseVar('b','deprecated');
  167. AssertVariableType('b');
  168. end;
  169. procedure TTestVarParser.TestSimpleVarPlatform;
  170. begin
  171. ParseVar('b','platform');
  172. AssertVariableType('b');
  173. end;
  174. procedure TTestVarParser.TestSimpleVarInitialized;
  175. begin
  176. ParseVar('b = 123','');
  177. AssertVariableType('b');
  178. AssertNotNull(TheVar.expr);
  179. AssertExpression('Variable value',TheVar.expr,pekNumber,'123');
  180. end;
  181. procedure TTestVarParser.TestSimpleVarInitializedDeprecated;
  182. begin
  183. ParseVar('b = 123','deprecated');
  184. AssertVariableType('b');
  185. AssertNotNull(TheVar.expr);
  186. AssertExpression('Variable value',TheVar.expr,pekNumber,'123');
  187. end;
  188. procedure TTestVarParser.TestSimpleVarInitializedPlatform;
  189. begin
  190. ParseVar('b = 123','platform');
  191. AssertVariableType('b');
  192. AssertNotNull(TheVar.expr);
  193. AssertExpression('Variable value',TheVar.expr,pekNumber,'123');
  194. end;
  195. procedure TTestVarParser.TestSimpleVarAbsolute;
  196. begin
  197. ParseVar('q absolute v','');
  198. AssertVariableType('q');
  199. AssertExpression('correct absolute location',TheVar.AbsoluteExpr,pekIdent,'v');
  200. end;
  201. procedure TTestVarParser.TestSimpleVarAbsoluteAddress;
  202. begin
  203. ParseVar('q absolute $123','');
  204. AssertVariableType('q');
  205. AssertExpression('correct absolute location',TheVar.AbsoluteExpr,pekNumber,'$123');
  206. end;
  207. procedure TTestVarParser.TestSimpleVarAbsoluteDot;
  208. var
  209. B: TBinaryExpr;
  210. begin
  211. ParseVar('q absolute v.w','');
  212. AssertVariableType('q');
  213. B:=AssertExpression('binary',TheVar.AbsoluteExpr,eopSubIdent);
  214. AssertExpression('correct absolute expr v',B.Left,pekIdent,'v');
  215. AssertExpression('correct absolute expr w',B.Right,pekIdent,'w');
  216. end;
  217. procedure TTestVarParser.TestSimpleVarAbsolute2Dots;
  218. var
  219. B: TBinaryExpr;
  220. begin
  221. ParseVar('q absolute v.w.x','');
  222. AssertVariableType('q');
  223. B:=AssertExpression('binary',TheVar.AbsoluteExpr,eopSubIdent);
  224. AssertExpression('correct absolute expr x',B.Right,pekIdent,'x');
  225. B:=AssertExpression('binary',B.Left,eopSubIdent);
  226. AssertExpression('correct absolute expr w',B.Right,pekIdent,'w');
  227. AssertExpression('correct absolute expr v',B.Left,pekIdent,'v');
  228. end;
  229. procedure TTestVarParser.TestVarProcedure;
  230. begin
  231. ParseVar('procedure','');
  232. AssertVariableType(TPasProcedureType);
  233. end;
  234. procedure TTestVarParser.TestVarProcedureCdecl;
  235. begin
  236. ParseVar('procedure; cdecl;','');
  237. AssertVariableType(TPasProcedureType);
  238. end;
  239. procedure TTestVarParser.TestVarFunctionFar;
  240. begin
  241. ParseVar('function (cinfo : j_decompress_ptr) : int; far;','');
  242. AssertVariableType(TPasFunctionType);
  243. end;
  244. procedure TTestVarParser.TestVarFunctionINitialized;
  245. begin
  246. ParseVar('function (device: pointer): pointer; cdecl = nil','');
  247. AssertVariableType(TPasFunctionType);
  248. end;
  249. procedure TTestVarParser.TestVarProcedureDeprecated;
  250. begin
  251. ParseVar('procedure','deprecated');
  252. AssertVariableType(TPasProcedureType);
  253. end;
  254. procedure TTestVarParser.TestVarRecord;
  255. Var
  256. R : TPasRecordtype;
  257. begin
  258. ParseVar('record x,y : intger; end','');
  259. AssertVariableType(TPasRecordType);
  260. R:=TheVar.VarType as TPasRecordType;
  261. AssertEquals('Correct number of fields',2,R.Members.Count);
  262. end;
  263. procedure TTestVarParser.TestVarRecordDeprecated;
  264. Var
  265. R : TPasRecordtype;
  266. begin
  267. ParseVar('record x,y : integer; end','deprecated');
  268. AssertVariableType(TPasRecordType);
  269. R:=TheVar.VarType as TPasRecordType;
  270. AssertEquals('Correct number of fields',2,R.Members.Count);
  271. end;
  272. procedure TTestVarParser.TestVarRecordPlatform;
  273. Var
  274. R : TPasRecordtype;
  275. begin
  276. ParseVar('record x,y : integer; end','platform');
  277. AssertVariableType(TPasRecordType);
  278. R:=TheVar.VarType as TPasRecordType;
  279. AssertEquals('Correct number of fields',2,R.Members.Count);
  280. end;
  281. procedure TTestVarParser.TestVarArray;
  282. Var
  283. R : TPasArrayType;
  284. begin
  285. ParseVar('Array[1..20] of integer','');
  286. AssertVariableType(TPasArrayType);
  287. R:=TheVar.VarType as TPasArrayType;
  288. AssertNotNull('Correct array type name',R.ElType);
  289. AssertEquals('Correct array type name',TPasunresolvedTypeRef,R.ElType.ClassType);
  290. end;
  291. procedure TTestVarParser.TestVarArrayDeprecated;
  292. Var
  293. R : TPasArrayType;
  294. begin
  295. ParseVar('Array[1..20] of integer','Deprecated');
  296. AssertVariableType(TPasArrayType);
  297. R:=TheVar.VarType as TPasArrayType;
  298. AssertNotNull('Correct array type name',R.ElType);
  299. AssertEquals('Correct array type name',TPasunresolvedTypeRef,R.ElType.ClassType);
  300. end;
  301. procedure TTestVarParser.TestVarDynArray;
  302. Var
  303. R : TPasArrayType;
  304. begin
  305. ParseVar('Array of integer','');
  306. AssertVariableType(TPasArrayType);
  307. R:=TheVar.VarType as TPasArrayType;
  308. AssertEquals('No index','',R.IndexRange);
  309. AssertNotNull('Correct array type name',R.ElType);
  310. AssertEquals('Correct array type name',TPasunresolvedTypeRef,R.ElType.ClassType);
  311. end;
  312. procedure TTestVarParser.TestVarExternal;
  313. begin
  314. ParseVar('integer; external','');
  315. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  316. end;
  317. procedure TTestVarParser.TestVarExternalNoSemiColon;
  318. begin
  319. ParseVar('integer external','');
  320. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  321. end;
  322. procedure TTestVarParser.TestVarExternalLib;
  323. begin
  324. ParseVar('integer; external name ''mylib''','');
  325. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  326. AssertNull('Library name',TheVar.LibraryName);
  327. AssertNotNull('Library symbol',TheVar.ExportName);
  328. end;
  329. procedure TTestVarParser.TestVarExternalLibNoName;
  330. begin
  331. // Found in e.g.apache headers
  332. ParseVar('integer; external ''mylib''','');
  333. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  334. AssertNotNull('Library name',TheVar.LibraryName);
  335. end;
  336. procedure TTestVarParser.TestVarExternalLibName;
  337. begin
  338. ParseVar('integer; external ''mylib'' name ''de''','');
  339. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  340. AssertNotNull('Library name',TheVar.LibraryName);
  341. AssertNotNull('Library symbol',TheVar.ExportName);
  342. end;
  343. procedure TTestVarParser.TestVarCVar;
  344. begin
  345. ParseVar('integer; cvar','');
  346. AssertEquals('Variable modifiers',[vmcvar],TheVar.VarModifiers);
  347. end;
  348. procedure TTestVarParser.TestVarCVarExternal;
  349. begin
  350. ParseVar('integer; cvar;external','');
  351. AssertEquals('Variable modifiers',[vmcvar,vmexternal],TheVar.VarModifiers);
  352. end;
  353. procedure TTestVarParser.TestVarCVarExport;
  354. begin
  355. ParseVar('integer; cvar; export','');
  356. AssertEquals('Variable modifiers',[vmCVar,vmExport],TheVar.VarModifiers);
  357. end;
  358. procedure TTestVarParser.TestVarPublic;
  359. begin
  360. ParseVar('integer; public','');
  361. AssertEquals('Variable modifiers',[vmpublic],TheVar.VarModifiers);
  362. end;
  363. procedure TTestVarParser.TestVarPublicName;
  364. begin
  365. ParseVar('integer; public name ''ce''','');
  366. AssertEquals('Variable modifiers',[vmpublic],TheVar.VarModifiers);
  367. AssertNotNull('Public export name',TheVar.ExportName);
  368. end;
  369. procedure TTestVarParser.TestVarDeprecatedExternalName;
  370. begin
  371. ParseVar('integer deprecated; external name ''me''','');
  372. CheckHint(TPasMemberHint(Getenumvalue(typeinfo(TPasMemberHint),'hdeprecated')));
  373. AssertEquals('Variable modifiers',[vmexternal],TheVar.VarModifiers);
  374. AssertNull('Library name',TheVar.LibraryName);
  375. AssertNotNull('Library symbol',TheVar.ExportName);
  376. end;
  377. procedure TTestVarParser.TestVarHintPriorToInit;
  378. Var
  379. E : TBoolConstExpr;
  380. begin
  381. ParseVar('boolean platform = false','');
  382. CheckHint(TPasMemberHint(Getenumvalue(typeinfo(TPasMemberHint),'hplatform')));
  383. AssertNotNull('Correctly initialized',Thevar.Expr);
  384. AssertEquals('Correctly initialized',TBoolConstExpr,Thevar.Expr.ClassType);
  385. E:=Thevar.Expr as TBoolConstExpr;
  386. AssertEquals('Correct initialization value',False, E.Value);
  387. end;
  388. procedure TTestVarParser.TestErrorRecovery;
  389. begin
  390. Add('Var');
  391. Add(' a : integer;');
  392. Add(' a = integer;');
  393. Add(' a : abc integer;');
  394. // Writeln(source.text);
  395. try
  396. Parser.MaxErrorCount:=3;
  397. Parser.OnLog:=@DoParserLog;
  398. ParseDeclarations;
  399. except
  400. On E : Exception do
  401. begin
  402. AssertEquals('Correct class',E.ClassType,EParserError);
  403. end;
  404. end;
  405. AssertErrorCount(2);
  406. end;
  407. initialization
  408. RegisterTests([TTestVarParser]);
  409. end.