tcvarparser.pas 11 KB

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