tests.rtti.invoke.pas 69 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758
  1. unit tests.rtti.invoke;
  2. {$ifdef fpc}
  3. {$mode objfpc}{$H+}
  4. {$endif}
  5. {.$define debug}
  6. interface
  7. uses
  8. {$IFDEF FPC}
  9. fpcunit,testregistry, testutils,
  10. {$ELSE FPC}
  11. TestFramework,
  12. {$ENDIF FPC}
  13. sysutils, typinfo, Rtti;
  14. type
  15. {$ifndef fpc}
  16. CodePointer = Pointer;
  17. {$endif}
  18. TTestInvoke = class(TTestCase)
  19. private type
  20. TInvokeFlag = (
  21. ifStatic,
  22. ifConstructor
  23. );
  24. TInvokeFlags = set of TInvokeFlag;
  25. private
  26. function EqualValues(aValue1, aValue2: TValue): Boolean;
  27. function DoInvoke(aCodeAddress: CodePointer; aArgs: TValueArray; aCallConv: TCallConv; aResultType: PTypeInfo; aFlags: TInvokeFlags; out aValid: Boolean): TValue;
  28. procedure DoStaticInvokeTestOrdinalCompare(const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv; aValues: TValueArray; aReturnType: PTypeInfo; aResult: Int64);
  29. procedure DoStaticInvokeTestAnsiStringCompare(const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv; aValues: TValueArray; aReturnType: PTypeInfo; constref aResult: AnsiString);
  30. procedure DoStaticInvokeTestUnicodeStringCompare(const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv; aValues: TValueArray; aReturnType: PTypeInfo; constref aResult: UnicodeString);
  31. procedure DoIntfInvoke(aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  32. procedure DoMethodInvoke(aInst: TObject; aMethod: TMethod; aTypeInfo: PTypeInfo; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  33. procedure DoProcVarInvoke(aInst: TObject; aProc: CodePointer; aTypeInfo: PTypeInfo; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  34. {$ifndef InLazIDE}
  35. {$ifdef fpc}generic{$endif} procedure GenDoMethodInvoke<T>(aInst: TObject; aMethod: T; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  36. {$ifdef fpc}generic{$endif} procedure GenDoProcvarInvoke<T>(aInst: TObject; aProc: T; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  37. {$ifdef fpc}generic{$endif} function GetRecValue<T>(aReverse: Boolean): TValue;
  38. {$endif}
  39. {$ifdef fpc}
  40. procedure Status(const aMsg: String);
  41. {$endif}
  42. published
  43. procedure TestShortString;
  44. procedure TestAnsiString;
  45. procedure TestWideString;
  46. procedure TestUnicodeString;
  47. procedure TestLongInt;
  48. procedure TestInt64;
  49. procedure TestTObject;
  50. procedure TestIntfMethods;
  51. procedure TestIntfMethodsRecs;
  52. procedure TestMethodVars;
  53. procedure TestMethodVarsRecs;
  54. procedure TestProcVars;
  55. procedure TestProcVarsRecs;
  56. end;
  57. {$ifndef fpc}
  58. TValueHelper = record helper for TValue
  59. function AsUnicodeString: UnicodeString;
  60. function AsAnsiString: AnsiString;
  61. end;
  62. {$endif}
  63. implementation
  64. {$ifndef fpc}
  65. function TValueHelper.AsUnicodeString: UnicodeString;
  66. begin
  67. Result := UnicodeString(AsString);
  68. end;
  69. function TValueHelper.AsAnsiString: AnsiString;
  70. begin
  71. Result := AnsiString(AsString);
  72. end;
  73. {$endif}
  74. function TTestInvoke.EqualValues(aValue1, aValue2: TValue): Boolean;
  75. var
  76. td1, td2: PTypeData;
  77. i: SizeInt;
  78. begin
  79. {$ifdef debug}
  80. Writeln('Empty: ', aValue1.IsEmpty, ' ', aValue2.IsEmpty);
  81. Writeln('Kind: ', aValue1.Kind, ' ', aValue2.Kind);
  82. Writeln('Array: ', aValue1.IsArray, ' ', aValue2.IsArray);
  83. {$endif}
  84. if aValue1.IsEmpty and aValue2.IsEmpty then
  85. Result := True
  86. else if aValue1.IsEmpty and not aValue2.IsEmpty then
  87. Result := False
  88. else if not aValue1.IsEmpty and aValue2.IsEmpty then
  89. Result := False
  90. else if aValue1.IsArray and aValue2.IsArray then begin
  91. if aValue1.GetArrayLength = aValue2.GetArrayLength then begin
  92. Result := True;
  93. for i := 0 to aValue1.GetArrayLength - 1 do
  94. if not EqualValues(aValue1.GetArrayElement(i), aValue2.GetArrayElement(i)) then begin
  95. Writeln('Element ', i, ' differs: ', HexStr(aValue1.GetArrayElement(i).AsOrdinal, 4), ' ', HexStr(aValue2.GetArrayElement(i).AsOrdinal, 4));
  96. Result := False;
  97. Break;
  98. end;
  99. end else
  100. Result := False;
  101. end else if aValue1.Kind = aValue2.Kind then begin
  102. td1 := aValue1.TypeData;
  103. td2 := aValue2.TypeData;
  104. case aValue1.Kind of
  105. tkBool:
  106. Result := aValue1.AsBoolean xor not aValue2.AsBoolean;
  107. tkSet:
  108. if td1^.SetSize = td2^.SetSize then
  109. if td1^.SetSize < SizeOf(SizeInt) then
  110. Result := aValue1.AsOrdinal = aValue2.AsOrdinal
  111. else
  112. Result := CompareMem(aValue1.GetReferenceToRawData, aValue2.GetReferenceToRawData, td1^.SetSize)
  113. else
  114. Result := False;
  115. tkEnumeration,
  116. tkChar,
  117. tkWChar,
  118. tkUChar,
  119. tkInt64,
  120. tkInteger:
  121. Result := aValue1.AsOrdinal = aValue2.AsOrdinal;
  122. tkQWord:
  123. Result := aValue1.AsUInt64 = aValue2.AsUInt64;
  124. tkSString,
  125. tkUString,
  126. tkAString,
  127. tkWString:
  128. Result := aValue1.AsString = aValue2.AsString;
  129. tkDynArray,
  130. tkArray:
  131. if aValue1.GetArrayLength = aValue2.GetArrayLength then begin
  132. Result := True;
  133. for i := 0 to aValue1.GetArrayLength - 1 do
  134. if not EqualValues(aValue1.GetArrayElement(i), aValue2.GetArrayElement(i)) then begin
  135. Result := False;
  136. Break;
  137. end;
  138. end else
  139. Result := False;
  140. tkClass,
  141. tkClassRef,
  142. tkInterface,
  143. tkInterfaceRaw,
  144. tkPointer:
  145. Result := PPointer(aValue1.GetReferenceToRawData)^ = PPointer(aValue2.GetReferenceToRawData)^;
  146. tkProcVar:
  147. Result := PCodePointer(aValue1.GetReferenceToRawData)^ = PCodePointer(aValue2.GetReferenceToRawData)^;
  148. tkRecord,
  149. tkObject,
  150. tkMethod,
  151. tkVariant: begin
  152. if aValue1.DataSize = aValue2.DataSize then
  153. Result := CompareMem(aValue1.GetReferenceToRawData, aValue2.GetReferenceToRawData, aValue1.DataSize)
  154. else
  155. Result := False;
  156. end
  157. else
  158. Result := False;
  159. end;
  160. end else
  161. Result := False;
  162. end;
  163. function TTestInvoke.DoInvoke(aCodeAddress: CodePointer; aArgs: TValueArray;
  164. aCallConv: TCallConv; aResultType: PTypeInfo; aFlags: TInvokeFlags; out aValid: Boolean): TValue;
  165. begin
  166. try
  167. Result := Rtti.Invoke(aCodeAddress, aArgs, aCallConv, aResultType, ifStatic in aFlags, ifConstructor in aFlags);
  168. aValid := True;
  169. except
  170. on e: ENotImplemented do begin
  171. Status('Ignoring unimplemented functionality of test');
  172. aValid := False;
  173. end else
  174. raise;
  175. end;
  176. end;
  177. procedure TTestInvoke.DoStaticInvokeTestOrdinalCompare(const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv; aValues: TValueArray; aReturnType: PTypeInfo; aResult: Int64);
  178. var
  179. resval: TValue;
  180. valid: Boolean;
  181. begin
  182. resval := DoInvoke(aAddress, aValues, aCallConv, aReturnType, [ifStatic], valid);
  183. if valid and Assigned(aReturnType) and (resval.AsOrdinal <> aResult) then begin
  184. Fail('Result of test "%s" is unexpected; expected: %s, got: %s', [aTestName, IntToStr(aResult), IntToStr(resval.AsOrdinal)]);
  185. end;
  186. end;
  187. procedure TTestInvoke.DoStaticInvokeTestAnsiStringCompare(
  188. const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv;
  189. aValues: TValueArray; aReturnType: PTypeInfo; constref aResult: AnsiString);
  190. var
  191. resval: TValue;
  192. valid: Boolean;
  193. begin
  194. resval := DoInvoke(aAddress, aValues, aCallConv, aReturnType, [ifStatic], valid);
  195. if valid and Assigned(aReturnType) and (resval.AsAnsiString <> aResult) then begin
  196. Fail('Result of test "%s" is unexpected; expected: "%s", got: "%s"', [aTestName, aResult, resval.AsString]);
  197. end;
  198. end;
  199. procedure TTestInvoke.DoStaticInvokeTestUnicodeStringCompare(
  200. const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv;
  201. aValues: TValueArray; aReturnType: PTypeInfo; constref aResult: UnicodeString
  202. );
  203. var
  204. resval: TValue;
  205. valid: Boolean;
  206. begin
  207. resval := DoInvoke(aAddress, aValues, aCallConv, aReturnType, [ifStatic], valid);
  208. if valid and Assigned(aReturnType) and (resval.AsUnicodeString <> aResult) then begin
  209. Fail('Result of test "%s" is unexpected; expected: "%s", got: "%s"', [aTestName, aResult, resval.AsString]);
  210. end;
  211. end;
  212. {$ifdef fpc}
  213. procedure TTestInvoke.Status(const aMsg: String);
  214. begin
  215. {$ifdef debug}
  216. Writeln(aMsg);
  217. {$endif}
  218. end;
  219. {$endif}
  220. function TestShortStringRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: ShortString): ShortString; register;
  221. begin
  222. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  223. end;
  224. function TestShortStringCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: ShortString): ShortString; cdecl;
  225. begin
  226. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  227. end;
  228. function TestShortStringStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: ShortString): ShortString; stdcall;
  229. begin
  230. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  231. end;
  232. function TestShortStringPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: ShortString): ShortString; pascal;
  233. begin
  234. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  235. end;
  236. procedure TTestInvoke.TestShortString;
  237. const
  238. strs: array[0..5] of ShortString = (
  239. 'This ',
  240. 'is a ',
  241. 'test ',
  242. 'of ',
  243. 'shortstring ',
  244. 'concatenation'
  245. );
  246. var
  247. values: TValueArray;
  248. resstr: ShortString;
  249. i: LongInt;
  250. begin
  251. SetLength(values, Length(strs));
  252. resstr := '';
  253. for i := Low(values) to High(values) do begin
  254. TValue.Make(@strs[i], TypeInfo(ShortString), values[i]);
  255. resstr := resstr + strs[i];
  256. end;
  257. DoStaticInvokeTestAnsiStringCompare('ShortString Register', @TestShortStringRegister, ccReg, values, TypeInfo(ShortString), resstr);
  258. DoStaticInvokeTestAnsiStringCompare('ShortString Cdecl', @TestShortStringCdecl, ccCdecl, values, TypeInfo(ShortString), resstr);
  259. DoStaticInvokeTestAnsiStringCompare('ShortString StdCall', @TestShortStringStdCall, ccStdCall, values, TypeInfo(ShortString), resstr);
  260. DoStaticInvokeTestAnsiStringCompare('ShortString Pascal', @TestShortStringPascal, ccPascal, values, TypeInfo(ShortString), resstr);
  261. end;
  262. function TestAnsiStringRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: AnsiString): AnsiString; register;
  263. begin
  264. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  265. end;
  266. function TestAnsiStringCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: AnsiString): AnsiString; cdecl;
  267. begin
  268. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  269. end;
  270. function TestAnsiStringStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: AnsiString): AnsiString; stdcall;
  271. begin
  272. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  273. end;
  274. function TestAnsiStringPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: AnsiString): AnsiString; pascal;
  275. begin
  276. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  277. end;
  278. procedure TTestInvoke.TestAnsiString;
  279. const
  280. strs: array[0..5] of AnsiString = (
  281. 'This ',
  282. 'is a ',
  283. 'test ',
  284. 'of ',
  285. 'AnsiString ',
  286. 'concatenation'
  287. );
  288. var
  289. values: TValueArray;
  290. resstr: AnsiString;
  291. i: LongInt;
  292. begin
  293. SetLength(values, Length(strs));
  294. resstr := '';
  295. for i := Low(values) to High(values) do begin
  296. TValue.Make(@strs[i], TypeInfo(AnsiString), values[i]);
  297. resstr := resstr + strs[i];
  298. end;
  299. DoStaticInvokeTestAnsiStringCompare('AnsiString Register', @TestAnsiStringRegister, ccReg, values, TypeInfo(AnsiString), resstr);
  300. DoStaticInvokeTestAnsiStringCompare('AnsiString Cdecl', @TestAnsiStringCdecl, ccCdecl, values, TypeInfo(AnsiString), resstr);
  301. DoStaticInvokeTestAnsiStringCompare('AnsiString StdCall', @TestAnsiStringStdCall, ccStdCall, values, TypeInfo(AnsiString), resstr);
  302. DoStaticInvokeTestAnsiStringCompare('AnsiString Pascal', @TestAnsiStringPascal, ccPascal, values, TypeInfo(AnsiString), resstr);
  303. end;
  304. function TestWideStringRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: WideString): WideString; register;
  305. begin
  306. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  307. end;
  308. function TestWideStringCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: WideString): WideString; cdecl;
  309. begin
  310. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  311. end;
  312. function TestWideStringStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: WideString): WideString; stdcall;
  313. begin
  314. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  315. end;
  316. function TestWideStringPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: WideString): WideString; pascal;
  317. begin
  318. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  319. end;
  320. procedure TTestInvoke.TestWideString;
  321. const
  322. strs: array[0..5] of WideString = (
  323. 'This ',
  324. 'is a ',
  325. 'test ',
  326. 'of ',
  327. 'WideString ',
  328. 'concatenation'
  329. );
  330. var
  331. values: TValueArray;
  332. resstr: WideString;
  333. i: LongInt;
  334. begin
  335. SetLength(values, Length(strs));
  336. resstr := '';
  337. for i := Low(values) to High(values) do begin
  338. TValue.Make(@strs[i], TypeInfo(WideString), values[i]);
  339. resstr := resstr + strs[i];
  340. end;
  341. DoStaticInvokeTestUnicodeStringCompare('WideString Register', @TestWideStringRegister, ccReg, values, TypeInfo(WideString), resstr);
  342. DoStaticInvokeTestUnicodeStringCompare('WideString Cdecl', @TestWideStringCdecl, ccCdecl, values, TypeInfo(WideString), resstr);
  343. DoStaticInvokeTestUnicodeStringCompare('WideString StdCall', @TestWideStringStdCall, ccStdCall, values, TypeInfo(WideString), resstr);
  344. DoStaticInvokeTestUnicodeStringCompare('WideString Pascal', @TestWideStringPascal, ccPascal, values, TypeInfo(WideString), resstr);
  345. end;
  346. function TestUnicodeStringRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: UnicodeString): UnicodeString; register;
  347. begin
  348. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  349. end;
  350. function TestUnicodeStringCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: UnicodeString): UnicodeString; cdecl;
  351. begin
  352. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  353. end;
  354. function TestUnicodeStringStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: UnicodeString): UnicodeString; stdcall;
  355. begin
  356. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  357. end;
  358. function TestUnicodeStringPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: UnicodeString): UnicodeString; pascal;
  359. begin
  360. Result := aArg1 + aArg2 + aArg3 + aArg4 + aArg5 + aArg6;
  361. end;
  362. procedure TTestInvoke.TestUnicodeString;
  363. const
  364. strs: array[0..5] of UnicodeString = (
  365. 'This ',
  366. 'is a ',
  367. 'test ',
  368. 'of ',
  369. 'UnicodeString ',
  370. 'concatenation'
  371. );
  372. var
  373. values: TValueArray;
  374. resstr: UnicodeString;
  375. i: LongInt;
  376. begin
  377. SetLength(values, Length(strs));
  378. resstr := '';
  379. for i := Low(values) to High(values) do begin
  380. TValue.Make(@strs[i], TypeInfo(UnicodeString), values[i]);
  381. resstr := resstr + strs[i];
  382. end;
  383. DoStaticInvokeTestUnicodeStringCompare('UnicodeString Register', @TestUnicodeStringRegister, ccReg, values, TypeInfo(UnicodeString), resstr);
  384. DoStaticInvokeTestUnicodeStringCompare('UnicodeString Cdecl', @TestUnicodeStringCdecl, ccCdecl, values, TypeInfo(UnicodeString), resstr);
  385. DoStaticInvokeTestUnicodeStringCompare('UnicodeString StdCall', @TestUnicodeStringStdCall, ccStdCall, values, TypeInfo(UnicodeString), resstr);
  386. DoStaticInvokeTestUnicodeStringCompare('UnicodeString Pascal', @TestUnicodeStringPascal, ccPascal, values, TypeInfo(UnicodeString), resstr);
  387. end;
  388. function TestLongIntRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: LongInt): LongInt; register;
  389. begin
  390. Result := aArg1 + aArg2 * 10 + aArg3 * 100 + aArg4 * 1000 + aArg5 * 10000 + aArg6 * 100000;
  391. end;
  392. function TestLongIntCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: LongInt): LongInt; cdecl;
  393. begin
  394. Result := aArg1 + aArg2 * 10 + aArg3 * 100 + aArg4 * 1000 + aArg5 * 10000 + aArg6 * 100000;
  395. end;
  396. function TestLongIntStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: LongInt): LongInt; stdcall;
  397. begin
  398. Result := aArg1 + aArg2 * 10 + aArg3 * 100 + aArg4 * 1000 + aArg5 * 10000 + aArg6 * 100000;
  399. end;
  400. function TestLongIntPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: LongInt): LongInt; pascal;
  401. begin
  402. Result := aArg1 + aArg2 * 10 + aArg3 * 100 + aArg4 * 1000 + aArg5 * 10000 + aArg6 * 100000;
  403. end;
  404. procedure TTestInvoke.TestLongInt;
  405. const
  406. vals: array[0..5] of LongInt = (
  407. 8,
  408. 4,
  409. 7,
  410. 3,
  411. 6,
  412. 1
  413. );
  414. var
  415. values: TValueArray;
  416. resval, factor: LongInt;
  417. i: LongInt;
  418. begin
  419. SetLength(values, Length(vals));
  420. resval := 0;
  421. factor := 1;
  422. for i := Low(values) to High(values) do begin
  423. TValue.Make(@vals[i], TypeInfo(LongInt), values[i]);
  424. resval := resval + vals[i] * factor;
  425. factor := factor * 10;
  426. end;
  427. DoStaticInvokeTestOrdinalCompare('LongInt Register', @TestLongIntRegister, ccReg, values, TypeInfo(LongInt), resval);
  428. DoStaticInvokeTestOrdinalCompare('LongInt Cdecl', @TestLongIntCdecl, ccCdecl, values, TypeInfo(LongInt), resval);
  429. DoStaticInvokeTestOrdinalCompare('LongInt StdCall', @TestLongIntStdCall, ccStdCall, values, TypeInfo(LongInt), resval);
  430. DoStaticInvokeTestOrdinalCompare('LongInt Pascal', @TestLongIntPascal, ccPascal, values, TypeInfo(LongInt), resval);
  431. end;
  432. function TestInt64Register(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: Int64): Int64; register;
  433. begin
  434. Result := aArg1 + aArg2 * 100 + aArg3 * 10000 + aArg4 * 1000000 + aArg5 * 100000000 + aArg6 * 10000000000;
  435. end;
  436. function TestInt64Cdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: Int64): Int64; cdecl;
  437. begin
  438. Result := aArg1 + aArg2 * 100 + aArg3 * 10000 + aArg4 * 1000000 + aArg5 * 100000000 + aArg6 * 10000000000;
  439. end;
  440. function TestInt64StdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: Int64): Int64; stdcall;
  441. begin
  442. Result := aArg1 + aArg2 * 100 + aArg3 * 10000 + aArg4 * 1000000 + aArg5 * 100000000 + aArg6 * 10000000000;
  443. end;
  444. function TestInt64Pascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: Int64): Int64; pascal;
  445. begin
  446. Result := aArg1 + aArg2 * 100 + aArg3 * 10000 + aArg4 * 1000000 + aArg5 * 100000000 + aArg6 * 10000000000;
  447. end;
  448. procedure TTestInvoke.TestInt64;
  449. const
  450. vals: array[0..5] of Int64 = (
  451. 8,
  452. 4,
  453. 7,
  454. 3,
  455. 6,
  456. 1
  457. );
  458. var
  459. values: TValueArray;
  460. resval, factor: Int64;
  461. i: LongInt;
  462. begin
  463. SetLength(values, Length(vals));
  464. resval := 0;
  465. factor := 1;
  466. for i := Low(values) to High(values) do begin
  467. TValue.Make(@vals[i], TypeInfo(Int64), values[i]);
  468. resval := resval + vals[i] * factor;
  469. factor := factor * 100;
  470. end;
  471. DoStaticInvokeTestOrdinalCompare('Int64 Register', @TestInt64Register, ccReg, values, TypeInfo(Int64), resval);
  472. DoStaticInvokeTestOrdinalCompare('Int64 Cdecl', @TestInt64Cdecl, ccCdecl, values, TypeInfo(Int64), resval);
  473. DoStaticInvokeTestOrdinalCompare('Int64 StdCall', @TestInt64StdCall, ccStdCall, values, TypeInfo(Int64), resval);
  474. DoStaticInvokeTestOrdinalCompare('Int64 Pascal', @TestInt64Pascal, ccPascal, values, TypeInfo(Int64), resval);
  475. end;
  476. type
  477. TTestClass = class
  478. fString: String;
  479. fValue: LongInt;
  480. end;
  481. function TestTTestClassRegister(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: TTestClass): TTestClass; register;
  482. begin
  483. Result := TTestClass.Create;
  484. Result.fString := aArg1.fString + aArg2.fString + aArg3.fString + aArg4.fString + aArg5.fString + aArg6.fString;
  485. Result.fValue := aArg1.fValue + aArg2.fValue * 10 + aArg3.fValue * 100 + aArg4.fValue * 1000 + aArg5.fValue * 10000 + aArg6.fValue * 100000;
  486. end;
  487. function TestTTestClassCdecl(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: TTestClass): TTestClass; cdecl;
  488. begin
  489. Result := TTestClass.Create;
  490. Result.fString := aArg1.fString + aArg2.fString + aArg3.fString + aArg4.fString + aArg5.fString + aArg6.fString;
  491. Result.fValue := aArg1.fValue + aArg2.fValue * 10 + aArg3.fValue * 100 + aArg4.fValue * 1000 + aArg5.fValue * 10000 + aArg6.fValue * 100000;
  492. end;
  493. function TestTTestClassStdCall(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: TTestClass): TTestClass; stdcall;
  494. begin
  495. Result := TTestClass.Create;
  496. Result.fString := aArg1.fString + aArg2.fString + aArg3.fString + aArg4.fString + aArg5.fString + aArg6.fString;
  497. Result.fValue := aArg1.fValue + aArg2.fValue * 10 + aArg3.fValue * 100 + aArg4.fValue * 1000 + aArg5.fValue * 10000 + aArg6.fValue * 100000;
  498. end;
  499. function TestTTestClassPascal(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6: TTestClass): TTestClass; pascal;
  500. begin
  501. Result := TTestClass.Create;
  502. Result.fString := aArg1.fString + aArg2.fString + aArg3.fString + aArg4.fString + aArg5.fString + aArg6.fString;
  503. Result.fValue := aArg1.fValue + aArg2.fValue * 10 + aArg3.fValue * 100 + aArg4.fValue * 1000 + aArg5.fValue * 10000 + aArg6.fValue * 100000;
  504. end;
  505. procedure TTestInvoke.TestTObject;
  506. procedure DoStaticInvokeTestClassCompare(
  507. const aTestName: String; aAddress: CodePointer; aCallConv: TCallConv;
  508. aValues: TValueArray; aReturnType: PTypeInfo; aResult: TTestClass
  509. );
  510. var
  511. resval: TValue;
  512. rescls: TTestClass;
  513. valid: Boolean;
  514. begin
  515. resval := DoInvoke(aAddress, aValues, aCallConv, aReturnType, [ifStatic], valid);
  516. if valid and Assigned(aReturnType) then begin
  517. rescls := TTestClass(PPointer(resval.GetReferenceToRawData)^);
  518. if (rescls.fString <> aResult.fString) or (rescls.fValue <> aResult.fValue) then
  519. Fail('Result of test "%s" is unexpected; expected: "%s"/%s, got: "%s"/%s', [aTestName, aResult.fString, IntToStr(aResult.fValue), rescls.fString, IntToStr(rescls.fValue)]);
  520. end;
  521. end;
  522. const
  523. strs: array[0..5] of AnsiString = (
  524. 'This ',
  525. 'is a ',
  526. 'test ',
  527. 'of ',
  528. 'AnsiString ',
  529. 'concatenation'
  530. );
  531. vals: array[0..5] of Int64 = (
  532. 8,
  533. 4,
  534. 7,
  535. 3,
  536. 6,
  537. 1
  538. );
  539. var
  540. values: TValueArray;
  541. t, rescls: TTestClass;
  542. i, factor: LongInt;
  543. begin
  544. SetLength(values, Length(vals));
  545. factor := 1;
  546. rescls := TTestClass.Create;
  547. for i := Low(values) to High(values) do begin
  548. t := TTestClass.Create;
  549. t.fString := strs[i];
  550. t.fValue := vals[i];
  551. TValue.Make(@t, TypeInfo(TTestClass), values[i]);
  552. rescls.fValue := rescls.fValue + vals[i] * factor;
  553. rescls.fString := rescls.fString + strs[i];
  554. factor := factor * 10;
  555. end;
  556. DoStaticInvokeTestClassCompare('TTestClass Register', @TestTTestClassRegister, ccReg, values, TypeInfo(TTestClass), rescls);
  557. DoStaticInvokeTestClassCompare('TTestClass Cdecl', @TestTTestClassCdecl, ccCdecl, values, TypeInfo(TTestClass), rescls);
  558. DoStaticInvokeTestClassCompare('TTestClass StdCall', @TestTTestClassStdCall, ccStdCall, values, TypeInfo(TTestClass), rescls);
  559. DoStaticInvokeTestClassCompare('TTestClass Pascal', @TestTTestClassPascal, ccPascal, values, TypeInfo(TTestClass), rescls);
  560. end;
  561. type
  562. TTestRecord1 = packed record
  563. b: array[0..0] of Byte;
  564. end;
  565. TTestRecord2 = packed record
  566. b: array[0..1] of Byte;
  567. end;
  568. TTestRecord3 = packed record
  569. b: array[0..2] of Byte;
  570. end;
  571. TTestRecord4 = packed record
  572. b: array[0..3] of Byte;
  573. end;
  574. TTestRecord5 = packed record
  575. b: array[0..4] of Byte;
  576. end;
  577. TTestRecord6 = packed record
  578. b: array[0..5] of Byte;
  579. end;
  580. TTestRecord7 = packed record
  581. b: array[0..6] of Byte;
  582. end;
  583. TTestRecord8 = packed record
  584. b: array[0..7] of Byte;
  585. end;
  586. TTestRecord9 = packed record
  587. b: array[0..8] of Byte;
  588. end;
  589. TTestRecord10 = packed record
  590. b: array[0..9] of Byte;
  591. end;
  592. {$M+}
  593. ITestInterface = interface
  594. procedure Test1;
  595. function Test2: SizeInt;
  596. function Test3(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt;
  597. procedure Test4(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString);
  598. function Test5: AnsiString;
  599. function Test6: UnicodeString;
  600. function Test7: WideString;
  601. function Test8: ShortString;
  602. procedure Test9(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt);
  603. procedure Test10(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString);
  604. procedure Test11(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString);
  605. procedure Test12(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt);
  606. function TestRecSize1(aArg1: TTestRecord1): TTestRecord1;
  607. function TestRecSize2(aArg1: TTestRecord2): TTestRecord2;
  608. function TestRecSize3(aArg1: TTestRecord3): TTestRecord3;
  609. function TestRecSize4(aArg1: TTestRecord4): TTestRecord4;
  610. function TestRecSize5(aArg1: TTestRecord5): TTestRecord5;
  611. function TestRecSize6(aArg1: TTestRecord6): TTestRecord6;
  612. function TestRecSize7(aArg1: TTestRecord7): TTestRecord7;
  613. function TestRecSize8(aArg1: TTestRecord8): TTestRecord8;
  614. function TestRecSize9(aArg1: TTestRecord9): TTestRecord9;
  615. function TestRecSize10(aArg1: TTestRecord10): TTestRecord10;
  616. end;
  617. {$M-}
  618. TTestInterfaceClass = class(TInterfacedObject, ITestInterface)
  619. private
  620. procedure Test1;
  621. function Test2: SizeInt;
  622. function Test3(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt;
  623. procedure Test4(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString);
  624. function Test5: AnsiString;
  625. function Test6: UnicodeString;
  626. function Test7: WideString;
  627. function Test8: ShortString;
  628. procedure Test9(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt);
  629. procedure Test10(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString);
  630. procedure Test11(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString);
  631. procedure Test12(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt);
  632. function TestRecSize1(aArg1: TTestRecord1): TTestRecord1;
  633. function TestRecSize2(aArg1: TTestRecord2): TTestRecord2;
  634. function TestRecSize3(aArg1: TTestRecord3): TTestRecord3;
  635. function TestRecSize4(aArg1: TTestRecord4): TTestRecord4;
  636. function TestRecSize5(aArg1: TTestRecord5): TTestRecord5;
  637. function TestRecSize6(aArg1: TTestRecord6): TTestRecord6;
  638. function TestRecSize7(aArg1: TTestRecord7): TTestRecord7;
  639. function TestRecSize8(aArg1: TTestRecord8): TTestRecord8;
  640. function TestRecSize9(aArg1: TTestRecord9): TTestRecord9;
  641. function TestRecSize10(aArg1: TTestRecord10): TTestRecord10;
  642. public
  643. InputArgs: array of TValue;
  644. OutputArgs: array of TValue;
  645. ResultValue: TValue;
  646. CalledMethod: SizeInt;
  647. InOutMapping: array of SizeInt;
  648. procedure Reset;
  649. public class var
  650. ProcVarInst: TTestInterfaceClass;
  651. ProcVarRecInst: TTestInterfaceClass;
  652. public const
  653. RecSizeMarker = SizeInt($80000000);
  654. end;
  655. TMethodTest1 = procedure of object;
  656. TMethodTest2 = function: SizeInt of object;
  657. TMethodTest3 = function(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt of object;
  658. TMethodTest4 = procedure(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString) of object;
  659. TMethodTest5 = function: AnsiString of object;
  660. TMethodTest6 = function: UnicodeString of object;
  661. TMethodTest7 = function: WideString of object;
  662. TMethodTest8 = function: ShortString of object;
  663. TMethodTest9 = procedure(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt) of object;
  664. TMethodTest10 = procedure(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString) of object;
  665. TMethodTest11 = procedure(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString) of object;
  666. TMethodTest12 = procedure(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt) of object;
  667. TMethodTestRecSize1 = function(aArg1: TTestRecord1): TTestRecord1 of object;
  668. TMethodTestRecSize2 = function(aArg1: TTestRecord2): TTestRecord2 of object;
  669. TMethodTestRecSize3 = function(aArg1: TTestRecord3): TTestRecord3 of object;
  670. TMethodTestRecSize4 = function(aArg1: TTestRecord4): TTestRecord4 of object;
  671. TMethodTestRecSize5 = function(aArg1: TTestRecord5): TTestRecord5 of object;
  672. TMethodTestRecSize6 = function(aArg1: TTestRecord6): TTestRecord6 of object;
  673. TMethodTestRecSize7 = function(aArg1: TTestRecord7): TTestRecord7 of object;
  674. TMethodTestRecSize8 = function(aArg1: TTestRecord8): TTestRecord8 of object;
  675. TMethodTestRecSize9 = function(aArg1: TTestRecord9): TTestRecord9 of object;
  676. TMethodTestRecSize10 = function(aArg1: TTestRecord10): TTestRecord10 of object;
  677. TProcVarTest1 = procedure;
  678. TProcVarTest2 = function: SizeInt;
  679. TProcVarTest3 = function(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt;
  680. TProcVarTest4 = procedure(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString);
  681. TProcVarTest5 = function: AnsiString;
  682. TProcVarTest6 = function: UnicodeString;
  683. TProcVarTest7 = function: WideString;
  684. TProcVarTest8 = function: ShortString;
  685. TProcVarTest9 = procedure(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt);
  686. TProcVarTest10 = procedure(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString);
  687. TProcVarTest11 = procedure(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString);
  688. TProcVarTest12 = procedure(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt);
  689. TProcVarTestRecSize1 = function(aArg1: TTestRecord1): TTestRecord1;
  690. TProcVarTestRecSize2 = function(aArg1: TTestRecord2): TTestRecord2;
  691. TProcVarTestRecSize3 = function(aArg1: TTestRecord3): TTestRecord3;
  692. TProcVarTestRecSize4 = function(aArg1: TTestRecord4): TTestRecord4;
  693. TProcVarTestRecSize5 = function(aArg1: TTestRecord5): TTestRecord5;
  694. TProcVarTestRecSize6 = function(aArg1: TTestRecord6): TTestRecord6;
  695. TProcVarTestRecSize7 = function(aArg1: TTestRecord7): TTestRecord7;
  696. TProcVarTestRecSize8 = function(aArg1: TTestRecord8): TTestRecord8;
  697. TProcVarTestRecSize9 = function(aArg1: TTestRecord9): TTestRecord9;
  698. TProcVarTestRecSize10 = function(aArg1: TTestRecord10): TTestRecord10;
  699. procedure TTestInterfaceClass.Test1;
  700. begin
  701. SetLength(InputArgs, 0);
  702. SetLength(OutputArgs, 0);
  703. ResultValue := TValue.Empty;
  704. CalledMethod := 1;
  705. end;
  706. function TTestInterfaceClass.Test2: SizeInt;
  707. begin
  708. SetLength(InputArgs, 0);
  709. SetLength(OutputArgs, 0);
  710. Result := 42;
  711. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  712. CalledMethod := 2;
  713. end;
  714. function TTestInterfaceClass.Test3(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt;
  715. begin
  716. SetLength(InputArgs, 10);
  717. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  718. TValue.Make(@aArg2, TypeInfo(aArg2), InputArgs[1]);
  719. TValue.Make(@aArg3, TypeInfo(aArg3), InputArgs[2]);
  720. TValue.Make(@aArg4, TypeInfo(aArg4), InputArgs[3]);
  721. TValue.Make(@aArg5, TypeInfo(aArg5), InputArgs[4]);
  722. TValue.Make(@aArg6, TypeInfo(aArg6), InputArgs[5]);
  723. TValue.Make(@aArg7, TypeInfo(aArg7), InputArgs[6]);
  724. TValue.Make(@aArg8, TypeInfo(aArg8), InputArgs[7]);
  725. TValue.Make(@aArg9, TypeInfo(aArg9), InputArgs[8]);
  726. TValue.Make(@aArg10, TypeInfo(aArg10), InputArgs[9]);
  727. SetLength(OutputArgs, 0);
  728. Result := 42;
  729. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  730. CalledMethod := 3;
  731. end;
  732. procedure TTestInterfaceClass.Test4(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString);
  733. begin
  734. SetLength(InputArgs, 4);
  735. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  736. TValue.Make(@aArg2, TypeInfo(aArg2), InputArgs[1]);
  737. TValue.Make(@aArg3, TypeInfo(aArg3), InputArgs[2]);
  738. TValue.Make(@aArg4, TypeInfo(aArg4), InputArgs[3]);
  739. SetLength(OutputArgs, 0);
  740. ResultValue := TValue.Empty;
  741. CalledMethod := 4;
  742. end;
  743. function TTestInterfaceClass.Test5: AnsiString;
  744. begin
  745. SetLength(InputArgs, 0);
  746. SetLength(OutputArgs, 0);
  747. Result := 'Hello World';
  748. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  749. CalledMethod := 5;
  750. end;
  751. function TTestInterfaceClass.Test6: UnicodeString;
  752. begin
  753. SetLength(InputArgs, 0);
  754. SetLength(OutputArgs, 0);
  755. Result := 'Hello World';
  756. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  757. CalledMethod := 6;
  758. end;
  759. function TTestInterfaceClass.Test7: WideString;
  760. begin
  761. SetLength(InputArgs, 0);
  762. SetLength(OutputArgs, 0);
  763. Result := 'Hello World';
  764. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  765. CalledMethod := 7;
  766. end;
  767. function TTestInterfaceClass.Test8: ShortString;
  768. begin
  769. SetLength(InputArgs, 0);
  770. SetLength(OutputArgs, 0);
  771. Result := 'Hello World';
  772. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  773. CalledMethod := 8;
  774. end;
  775. procedure TTestInterfaceClass.Test9(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt);
  776. begin
  777. SetLength(InputArgs, 4);
  778. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  779. TValue.Make(@aArg2, TypeInfo(aArg2), InputArgs[1]);
  780. TValue.Make(@aArg3, TypeInfo(aArg3), InputArgs[2]);
  781. TValue.Make(@aArg4, TypeInfo(aArg4), InputArgs[3]);
  782. aArg2 := $1234;
  783. aArg3 := $5678;
  784. SetLength(OutputArgs, 2);
  785. TValue.Make(@aArg2, TypeInfo(aArg2), OutputArgs[0]);
  786. TValue.Make(@aArg3, TypeInfo(aArg3), OutputArgs[1]);
  787. SetLength(InOutMapping, 2);
  788. InOutMapping[0] := 1;
  789. InOutMapping[1] := 2;
  790. ResultValue := TValue.Empty;
  791. CalledMethod := 9;
  792. end;
  793. procedure TTestInterfaceClass.Test10(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString);
  794. begin
  795. SetLength(InputArgs, 4);
  796. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  797. TValue.Make(@aArg2, TypeInfo(aArg2), InputArgs[1]);
  798. TValue.Make(@aArg3, TypeInfo(aArg3), InputArgs[2]);
  799. TValue.Make(@aArg4, TypeInfo(aArg4), InputArgs[3]);
  800. aArg2 := 'Foo';
  801. aArg3 := 'Bar';
  802. SetLength(OutputArgs, 2);
  803. TValue.Make(@aArg2, TypeInfo(aArg2), OutputArgs[0]);
  804. TValue.Make(@aArg3, TypeInfo(aArg3), OutputArgs[1]);
  805. SetLength(InOutMapping, 2);
  806. InOutMapping[0] := 1;
  807. InOutMapping[1] := 2;
  808. ResultValue := TValue.Empty;
  809. CalledMethod := 10;
  810. end;
  811. procedure TTestInterfaceClass.Test11(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString);
  812. begin
  813. SetLength(InputArgs, 4);
  814. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  815. TValue.Make(@aArg2, TypeInfo(aArg2), InputArgs[1]);
  816. TValue.Make(@aArg3, TypeInfo(aArg3), InputArgs[2]);
  817. TValue.Make(@aArg4, TypeInfo(aArg4), InputArgs[3]);
  818. aArg2 := 'Foo';
  819. aArg3 := 'Bar';
  820. SetLength(OutputArgs, 2);
  821. TValue.Make(@aArg2, TypeInfo(aArg2), OutputArgs[0]);
  822. TValue.Make(@aArg3, TypeInfo(aArg3), OutputArgs[1]);
  823. SetLength(InOutMapping, 2);
  824. InOutMapping[0] := 1;
  825. InOutMapping[1] := 2;
  826. ResultValue := TValue.Empty;
  827. CalledMethod := 11;
  828. end;
  829. procedure TTestInterfaceClass.Test12(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt);
  830. {$ifdef fpc}
  831. var
  832. i: SizeInt;
  833. start: SizeInt;
  834. {$endif}
  835. begin
  836. {$ifdef fpc}
  837. SetLength(InputArgs, 4);
  838. InputArgs[0] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg1);
  839. InputArgs[1] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg2);
  840. InputArgs[2] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg3);
  841. InputArgs[3] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg4);
  842. SetLength(OutputArgs, 2);
  843. start := $4321;
  844. for i := 0 to High(aArg2) do
  845. aArg2[i] := start + i;
  846. start := $9876;
  847. for i := 0 to High(aArg3) do
  848. aArg3[i] := start + i;
  849. OutputArgs[0] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg2);
  850. OutputArgs[1] := specialize OpenArrayToDynArrayValue<SizeInt>(aArg3);
  851. SetLength(InOutMapping, 2);
  852. InOutMapping[0] := 1;
  853. InOutMapping[1] := 2;
  854. ResultValue := TValue.Empty;
  855. CalledMethod := 12;
  856. {$endif}
  857. end;
  858. function TTestInterfaceClass.TestRecSize1(aArg1: TTestRecord1): TTestRecord1;
  859. var
  860. i: LongInt;
  861. begin
  862. SetLength(InputArgs, 1);
  863. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  864. SetLength(OutputArgs, 0);
  865. for i := 0 to High(aArg1.b) do
  866. Result.b[High(Result.b) - i] := aArg1.b[i];
  867. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  868. CalledMethod := 1 or RecSizeMarker;
  869. end;
  870. function TTestInterfaceClass.TestRecSize2(aArg1: TTestRecord2): TTestRecord2;
  871. var
  872. i: LongInt;
  873. begin
  874. SetLength(InputArgs, 1);
  875. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  876. SetLength(OutputArgs, 0);
  877. for i := 0 to High(aArg1.b) do
  878. Result.b[High(Result.b) - i] := aArg1.b[i];
  879. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  880. CalledMethod := 2 or RecSizeMarker;
  881. end;
  882. function TTestInterfaceClass.TestRecSize3(aArg1: TTestRecord3): TTestRecord3;
  883. var
  884. i: LongInt;
  885. begin
  886. SetLength(InputArgs, 1);
  887. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  888. SetLength(OutputArgs, 0);
  889. for i := 0 to High(aArg1.b) do
  890. Result.b[High(Result.b) - i] := aArg1.b[i];
  891. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  892. CalledMethod := 3 or RecSizeMarker;
  893. end;
  894. function TTestInterfaceClass.TestRecSize4(aArg1: TTestRecord4): TTestRecord4;
  895. var
  896. i: LongInt;
  897. begin
  898. SetLength(InputArgs, 1);
  899. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  900. SetLength(OutputArgs, 0);
  901. for i := 0 to High(aArg1.b) do
  902. Result.b[High(Result.b) - i] := aArg1.b[i];
  903. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  904. CalledMethod := 4 or RecSizeMarker;
  905. end;
  906. function TTestInterfaceClass.TestRecSize5(aArg1: TTestRecord5): TTestRecord5;
  907. var
  908. i: LongInt;
  909. begin
  910. SetLength(InputArgs, 1);
  911. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  912. SetLength(OutputArgs, 0);
  913. for i := 0 to High(aArg1.b) do
  914. Result.b[High(Result.b) - i] := aArg1.b[i];
  915. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  916. CalledMethod := 5 or RecSizeMarker;
  917. end;
  918. function TTestInterfaceClass.TestRecSize6(aArg1: TTestRecord6): TTestRecord6;
  919. var
  920. i: LongInt;
  921. begin
  922. SetLength(InputArgs, 1);
  923. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  924. SetLength(OutputArgs, 0);
  925. for i := 0 to High(aArg1.b) do
  926. Result.b[High(Result.b) - i] := aArg1.b[i];
  927. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  928. CalledMethod := 6 or RecSizeMarker;
  929. end;
  930. function TTestInterfaceClass.TestRecSize7(aArg1: TTestRecord7): TTestRecord7;
  931. var
  932. i: LongInt;
  933. begin
  934. SetLength(InputArgs, 1);
  935. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  936. SetLength(OutputArgs, 0);
  937. for i := 0 to High(aArg1.b) do
  938. Result.b[High(Result.b) - i] := aArg1.b[i];
  939. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  940. CalledMethod := 7 or RecSizeMarker;
  941. end;
  942. function TTestInterfaceClass.TestRecSize8(aArg1: TTestRecord8): TTestRecord8;
  943. var
  944. i: LongInt;
  945. begin
  946. SetLength(InputArgs, 1);
  947. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  948. SetLength(OutputArgs, 0);
  949. for i := 0 to High(aArg1.b) do
  950. Result.b[High(Result.b) - i] := aArg1.b[i];
  951. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  952. CalledMethod := 8 or RecSizeMarker;
  953. end;
  954. function TTestInterfaceClass.TestRecSize9(aArg1: TTestRecord9): TTestRecord9;
  955. var
  956. i: LongInt;
  957. begin
  958. SetLength(InputArgs, 1);
  959. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  960. SetLength(OutputArgs, 0);
  961. for i := 0 to High(aArg1.b) do
  962. Result.b[High(Result.b) - i] := aArg1.b[i];
  963. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  964. CalledMethod := 9 or RecSizeMarker;
  965. end;
  966. function TTestInterfaceClass.TestRecSize10(aArg1: TTestRecord10): TTestRecord10;
  967. var
  968. i: LongInt;
  969. begin
  970. SetLength(InputArgs, 1);
  971. TValue.Make(@aArg1, TypeInfo(aArg1), InputArgs[0]);
  972. SetLength(OutputArgs, 0);
  973. for i := 0 to High(aArg1.b) do
  974. Result.b[High(Result.b) - i] := aArg1.b[i];
  975. TValue.Make(@Result, TypeInfo(Result), ResultValue);
  976. CalledMethod := 10 or RecSizeMarker;
  977. end;
  978. procedure TTestInterfaceClass.Reset;
  979. begin
  980. InputArgs := Nil;
  981. OutputArgs := Nil;
  982. InOutMapping := Nil;
  983. ResultValue := TValue.Empty;
  984. CalledMethod := 0;
  985. end;
  986. procedure ProcTest1;
  987. begin
  988. TTestInterfaceClass.ProcVarInst.Test1;
  989. end;
  990. function ProcTest2: SizeInt;
  991. begin
  992. Result := TTestInterfaceClass.ProcVarInst.Test2;
  993. end;
  994. function ProcTest3(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10: SizeInt): SizeInt;
  995. begin
  996. Result := TTestInterfaceClass.ProcVarInst.Test3(aArg1, aArg2, aArg3, aArg4, aArg5, aArg6, aArg7, aArg8, aArg9, aArg10);
  997. end;
  998. procedure ProcTest4(aArg1: AnsiString; aArg2: UnicodeString; aArg3: WideString; aArg4: ShortString);
  999. begin
  1000. TTestInterfaceClass.ProcVarInst.Test4(aArg1, aArg2, aArg3, aArg4);
  1001. end;
  1002. function ProcTest5: AnsiString;
  1003. begin
  1004. Result := TTestInterfaceClass.ProcVarInst.Test5;
  1005. end;
  1006. function ProcTest6: UnicodeString;
  1007. begin
  1008. Result := TTestInterfaceClass.ProcVarInst.Test6;
  1009. end;
  1010. function ProcTest7: WideString;
  1011. begin
  1012. Result := TTestInterfaceClass.ProcVarInst.Test7;
  1013. end;
  1014. function ProcTest8: ShortString;
  1015. begin
  1016. Result := TTestInterfaceClass.ProcVarInst.Test8;
  1017. end;
  1018. procedure ProcTest9(aArg1: SizeInt; var aArg2: SizeInt; out aArg3: SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: SizeInt);
  1019. begin
  1020. TTestInterfaceClass.ProcVarInst.Test9(aArg1, aArg2, aArg3, aArg4);
  1021. end;
  1022. procedure ProcTest10(aArg1: AnsiString; var aArg2: AnsiString; out aArg3: AnsiString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: AnsiString);
  1023. begin
  1024. TTestInterfaceClass.ProcVarInst.Test10(aArg1, aArg2, aArg3, aArg4);
  1025. end;
  1026. procedure ProcTest11(aArg1: ShortString; var aArg2: ShortString; out aArg3: ShortString; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: ShortString);
  1027. begin
  1028. TTestInterfaceClass.ProcVarInst.Test11(aArg1, aArg2, aArg3, aArg4);
  1029. end;
  1030. procedure ProcTest12(aArg1: array of SizeInt; var aArg2: array of SizeInt; out aArg3: array of SizeInt; {$ifdef fpc}constref{$else}const [ref]{$endif} aArg4: array of SizeInt);
  1031. begin
  1032. TTestInterfaceClass.ProcVarInst.Test12(aArg1, aArg2, aArg3, aArg4);
  1033. end;
  1034. function ProcTestRecSize1(aArg1: TTestRecord1): TTestRecord1;
  1035. begin
  1036. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize1(aArg1);
  1037. end;
  1038. function ProcTestRecSize2(aArg1: TTestRecord2): TTestRecord2;
  1039. begin
  1040. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize2(aArg1);
  1041. end;
  1042. function ProcTestRecSize3(aArg1: TTestRecord3): TTestRecord3;
  1043. begin
  1044. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize3(aArg1);
  1045. end;
  1046. function ProcTestRecSize4(aArg1: TTestRecord4): TTestRecord4;
  1047. begin
  1048. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize4(aArg1);
  1049. end;
  1050. function ProcTestRecSize5(aArg1: TTestRecord5): TTestRecord5;
  1051. begin
  1052. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize5(aArg1);
  1053. end;
  1054. function ProcTestRecSize6(aArg1: TTestRecord6): TTestRecord6;
  1055. begin
  1056. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize6(aArg1);
  1057. end;
  1058. function ProcTestRecSize7(aArg1: TTestRecord7): TTestRecord7;
  1059. begin
  1060. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize7(aArg1);
  1061. end;
  1062. function ProcTestRecSize8(aArg1: TTestRecord8): TTestRecord8;
  1063. begin
  1064. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize8(aArg1);
  1065. end;
  1066. function ProcTestRecSize9(aArg1: TTestRecord9): TTestRecord9;
  1067. begin
  1068. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize9(aArg1);
  1069. end;
  1070. function ProcTestRecSize10(aArg1: TTestRecord10): TTestRecord10;
  1071. begin
  1072. Result := TTestInterfaceClass.ProcVarRecInst.TestRecSize10(aArg1);
  1073. end;
  1074. function CopyValue({$ifdef fpc}constref{$else}const [ref]{$endif} aValue: TValue): TValue;
  1075. var
  1076. arrptr: Pointer;
  1077. len, i: SizeInt;
  1078. begin
  1079. if aValue.Kind = tkDynArray then begin
  1080. { we need to decouple the source reference, so we're going to be a bit
  1081. cheeky here }
  1082. len := aValue.GetArrayLength;
  1083. arrptr := Nil;
  1084. DynArraySetLength(arrptr, aValue.TypeInfo, 1, @len);
  1085. TValue.Make(@arrptr, aValue.TypeInfo, Result);
  1086. for i := 0 to len - 1 do
  1087. Result.SetArrayElement(i, aValue.GetArrayElement(i));
  1088. end else
  1089. TValue.Make(aValue.GetReferenceToRawData, aValue.TypeInfo, Result);
  1090. end;
  1091. procedure TTestInvoke.DoIntfInvoke(aIndex: SizeInt; aInputArgs,
  1092. aOutputArgs: TValueArray; aResult: TValue);
  1093. var
  1094. cls: TTestInterfaceClass;
  1095. intf: ITestInterface;
  1096. name: String;
  1097. context: TRttiContext;
  1098. t: TRttiType;
  1099. inst, res: TValue;
  1100. method: TRttiMethod;
  1101. i: SizeInt;
  1102. input: array of TValue;
  1103. begin
  1104. cls := TTestInterfaceClass.Create;
  1105. intf := cls;
  1106. TValue.Make(@intf, TypeInfo(intf), inst);
  1107. if aIndex and TTestInterfaceClass.RecSizeMarker <> 0 then
  1108. name := 'TestRecSize' + IntToStr(aIndex and not TTestInterfaceClass.RecSizeMarker)
  1109. else
  1110. name := 'Test' + IntToStr(aIndex);
  1111. context := TRttiContext.Create;
  1112. try
  1113. t := context.GetType(TypeInfo(ITestInterface));
  1114. method := t.GetMethod(name);
  1115. Check(Assigned(method), 'Method not found: ' + name);
  1116. { arguments might be modified by Invoke (Note: Copy() does not uniquify the
  1117. IValueData of managed types) }
  1118. SetLength(input, Length(aInputArgs));
  1119. for i := 0 to High(input) do
  1120. input[i] := CopyValue(aInputArgs[i]);
  1121. try
  1122. res := method.Invoke(inst, aInputArgs);
  1123. except
  1124. DumpExceptionBacktrace(output);
  1125. raise;
  1126. end;
  1127. CheckEquals(aIndex, cls.CalledMethod, 'Wrong method called for ' + name);
  1128. Check(EqualValues(cls.ResultValue, res), 'Reported result value differs from returned for ' + name);
  1129. Check(EqualValues(aResult, res), 'Expected result value differs from returned for ' + name);
  1130. CheckEquals(Length(aInputArgs), Length(cls.InputArgs), 'Count of input args differs for ' + name);
  1131. CheckEquals(Length(cls.OutputArgs), Length(cls.InOutMapping), 'Count of output args and in-out-mapping differs for ' + name);
  1132. CheckEquals(Length(aOutputArgs), Length(cls.OutputArgs), 'Count of output args differs for ' + name);
  1133. for i := 0 to High(aInputArgs) do begin
  1134. Check(EqualValues(input[i], cls.InputArgs[i]), Format('Input argument %d differs for %s', [i + 1, name]));
  1135. end;
  1136. for i := 0 to High(aOutputArgs) do begin
  1137. Check(EqualValues(aOutputArgs[i], cls.OutputArgs[i]), Format('Output argument %d differs for %s', [i + 1, name]));
  1138. Check(EqualValues(aOutputArgs[i], aInputArgs[cls.InOutMapping[i]]), Format('New output argument %d differs from expected output for %s', [i + 1, name]));
  1139. end;
  1140. finally
  1141. context.Free;
  1142. end;
  1143. end;
  1144. procedure TTestInvoke.DoMethodInvoke(aInst: TObject; aMethod: TMethod;
  1145. aTypeInfo: PTypeInfo; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  1146. var
  1147. cls: TTestInterfaceClass;
  1148. name: String;
  1149. context: TRttiContext;
  1150. t: TRttiType;
  1151. callable, res: TValue;
  1152. method: TRttiMethodType;
  1153. i: SizeInt;
  1154. input: array of TValue;
  1155. begin
  1156. cls := aInst as TTestInterfaceClass;
  1157. cls.Reset;
  1158. if aIndex and TTestInterfaceClass.RecSizeMarker <> 0 then
  1159. name := 'TestRecSize' + IntToStr(aIndex and not TTestInterfaceClass.RecSizeMarker)
  1160. else
  1161. name := 'Test' + IntToStr(aIndex);
  1162. TValue.Make(@aMethod, aTypeInfo, callable);
  1163. context := TRttiContext.Create;
  1164. try
  1165. t := context.GetType(aTypeInfo);
  1166. Check(t is TRttiMethodType, 'Not a method variable: ' + aTypeInfo^.Name);
  1167. method := t as TRttiMethodType;
  1168. { arguments might be modified by Invoke (Note: Copy() does not uniquify the
  1169. IValueData of managed types) }
  1170. SetLength(input, Length(aInputArgs));
  1171. for i := 0 to High(input) do
  1172. input[i] := CopyValue(aInputArgs[i]);
  1173. res := method.Invoke(callable, aInputArgs);
  1174. CheckEquals(aIndex, cls.CalledMethod, 'Wrong method called for ' + name);
  1175. Check(EqualValues(cls.ResultValue, res), 'Reported result value differs from returned for ' + name);
  1176. Check(EqualValues(aResult, res), 'Expected result value differs from returned for ' + name);
  1177. CheckEquals(Length(aInputArgs), Length(cls.InputArgs), 'Count of input args differs for ' + name);
  1178. CheckEquals(Length(cls.OutputArgs), Length(cls.InOutMapping), 'Count of output args and in-out-mapping differs for ' + name);
  1179. CheckEquals(Length(aOutputArgs), Length(cls.OutputArgs), 'Count of output args differs for ' + name);
  1180. for i := 0 to High(aInputArgs) do begin
  1181. Check(EqualValues(input[i], cls.InputArgs[i]), Format('Input argument %d differs for %s', [i + 1, name]));
  1182. end;
  1183. for i := 0 to High(aOutputArgs) do begin
  1184. Check(EqualValues(aOutputArgs[i], cls.OutputArgs[i]), Format('Output argument %d differs for %s', [i + 1, name]));
  1185. Check(EqualValues(aOutputArgs[i], aInputArgs[cls.InOutMapping[i]]), Format('New output argument %d differs from expected output for %s', [i + 1, name]));
  1186. end;
  1187. finally
  1188. context.Free;
  1189. end;
  1190. end;
  1191. procedure TTestInvoke.DoProcVarInvoke(aInst: TObject; aProc: CodePointer;
  1192. aTypeInfo: PTypeInfo; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  1193. var
  1194. cls: TTestInterfaceClass;
  1195. name: String;
  1196. context: TRttiContext;
  1197. t: TRttiType;
  1198. callable, res: TValue;
  1199. proc: TRttiProcedureType;
  1200. i: SizeInt;
  1201. input: array of TValue;
  1202. begin
  1203. cls := aInst as TTestInterfaceClass;
  1204. cls.Reset;
  1205. if aIndex and TTestInterfaceClass.RecSizeMarker <> 0 then begin
  1206. name := 'TestRecSize' + IntToStr(aIndex and not TTestInterfaceClass.RecSizeMarker);
  1207. TTestInterfaceClass.ProcVarRecInst := cls;
  1208. end else begin
  1209. name := 'Test' + IntToStr(aIndex);
  1210. TTestInterfaceClass.ProcVarInst := cls;
  1211. end;
  1212. TValue.Make(@aProc, aTypeInfo, callable);
  1213. context := TRttiContext.Create;
  1214. try
  1215. t := context.GetType(aTypeInfo);
  1216. Check(t is TRttiProcedureType, 'Not a procedure variable: ' + aTypeInfo^.Name);
  1217. proc := t as TRttiProcedureType;
  1218. { arguments might be modified by Invoke (Note: Copy() does not uniquify the
  1219. IValueData of managed types) }
  1220. SetLength(input, Length(aInputArgs));
  1221. for i := 0 to High(input) do
  1222. input[i] := CopyValue(aInputArgs[i]);
  1223. res := proc.Invoke(callable, aInputArgs);
  1224. CheckEquals(aIndex, cls.CalledMethod, 'Wrong method called for ' + name);
  1225. Check(EqualValues(cls.ResultValue, res), 'Reported result value differs from returned for ' + name);
  1226. Check(EqualValues(aResult, res), 'Expected result value differs from returned for ' + name);
  1227. CheckEquals(Length(aInputArgs), Length(cls.InputArgs), 'Count of input args differs for ' + name);
  1228. CheckEquals(Length(cls.OutputArgs), Length(cls.InOutMapping), 'Count of output args and in-out-mapping differs for ' + name);
  1229. CheckEquals(Length(aOutputArgs), Length(cls.OutputArgs), 'Count of output args differs for ' + name);
  1230. for i := 0 to High(aInputArgs) do begin
  1231. Check(EqualValues(input[i], cls.InputArgs[i]), Format('Input argument %d differs for %s', [i + 1, name]));
  1232. end;
  1233. for i := 0 to High(aOutputArgs) do begin
  1234. Check(EqualValues(aOutputArgs[i], cls.OutputArgs[i]), Format('Output argument %d differs for %s', [i + 1, name]));
  1235. Check(EqualValues(aOutputArgs[i], aInputArgs[cls.InOutMapping[i]]), Format('New output argument %d differs from expected output for %s', [i + 1, name]));
  1236. end;
  1237. finally
  1238. context.Free;
  1239. end;
  1240. end;
  1241. {$ifndef InLazIDE}
  1242. {$ifdef fpc}generic{$endif} procedure TTestInvoke.GenDoMethodInvoke<T>(aInst: TObject; aMethod: T; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  1243. begin
  1244. DoMethodInvoke(aInst, TMethod(aMethod), TypeInfo(T), aIndex, aInputArgs, aOutputArgs, aResult);
  1245. end;
  1246. {$ifdef fpc}generic{$endif} procedure TTestInvoke.GenDoProcVarInvoke<T>(aInst: TObject; aProc: T; aIndex: SizeInt; aInputArgs, aOutputArgs: TValueArray; aResult: TValue);
  1247. begin
  1248. DoProcVarInvoke(aInst, CodePointer(aProc), TypeInfo(T), aIndex, aInputArgs, aOutputArgs, aResult);
  1249. end;
  1250. {$ifdef fpc}generic{$endif} function TTestInvoke.GetRecValue<T>(aReverse: Boolean): TValue;
  1251. var
  1252. i: LongInt;
  1253. arr: array of Byte;
  1254. begin
  1255. SetLength(arr, SizeOf(T));
  1256. RandSeed := $54827982;
  1257. if not aReverse then begin
  1258. for i := 0 to High(arr) do
  1259. arr[i] := Random($ff);
  1260. end else begin
  1261. for i := High(arr) downto 0 do
  1262. arr[i] := Random($ff);
  1263. end;
  1264. TValue.Make(@arr[0], PTypeInfo(TypeInfo(T)), Result);
  1265. end;
  1266. {$endif}
  1267. function GetIntValue(aValue: SizeInt): TValue;
  1268. begin
  1269. Result := TValue.{$ifdef fpc}specialize{$endif}From<SizeInt>(aValue);
  1270. end;
  1271. function GetAnsiString(const aValue: AnsiString): TValue;
  1272. begin
  1273. Result := TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>(aValue);
  1274. end;
  1275. function GetShortString(const aValue: ShortString): TValue;
  1276. begin
  1277. Result := TValue.{$ifdef fpc}specialize{$endif}From<ShortString>(aValue);
  1278. end;
  1279. {$ifdef fpc}
  1280. function GetArray(const aArg: array of SizeInt): TValue;
  1281. begin
  1282. Result := specialize OpenArrayToDynArrayValue<SizeInt>(aArg);
  1283. end;
  1284. {$endif}
  1285. procedure TTestInvoke.TestIntfMethods;
  1286. begin
  1287. DoIntfInvoke(1, [], [], TValue.Empty);
  1288. DoIntfInvoke(2, [], [], TValue.{$ifdef fpc}specialize{$endif}From<SizeInt>(42));
  1289. DoIntfInvoke(3, [
  1290. GetIntValue(7), GetIntValue(2), GetIntValue(5), GetIntValue(1), GetIntValue(10), GetIntValue(8), GetIntValue(6), GetIntValue(3), GetIntValue(9), GetIntValue(3)
  1291. ], [], GetIntValue(42));
  1292. DoIntfInvoke(4, [
  1293. TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Alpha'),
  1294. TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Beta'),
  1295. TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Gamma'),
  1296. TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Delta')
  1297. ], [], TValue.Empty);
  1298. DoIntfInvoke(5, [], [], TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Hello World'));
  1299. DoIntfInvoke(6, [], [], TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Hello World'));
  1300. DoIntfInvoke(7, [], [], TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Hello World'));
  1301. DoIntfInvoke(8, [], [], TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Hello World'));
  1302. DoIntfInvoke(9, [
  1303. GetIntValue($1234), GetIntValue($4321), GetIntValue($8765), GetIntValue($5678)
  1304. ], [
  1305. GetIntValue($1234), GetIntValue($5678)
  1306. ], TValue.Empty);
  1307. DoIntfInvoke(10, [
  1308. GetAnsiString('Alpha'), GetAnsiString('Beta'), GetAnsiString(''), GetAnsiString('Delta')
  1309. ], [
  1310. GetAnsiString('Foo'), GetAnsiString('Bar')
  1311. ], TValue.Empty);
  1312. DoIntfInvoke(11, [
  1313. GetShortString('Alpha'), GetShortString('Beta'), GetShortString(''), GetShortString('Delta')
  1314. ], [
  1315. GetShortString('Foo'), GetShortString('Bar')
  1316. ], TValue.Empty);
  1317. {$ifdef fpc}
  1318. DoIntfInvoke(12, [
  1319. GetArray([$1234, $2345, $3456, $4567]), GetArray([$4321, $5431, $6543, $7654]), GetArray([$5678, $6789, $7890, $8901]), GetArray([$8765, $7654, $6543, $5432])
  1320. ], [
  1321. GetArray([$4321, $4322, $4323, $4324]), GetArray([$9876, $9877, $9878, $9879])
  1322. ], TValue.Empty);
  1323. {$endif}
  1324. end;
  1325. procedure TTestInvoke.TestIntfMethodsRecs;
  1326. begin
  1327. DoIntfInvoke(1 or TTestInterfaceClass.RecSizeMarker,
  1328. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(False)], [],
  1329. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(True));
  1330. DoIntfInvoke(2 or TTestInterfaceClass.RecSizeMarker,
  1331. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(False)], [],
  1332. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(True));
  1333. DoIntfInvoke(3 or TTestInterfaceClass.RecSizeMarker,
  1334. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(False)], [],
  1335. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(True));
  1336. DoIntfInvoke(4 or TTestInterfaceClass.RecSizeMarker,
  1337. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(False)], [],
  1338. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(True));
  1339. DoIntfInvoke(5 or TTestInterfaceClass.RecSizeMarker,
  1340. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(False)], [],
  1341. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(True));
  1342. DoIntfInvoke(6 or TTestInterfaceClass.RecSizeMarker,
  1343. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(False)], [],
  1344. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(True));
  1345. DoIntfInvoke(7 or TTestInterfaceClass.RecSizeMarker,
  1346. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(False)], [],
  1347. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(True));
  1348. DoIntfInvoke(8 or TTestInterfaceClass.RecSizeMarker,
  1349. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(False)], [],
  1350. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(True));
  1351. DoIntfInvoke(9 or TTestInterfaceClass.RecSizeMarker,
  1352. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(False)], [],
  1353. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(True));
  1354. DoIntfInvoke(10 or TTestInterfaceClass.RecSizeMarker,
  1355. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(False)], [],
  1356. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(True));
  1357. end;
  1358. procedure TTestInvoke.TestMethodVars;
  1359. var
  1360. cls: TTestInterfaceClass;
  1361. begin
  1362. cls := TTestInterfaceClass.Create;
  1363. try
  1364. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest1>(cls, {$ifdef fpc}@{$endif}cls.Test1, 1, [], [], TValue.Empty);
  1365. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest2>(cls, {$ifdef fpc}@{$endif}cls.Test2, 2, [], [], TValue.{$ifdef fpc}{$ifdef fpc}specialize{$endif}{$endif}From<SizeInt>(42));
  1366. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest3>(cls, {$ifdef fpc}@{$endif}cls.Test3, 3, [
  1367. GetIntValue(7), GetIntValue(2), GetIntValue(5), GetIntValue(1), GetIntValue(10), GetIntValue(8), GetIntValue(6), GetIntValue(3), GetIntValue(9), GetIntValue(3)
  1368. ], [], GetIntValue(42));
  1369. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest4>(cls, {$ifdef fpc}@{$endif}cls.Test4, 4, [
  1370. TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Alpha'),
  1371. TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Beta'),
  1372. TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Gamma'),
  1373. TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Delta')
  1374. ], [], TValue.Empty);
  1375. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest5>(cls, {$ifdef fpc}@{$endif}cls.Test5, 5, [], [], TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Hello World'));
  1376. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest6>(cls, {$ifdef fpc}@{$endif}cls.Test6, 6, [], [], TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Hello World'));
  1377. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest7>(cls, {$ifdef fpc}@{$endif}cls.Test7, 7, [], [], TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Hello World'));
  1378. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest8>(cls, {$ifdef fpc}@{$endif}cls.Test8, 8, [], [], TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Hello World'));
  1379. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest9>(cls, {$ifdef fpc}@{$endif}cls.Test9, 9, [
  1380. GetIntValue($1234), GetIntValue($4321), GetIntValue($8765), GetIntValue($5678)
  1381. ], [
  1382. GetIntValue($1234), GetIntValue($5678)
  1383. ], TValue.Empty);
  1384. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest10>(cls, {$ifdef fpc}@{$endif}cls.Test10, 10, [
  1385. GetAnsiString('Alpha'), GetAnsiString('Beta'), GetAnsiString(''), GetAnsiString('Delta')
  1386. ], [
  1387. GetAnsiString('Foo'), GetAnsiString('Bar')
  1388. ], TValue.Empty);
  1389. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTest11>(cls, {$ifdef fpc}@{$endif}cls.Test11, 11, [
  1390. GetShortString('Alpha'), GetShortString('Beta'), GetShortString(''), GetShortString('Delta')
  1391. ], [
  1392. GetShortString('Foo'), GetShortString('Bar')
  1393. ], TValue.Empty);
  1394. {$ifdef fpc}
  1395. specialize GenDoMethodInvoke<TMethodTest12>(cls, {$ifdef fpc}@{$endif}cls.Test12, 12, [
  1396. GetArray([$1234, $2345, $3456, $4567]), GetArray([$4321, $5431, $6543, $7654]), GetArray([$5678, $6789, $7890, $8901]), GetArray([$8765, $7654, $6543, $5432])
  1397. ], [
  1398. GetArray([$4321, $4322, $4323, $4324]), GetArray([$9876, $9877, $9878, $9879])
  1399. ], TValue.Empty);
  1400. {$endif}
  1401. finally
  1402. cls.Free;
  1403. end;
  1404. end;
  1405. procedure TTestInvoke.TestMethodVarsRecs;
  1406. var
  1407. cls: TTestInterfaceClass;
  1408. begin
  1409. cls := TTestInterfaceClass.Create;
  1410. try
  1411. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize1>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize1, 1 or TTestInterfaceClass.RecSizeMarker,
  1412. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(False)], [],
  1413. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(True));
  1414. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize2>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize2, 2 or TTestInterfaceClass.RecSizeMarker,
  1415. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(False)], [],
  1416. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(True));
  1417. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize3>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize3, 3 or TTestInterfaceClass.RecSizeMarker,
  1418. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(False)], [],
  1419. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(True));
  1420. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize4>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize4, 4 or TTestInterfaceClass.RecSizeMarker,
  1421. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(False)], [],
  1422. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(True));
  1423. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize5>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize5, 5 or TTestInterfaceClass.RecSizeMarker,
  1424. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(False)], [],
  1425. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(True));
  1426. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize6>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize6, 6 or TTestInterfaceClass.RecSizeMarker,
  1427. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(False)], [],
  1428. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(True));
  1429. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize7>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize7, 7 or TTestInterfaceClass.RecSizeMarker,
  1430. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(False)], [],
  1431. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(True));
  1432. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize8>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize8, 8 or TTestInterfaceClass.RecSizeMarker,
  1433. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(False)], [],
  1434. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(True));
  1435. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize9>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize9, 9 or TTestInterfaceClass.RecSizeMarker,
  1436. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(False)], [],
  1437. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(True));
  1438. {$ifdef fpc}specialize{$endif} GenDoMethodInvoke<TMethodTestRecSize10>(cls, {$ifdef fpc}@{$endif}cls.TestRecSize10, 10 or TTestInterfaceClass.RecSizeMarker,
  1439. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(False)], [],
  1440. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(True));
  1441. finally
  1442. cls.Free;
  1443. end;
  1444. end;
  1445. procedure TTestInvoke.TestProcVars;
  1446. var
  1447. cls: TTestInterfaceClass;
  1448. begin
  1449. cls := TTestInterfaceClass.Create;
  1450. try
  1451. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest1>(cls, {$ifdef fpc}@{$endif}ProcTest1, 1, [], [], TValue.Empty);
  1452. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest2>(cls, {$ifdef fpc}@{$endif}ProcTest2, 2, [], [], TValue.{$ifdef fpc}{$ifdef fpc}specialize{$endif}{$endif}From<SizeInt>(42));
  1453. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest3>(cls, {$ifdef fpc}@{$endif}ProcTest3, 3, [
  1454. GetIntValue(7), GetIntValue(2), GetIntValue(5), GetIntValue(1), GetIntValue(10), GetIntValue(8), GetIntValue(6), GetIntValue(3), GetIntValue(9), GetIntValue(3)
  1455. ], [], GetIntValue(42));
  1456. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest4>(cls, {$ifdef fpc}@{$endif}ProcTest4, 4, [
  1457. TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Alpha'),
  1458. TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Beta'),
  1459. TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Gamma'),
  1460. TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Delta')
  1461. ], [], TValue.Empty);
  1462. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest5>(cls, {$ifdef fpc}@{$endif}ProcTest5, 5, [], [], TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>('Hello World'));
  1463. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest6>(cls, {$ifdef fpc}@{$endif}ProcTest6, 6, [], [], TValue.{$ifdef fpc}specialize{$endif}From<UnicodeString>('Hello World'));
  1464. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest7>(cls, {$ifdef fpc}@{$endif}ProcTest7, 7, [], [], TValue.{$ifdef fpc}specialize{$endif}From<WideString>('Hello World'));
  1465. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest8>(cls, {$ifdef fpc}@{$endif}ProcTest8, 8, [], [], TValue.{$ifdef fpc}specialize{$endif}From<ShortString>('Hello World'));
  1466. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest9>(cls, {$ifdef fpc}@{$endif}ProcTest9, 9, [
  1467. GetIntValue($1234), GetIntValue($4321), GetIntValue($8765), GetIntValue($5678)
  1468. ], [
  1469. GetIntValue($1234), GetIntValue($5678)
  1470. ], TValue.Empty);
  1471. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest10>(cls, {$ifdef fpc}@{$endif}ProcTest10, 10, [
  1472. GetAnsiString('Alpha'), GetAnsiString('Beta'), GetAnsiString(''), GetAnsiString('Delta')
  1473. ], [
  1474. GetAnsiString('Foo'), GetAnsiString('Bar')
  1475. ], TValue.Empty);
  1476. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTest11>(cls, {$ifdef fpc}@{$endif}ProcTest11, 11, [
  1477. GetShortString('Alpha'), GetShortString('Beta'), GetShortString(''), GetShortString('Delta')
  1478. ], [
  1479. GetShortString('Foo'), GetShortString('Bar')
  1480. ], TValue.Empty);
  1481. {$ifdef fpc}
  1482. specialize GenDoProcVarInvoke<TProcVarTest12>(cls, {$ifdef fpc}@{$endif}ProcTest12, 12, [
  1483. GetArray([$1234, $2345, $3456, $4567]), GetArray([$4321, $5431, $6543, $7654]), GetArray([$5678, $6789, $7890, $8901]), GetArray([$8765, $7654, $6543, $5432])
  1484. ], [
  1485. GetArray([$4321, $4322, $4323, $4324]), GetArray([$9876, $9877, $9878, $9879])
  1486. ], TValue.Empty);
  1487. {$endif}
  1488. finally
  1489. cls.Free;
  1490. end;
  1491. end;
  1492. procedure TTestInvoke.TestProcVarsRecs;
  1493. var
  1494. cls: TTestInterfaceClass;
  1495. begin
  1496. cls := TTestInterfaceClass.Create;
  1497. try
  1498. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize1>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize1, 1 or TTestInterfaceClass.RecSizeMarker,
  1499. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(False)], [],
  1500. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord1>(True));
  1501. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize2>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize2, 2 or TTestInterfaceClass.RecSizeMarker,
  1502. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(False)], [],
  1503. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord2>(True));
  1504. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize3>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize3, 3 or TTestInterfaceClass.RecSizeMarker,
  1505. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(False)], [],
  1506. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord3>(True));
  1507. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize4>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize4, 4 or TTestInterfaceClass.RecSizeMarker,
  1508. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(False)], [],
  1509. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord4>(True));
  1510. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize5>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize5, 5 or TTestInterfaceClass.RecSizeMarker,
  1511. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(False)], [],
  1512. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord5>(True));
  1513. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize6>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize6, 6 or TTestInterfaceClass.RecSizeMarker,
  1514. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(False)], [],
  1515. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord6>(True));
  1516. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize7>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize7, 7 or TTestInterfaceClass.RecSizeMarker,
  1517. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(False)], [],
  1518. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord7>(True));
  1519. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize8>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize8, 8 or TTestInterfaceClass.RecSizeMarker,
  1520. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(False)], [],
  1521. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord8>(True));
  1522. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize9>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize9, 9 or TTestInterfaceClass.RecSizeMarker,
  1523. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(False)], [],
  1524. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord9>(True));
  1525. {$ifdef fpc}specialize{$endif} GenDoProcVarInvoke<TProcVarTestRecSize10>(cls, {$ifdef fpc}@{$endif}ProcTestRecSize10, 10 or TTestInterfaceClass.RecSizeMarker,
  1526. [{$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(False)], [],
  1527. {$ifdef fpc}specialize{$endif} GetRecValue<TTestRecord10>(True));
  1528. finally
  1529. cls.Free;
  1530. end;
  1531. end;
  1532. begin
  1533. {$ifdef fpc}
  1534. RegisterTest(TTestInvoke);
  1535. {$else fpc}
  1536. RegisterTest(TTestInvoke.Suite);
  1537. {$endif fpc}
  1538. end.