tcvarparser.pas 12 KB

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