2
0

objpas.inc 58 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918
  1. {
  2. This file is part of the Free Pascal run time library.
  3. Copyright (c) 1999-2000 by the Free Pascal development team
  4. This unit makes Free Pascal as much as possible Delphi compatible
  5. See the file COPYING.FPC, included in this distribution,
  6. for details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************}
  11. {$ifdef FPC_HAS_FEATURE_VARIANTS}
  12. procedure DoDispCallByIDError(res : Pointer; const disp : IDispatch;desc : PDispDesc; params : Pointer);
  13. begin
  14. handleerroraddrframeind(RuntimeErrorExitCodes[reVarDispatch],
  15. get_pc_addr,get_frame);
  16. end;
  17. procedure fpc_dispatch_by_id(Result: Pointer; const Dispatch: pointer;
  18. DispDesc: Pointer; Params: Pointer); compilerproc;
  19. type
  20. TDispProc = procedure(res : Pointer; const disp : IDispatch;desc : PDispDesc; params : Pointer);
  21. begin
  22. TDispProc(DispCallByIDProc)(Result,IDispatch(Dispatch),DispDesc,Params);
  23. end;
  24. {$endif FPC_HAS_FEATURE_VARIANTS}
  25. {****************************************************************************
  26. Internal Routines called from the Compiler
  27. ****************************************************************************}
  28. { the reverse order of the parameters make code generation easier }
  29. function fpc_do_is(aclass : tclass;aobject : tobject) : boolean;[public,alias: 'FPC_DO_IS']; compilerproc;
  30. begin
  31. fpc_do_is:=assigned(aobject) and assigned(aclass) and
  32. aobject.inheritsfrom(aclass);
  33. end;
  34. { the reverse order of the parameters make code generation easier }
  35. function fpc_do_as(aclass : tclass;aobject : tobject): tobject;[public,alias: 'FPC_DO_AS']; compilerproc;
  36. begin
  37. if assigned(aobject) and not(aobject.inheritsfrom(aclass)) then
  38. handleerroraddrframeInd(219,get_pc_addr,get_frame);
  39. result := aobject;
  40. end;
  41. { interface helpers }
  42. procedure fpc_intf_decr_ref(var i: pointer);[public,alias: 'FPC_INTF_DECR_REF']; compilerproc;
  43. var
  44. iFetch: pointer;
  45. begin
  46. iFetch:=i;
  47. if assigned(iFetch) then
  48. begin
  49. i:=nil;
  50. IUnknown(iFetch)._Release;
  51. end;
  52. end;
  53. { local declaration for intf_decr_ref for local access }
  54. procedure intf_decr_ref(var i: pointer); [external name 'FPC_INTF_DECR_REF'];
  55. procedure fpc_intf_incr_ref(i: pointer);[public,alias: 'FPC_INTF_INCR_REF']; compilerproc;
  56. begin
  57. if assigned(i) then
  58. IUnknown(i)._AddRef;
  59. end;
  60. { local declaration of intf_incr_ref for local access }
  61. procedure intf_incr_ref(i: pointer); [external name 'FPC_INTF_INCR_REF'];
  62. procedure fpc_intf_assign(var D: pointer; const S: pointer);[public,alias: 'FPC_INTF_ASSIGN']; compilerproc;
  63. begin
  64. if assigned(S) then
  65. IUnknown(S)._AddRef;
  66. if assigned(D) then
  67. IUnknown(D)._Release;
  68. D:=S;
  69. end;
  70. procedure fpc_intf_assign(var D: pointer; const s: pointer); [external name 'FPC_INTF_ASSIGN'];
  71. {procedure fpc_intf_assign_by_iid(var D: pointer; const S: pointer; const iid: TGUID);[public,alias: 'FPC_INTF_ASSIGN2']; compilerproc;
  72. var
  73. tmp : pointer;
  74. begin
  75. if assigned(S) then
  76. begin
  77. tmp:=nil;
  78. if IUnknown(S).QueryInterface(iid,tmp)<>S_OK then
  79. handleerror(219);
  80. if assigned(D) then
  81. IUnknown(D)._Release;
  82. D:=tmp;
  83. end
  84. else
  85. begin
  86. if assigned(D) then
  87. IUnknown(D)._Release;
  88. D:=nil;
  89. end;
  90. end;}
  91. function fpc_intf_is(const S: pointer; const iid: TGUID): Boolean;[public,alias: 'FPC_INTF_IS']; compilerproc;
  92. var
  93. tmpi: pointer;
  94. begin
  95. tmpi:=nil;
  96. fpc_intf_is:=Assigned(S) and (IUnknown(S).QueryInterface(iid,tmpi)=S_OK);
  97. if Assigned(tmpi) then
  98. IUnknown(tmpi)._Release;
  99. end;
  100. function fpc_intf_is_class(const S: pointer; const aclass: tclass): Boolean;[public,alias: 'FPC_INTF_IS_CLASS']; compilerproc;
  101. var
  102. tmpo: tobject;
  103. begin
  104. fpc_intf_is_class:=Assigned(S) and (IUnknown(S).QueryInterface(IObjectInstance,tmpo)=S_OK) and tmpo.InheritsFrom(aclass);
  105. end;
  106. function fpc_class_is_intf(const S: pointer; const iid: TGUID): Boolean;[public,alias: 'FPC_CLASS_IS_INTF']; compilerproc;
  107. var
  108. tmpi: pointer;
  109. tmpi2: pointer; // weak!
  110. begin
  111. tmpi:=nil;
  112. tmpi2:=nil;
  113. fpc_class_is_intf:=Assigned(S) and ((TObject(S).GetInterfaceWeak(IUnknown,tmpi2) and (IUnknown(tmpi2).QueryInterface(IID,tmpi)=S_OK)) or
  114. TObject(S).GetInterface(IID,tmpi));
  115. if Assigned(tmpi) then
  116. IUnknown(tmpi)._Release;
  117. end;
  118. function fpc_class_is_corbaintf(const S: pointer; const iid: Shortstring): Boolean;[public,alias: 'FPC_CLASS_IS_CORBAINTF']; compilerproc;
  119. begin
  120. fpc_class_is_corbaintf:=Assigned(S) and Assigned(TObject(S).GetInterfaceEntryByStr(iid));
  121. end;
  122. function fpc_intf_cast(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_INTF_CAST']; compilerproc;
  123. var
  124. tmpi: pointer;
  125. begin
  126. tmpi:=nil;
  127. if Assigned(S) and (IUnknown(S).QueryInterface(iid,tmpi)=S_OK) then
  128. pointer(fpc_intf_cast):=tmpi
  129. else
  130. fpc_intf_cast:= nil;
  131. end;
  132. function fpc_intf_cast_class(const S: pointer; const aclass: tclass): pointer;[public,alias: 'FPC_INTF_CAST_CLASS']; compilerproc;
  133. var
  134. tmpo: tobject;
  135. begin
  136. if Assigned(S) and (IUnknown(S).QueryInterface(IObjectInstance,tmpo)=S_OK) and tmpo.InheritsFrom(aclass) then
  137. fpc_intf_cast_class:=tmpo
  138. else
  139. fpc_intf_cast_class:=nil;
  140. end;
  141. function fpc_class_cast_intf(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_CLASS_CAST_INTF']; compilerproc;
  142. var
  143. tmpi: pointer;
  144. tmpi2: pointer; // weak!
  145. begin
  146. tmpi:=nil;
  147. tmpi2:=nil;
  148. if Assigned(S) and ((TObject(S).GetInterfaceWeak(IUnknown,tmpi2) and (IUnknown(tmpi2).QueryInterface(IID,tmpi)=S_OK)) or
  149. TObject(S).GetInterface(IID,tmpi)) then
  150. begin
  151. // decrease reference count
  152. fpc_class_cast_intf:=nil;
  153. pointer(fpc_class_cast_intf):=tmpi
  154. end
  155. else
  156. fpc_class_cast_intf:=nil;
  157. end;
  158. function fpc_class_cast_corbaintf(const S: pointer; const iid: Shortstring): Pointer;[public,alias: 'FPC_CLASS_CAST_CORBAINTF']; compilerproc;
  159. var
  160. tmpi: pointer;
  161. begin
  162. if Assigned(S) and TObject(S).GetInterface(iid,tmpi) then
  163. fpc_class_cast_corbaintf:=tmpi
  164. else
  165. fpc_class_cast_corbaintf:=nil;
  166. end;
  167. function fpc_intf_as(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_INTF_AS']; compilerproc;
  168. var
  169. tmpi: pointer; // _AddRef before _Release
  170. begin
  171. if assigned(S) then
  172. begin
  173. tmpi:=nil;
  174. if IUnknown(S).QueryInterface(iid,tmpi)<>S_OK then
  175. handleerror(219);
  176. // decrease reference count
  177. fpc_intf_as:=nil;
  178. pointer(fpc_intf_as):=tmpi;
  179. end
  180. else
  181. fpc_intf_as:=nil;
  182. end;
  183. function fpc_intf_as_class(const S: pointer; const aclass: tclass): pointer;[public,alias: 'FPC_INTF_AS_CLASS']; compilerproc;
  184. var
  185. tmpo: tobject;
  186. begin
  187. if assigned(S) then
  188. begin
  189. if not ((IUnknown(S).QueryInterface(IObjectInstance,tmpo)=S_OK) and tmpo.inheritsfrom(aclass)) then
  190. handleerror(219);
  191. fpc_intf_as_class:=tmpo;
  192. end
  193. else
  194. fpc_intf_as_class:=nil;
  195. end;
  196. function fpc_class_as_intf(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_CLASS_AS_INTF']; compilerproc;
  197. var
  198. tmpi: pointer; // _AddRef before _Release
  199. tmpi2: pointer; // weak!
  200. begin
  201. if assigned(S) then
  202. begin
  203. tmpi:=nil;
  204. tmpi2:=nil;
  205. if not ((TObject(S).GetInterfaceWeak(IUnknown,tmpi2) and (IUnknown(tmpi2).QueryInterface(IID,tmpi)=S_OK)) or TObject(S).GetInterface(IID,tmpi)) then
  206. handleerror(219);
  207. // decrease reference count
  208. fpc_class_as_intf:=nil;
  209. pointer(fpc_class_as_intf):=tmpi;
  210. end
  211. else
  212. fpc_class_as_intf:=nil;
  213. end;
  214. function fpc_class_as_corbaintf(const S: pointer; const iid: Shortstring): Pointer;[public,alias: 'FPC_CLASS_AS_CORBAINTF']; compilerproc;
  215. var
  216. tmpi: pointer; // _AddRef before _Release
  217. begin
  218. if assigned(S) then
  219. begin
  220. tmpi:=nil;
  221. if not TObject(S).GetInterface(iid,tmpi) then
  222. handleerror(219);
  223. fpc_class_as_corbaintf:=tmpi;
  224. end
  225. else
  226. fpc_class_as_corbaintf:=nil;
  227. end;
  228. {****************************************************************************
  229. TVMT
  230. ****************************************************************************}
  231. function TVmt.GetvParent: PVmt;
  232. begin
  233. if Assigned(vParentRef) then
  234. GetvParent:=vParentRef^
  235. else
  236. GetvParent:=Nil;
  237. end;
  238. {****************************************************************************
  239. TGUID
  240. ****************************************************************************}
  241. class operator TGUID.=(const aLeft, aRight: TGUID): Boolean;
  242. var
  243. P1,P2 : ^Cardinal;
  244. begin
  245. P1:=PCardinal(@aLeft);
  246. P2:=PCardinal(@aRight);
  247. Result:=(P1[0]=P2[0]) and (P1[1]=P2[1]) and (P1[2]=P2[2]) and (P1[3]=P2[3]);
  248. end;
  249. class operator TGUID.<>(const aLeft, aRight: TGUID): Boolean;
  250. begin
  251. Result:=Not (aLeft=aRight);
  252. end;
  253. class function TGUID.Empty: TGUID; static;
  254. begin
  255. Result:=Default(TGUID);
  256. end;
  257. class function TGUID.Create(const aData; aBigEndian: Boolean = False): TGUID; overload; static;
  258. begin
  259. Result:=Create(PByte(@aData),aBigEndian);
  260. end;
  261. class function TGUID.Create(const aData : PByte; aBigEndian: Boolean = False): TGUID; overload; static;
  262. const
  263. SysBigendian = {$IFDEF FPC_LITTLE_ENDIAN} false {$ELSE} true {$ENDIF};
  264. begin
  265. Result := PGuid(aData)^;
  266. if (aBigEndian=SysBigEndian) then
  267. exit;
  268. Result.D1:=SwapEndian(Result.D1);
  269. Result.D2:=SwapEndian(Result.D2);
  270. Result.D3:=SwapEndian(Result.D3);
  271. end;
  272. class function TGUID.Create(const aData: array of Byte; aStartIndex: Cardinal; aBigEndian: Boolean = False): TGUID; overload; static;
  273. begin
  274. if ((Length(aData)-aStartIndex)<16) then
  275. Result:=Empty
  276. else
  277. Result:=Create(PByte(@aData[aStartIndex]),aBigEndian);
  278. end;
  279. function TGUID.IsEmpty: Boolean;
  280. var
  281. P : ^Cardinal;
  282. begin
  283. P:=PCardinal(@Self);
  284. Result:=(P[0]=0) and (P[1]=0) and (P[2]=0) and (P[3]=0)
  285. end;
  286. {****************************************************************************
  287. TINTERFACEENTRY
  288. ****************************************************************************}
  289. function tinterfaceentry.GetIID: pguid;
  290. begin
  291. if Assigned(IIDRef) then
  292. GetIID:=IIDRef^
  293. else
  294. GetIID:=Nil;
  295. end;
  296. function tinterfaceentry.GetIIDStr: pshortstring;
  297. begin
  298. if Assigned(IIDStrRef) then
  299. GetIIDStr:=IIDStrRef^
  300. else
  301. GetIIDStr:=Nil;
  302. end;
  303. {****************************************************************************
  304. TOBJECT
  305. ****************************************************************************}
  306. constructor TObject.Create;
  307. begin
  308. end;
  309. destructor TObject.Destroy;
  310. begin
  311. end;
  312. procedure TObject.Free;
  313. begin
  314. // the call via self avoids a warning
  315. if self<>nil then
  316. self.destroy;
  317. end;
  318. class function TObject.InstanceSize : SizeInt;
  319. begin
  320. InstanceSize := PVmt(Self)^.vInstanceSize;
  321. end;
  322. procedure InitInterfacePointers(objclass: tclass;instance : pointer);
  323. var
  324. ovmt: PVmt;
  325. i: longint;
  326. intftable: pinterfacetable;
  327. Res: pinterfaceentry;
  328. begin
  329. ovmt := PVmt(objclass);
  330. while assigned(ovmt) and assigned(ovmt^.vIntfTable) do
  331. begin
  332. intftable:=ovmt^.vIntfTable;
  333. i:=intftable^.EntryCount;
  334. Res:=@intftable^.Entries[0];
  335. while i>0 do begin
  336. if Res^.IType = etStandard then
  337. ppointer(@(pbyte(instance)[Res^.IOffset]))^:=
  338. pointer(Res^.VTable);
  339. inc(Res);
  340. dec(i);
  341. end;
  342. ovmt:=ovmt^.vParent;
  343. end;
  344. end;
  345. class function TObject.InitInstance(instance : pointer) : tobject;
  346. var
  347. vmt : PVmt;
  348. inittable : pointer;
  349. {$ifdef FPC_HAS_FEATURE_RTTI}
  350. mopinittable : PRTTIRecordOpOffsetTable;
  351. {$endif def FPC_HAS_FEATURE_RTTI}
  352. i : longint;
  353. begin
  354. { the size is saved at offset 0 }
  355. fillchar(instance^, InstanceSize, 0);
  356. { insert VMT pointer into the new created memory area }
  357. { (in class methods self contains the VMT!) }
  358. {$IFNDEF SYSTEM_HAS_FEATURE_MONITOR}
  359. ppointer(instance)^:=pointer(self);
  360. {$ELSE}
  361. {$IFDEF VER3_2}
  362. // In 3.2.x Compiler (used during bootstrap) still inserts VMT at offset...
  363. ppointer(PByte(instance)+SizeOf(Pointer))^:=pointer(self);
  364. {$ELSE}
  365. // As of 3.3.x compiler forces insert of VMT at first pos.
  366. ppointer(instance)^:=pointer(self);
  367. {$ENDIF}
  368. {$ENDIF}
  369. if assigned(PVmt(self)^.vIntfTable) then
  370. InitInterfacePointers(self,instance);
  371. {$ifdef FPC_HAS_FEATURE_RTTI}
  372. { for management operators like initialize call int_initialize }
  373. vmt := PVmt(self);
  374. if assigned(vmt) then
  375. begin
  376. inittable:=vmt^.vInitTable;
  377. if assigned(inittable) then
  378. begin
  379. mopinittable:=RTTIRecordMopInitTable(inittable);
  380. if assigned(mopinittable) then
  381. begin
  382. {$push}
  383. { ensure that no range check errors pop up with the [0..0] array }
  384. {$R-}
  385. for i:=0 to mopinittable^.Count-1 do
  386. TRTTIRecVarOp(mopinittable^.Entries[i].ManagmentOperator)(PByte(Instance)+mopinittable^.Entries[i].FieldOffset);
  387. {$pop}
  388. end;
  389. end;
  390. end;
  391. {$endif def FPC_HAS_FEATURE_RTTI}
  392. InitInstance:=TObject(Instance);
  393. end;
  394. class function TObject.ClassParent : tclass;
  395. begin
  396. { type of self is class of tobject => it points to the vmt }
  397. { the parent vmt is saved at offset vmtParent }
  398. classparent:=tclass(PVmt(Self)^.vParent);
  399. end;
  400. class function TObject.NewInstance : tobject;
  401. var
  402. p : pointer;
  403. begin
  404. getmem(p, InstanceSize);
  405. if p <> nil then
  406. InitInstance(p);
  407. NewInstance:=TObject(p);
  408. end;
  409. procedure TObject.FreeInstance;
  410. begin
  411. CleanupInstance;
  412. FreeMem(Pointer(Self));
  413. end;
  414. class function TObject.ClassType : TClass;
  415. begin
  416. ClassType:=TClass(Pointer(Self))
  417. end;
  418. type
  419. {$PUSH}
  420. {$PACKRECORDS NORMAL}
  421. tmethodnamerec =
  422. {$IFNDEF FPC_REQUIRES_PROPER_ALIGNMENT}
  423. packed
  424. {$ENDIF FPC_REQUIRES_PROPER_ALIGNMENT}
  425. record
  426. name : pshortstring;
  427. addr : codepointer;
  428. end;
  429. tmethodnametable =
  430. {$IFNDEF FPC_REQUIRES_PROPER_ALIGNMENT}
  431. packed
  432. {$ENDIF FPC_REQUIRES_PROPER_ALIGNMENT}
  433. record
  434. count : dword;
  435. entries : packed array[0..0] of tmethodnamerec;
  436. end;
  437. {$POP}
  438. pmethodnametable = ^tmethodnametable;
  439. class function TObject.MethodAddress(const name : shortstring) : codepointer;
  440. var
  441. methodtable : pmethodnametable;
  442. i : longint; // in case count=0
  443. ovmt : PVmt;
  444. begin
  445. ovmt:=PVmt(self);
  446. while assigned(ovmt) do
  447. begin
  448. methodtable:=pmethodnametable(ovmt^.vMethodTable);
  449. if assigned(methodtable) then
  450. begin
  451. for i:=0 to methodtable^.count-1 do
  452. if ShortCompareText(methodtable^.entries[i].name^, name)=0 then
  453. begin
  454. MethodAddress:=methodtable^.entries[i].addr;
  455. exit;
  456. end;
  457. end;
  458. ovmt := ovmt^.vParent;
  459. end;
  460. MethodAddress:=nil;
  461. end;
  462. class function TObject.MethodName(address : codepointer) : shortstring;
  463. var
  464. methodtable : pmethodnametable;
  465. i : longint; // in case count=0
  466. ovmt : PVmt;
  467. begin
  468. ovmt:=PVmt(self);
  469. while assigned(ovmt) do
  470. begin
  471. methodtable:=pmethodnametable(ovmt^.vMethodTable);
  472. if assigned(methodtable) then
  473. begin
  474. for i:=0 to methodtable^.count-1 do
  475. if methodtable^.entries[i].addr=address then
  476. begin
  477. MethodName:=methodtable^.entries[i].name^;
  478. exit;
  479. end;
  480. end;
  481. ovmt := ovmt^.vParent;
  482. end;
  483. MethodName:='';
  484. end;
  485. function TObject.FieldAddress(const name : shortstring) : pointer;
  486. {The following is copied to the typinfo unit. If it is changed here, change it there as well ! }
  487. type
  488. {$PUSH}
  489. {$PACKRECORDS NORMAL}
  490. PFieldInfo = ^TFieldInfo;
  491. TFieldInfo =
  492. {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
  493. packed
  494. {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
  495. record
  496. FieldOffset: SizeUInt;
  497. ClassTypeIndex: Word;
  498. Name: ShortString;
  499. end;
  500. PFieldTable = ^TFieldTable;
  501. TFieldTable =
  502. {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
  503. packed
  504. {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
  505. record
  506. FieldCount: Word;
  507. ClassTable: Pointer;
  508. { should be array[Word] of TFieldInfo; but
  509. Elements have variant size! force at least proper alignment }
  510. Fields: array[0..0] of TFieldInfo
  511. end;
  512. {$POP}
  513. var
  514. ovmt: PVmt;
  515. FieldTable: PFieldTable;
  516. FieldInfo: PFieldInfo;
  517. i: longint;
  518. begin
  519. if Length(name) > 0 then
  520. begin
  521. ovmt := PVmt(ClassType);
  522. while ovmt <> nil do
  523. begin
  524. FieldTable := PFieldTable(ovmt^.vFieldTable);
  525. if FieldTable <> nil then
  526. begin
  527. FieldInfo := @FieldTable^.Fields[0];
  528. for i := 0 to FieldTable^.FieldCount - 1 do
  529. begin
  530. if ShortCompareText(FieldInfo^.Name, name) = 0 then
  531. begin
  532. fieldaddress := Pointer(Self) + FieldInfo^.FieldOffset;
  533. exit;
  534. end;
  535. FieldInfo := PFieldInfo(PByte(@FieldInfo^.Name) + 1 + Length(FieldInfo^.Name));
  536. {$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
  537. { align to largest field of TFieldInfo }
  538. FieldInfo := Align(FieldInfo, SizeOf(PtrUInt));
  539. {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
  540. end;
  541. end;
  542. { Try again with the parent class type }
  543. ovmt:=ovmt^.vParent;
  544. end;
  545. end;
  546. fieldaddress:=nil;
  547. end;
  548. function TObject.SafeCallException(exceptobject : tobject;
  549. exceptaddr : codepointer) : HResult;
  550. begin
  551. safecallexception:=E_UNEXPECTED;
  552. end;
  553. class function TObject.ClassInfo : pointer;
  554. begin
  555. ClassInfo := PVmt(Self)^.vTypeInfo;
  556. end;
  557. class function TObject.ClassName : ShortString;
  558. begin
  559. ClassName := PVmt(Self)^.vClassName^;
  560. end;
  561. class function TObject.ClassNameIs(const name : RTLString) : boolean;
  562. var
  563. SS : ShortString;
  564. begin
  565. SS:=ShortString(Name);
  566. ClassNameIs:=ShortCompareText(PVmt(Self)^.vClassName^, SS) = 0;
  567. end;
  568. class function TObject.InheritsFrom(aclass : TClass) : Boolean;
  569. var
  570. vmt: PVmt;
  571. begin
  572. if assigned(aclass) then
  573. begin
  574. vmt:=PVmt(self);
  575. while assigned(vmt) and (vmt <> PVmt(aclass)) do
  576. vmt := vmt^.vParent;
  577. InheritsFrom := (vmt = PVmt(aclass));
  578. end
  579. else
  580. inheritsFrom := False;
  581. end;
  582. class function TObject.stringmessagetable : pstringmessagetable;
  583. begin
  584. stringmessagetable:=PVmt(Self)^.vMsgStrPtr;
  585. end;
  586. type
  587. tmessagehandler = procedure(var msg) of object;
  588. procedure TObject.Dispatch(var message);
  589. type
  590. {$PUSH}
  591. {$PACKRECORDS NORMAL}
  592. PMsgIntTable = ^TMsgIntTable;
  593. TMsgIntTable = record
  594. index : dword;
  595. method : codepointer;
  596. end;
  597. PMsgInt = ^TMsgInt;
  598. TMsgInt = record
  599. count : longint;
  600. msgs : array[0..0] of TMsgIntTable;
  601. end;
  602. {$POP}
  603. var
  604. index : dword;
  605. count,i : longint;
  606. msgtable : PMsgIntTable;
  607. p : PMsgInt;
  608. ovmt : PVmt;
  609. msghandler : tmessagehandler;
  610. begin
  611. index:=dword(message);
  612. ovmt := PVmt(ClassType);
  613. while assigned(ovmt) do
  614. begin
  615. // See if we have messages at all in this class.
  616. p:=PMsgInt(ovmt^.vDynamicTable);
  617. If Assigned(p) then
  618. begin
  619. msgtable:=@p^.msgs;
  620. count:=p^.count;
  621. end
  622. else
  623. Count:=0;
  624. { later, we can implement a binary search here }
  625. for i:=0 to count-1 do
  626. begin
  627. if index=msgtable[i].index then
  628. begin
  629. TMethod(msghandler).Code:=msgtable[i].method;
  630. TMethod(msghandler).Data:=self;
  631. msghandler(message);
  632. exit;
  633. end;
  634. end;
  635. ovmt:=ovmt^.vParent;
  636. end;
  637. DefaultHandler(message);
  638. end;
  639. procedure TObject.DispatchStr(var message);
  640. var
  641. name : shortstring;
  642. count,i : longint;
  643. msgstrtable : pmsgstrtable;
  644. p: pstringmessagetable;
  645. ovmt : PVmt;
  646. msghandler : tmessagehandler;
  647. begin
  648. name:=pshortstring(@message)^;
  649. ovmt:=PVmt(ClassType);
  650. while assigned(ovmt) do
  651. begin
  652. p := ovmt^.vMsgStrPtr;
  653. if (P<>Nil) and (p^.count<>0) then
  654. begin
  655. count:=p^.count;
  656. msgstrtable:=@p^.msgstrtable;
  657. end
  658. else
  659. Count:=0;
  660. { later, we can implement a binary search here }
  661. for i:=0 to count-1 do
  662. begin
  663. if name=msgstrtable[i].name^ then
  664. begin
  665. TMethod(msghandler).Code:=msgstrtable[i].method;
  666. TMethod(msghandler).Data:=self;
  667. msghandler(message);
  668. exit;
  669. end;
  670. end;
  671. ovmt:=ovmt^.vParent;
  672. end;
  673. DefaultHandlerStr(message);
  674. end;
  675. procedure TObject.DefaultHandler(var message);
  676. begin
  677. end;
  678. procedure TObject.DefaultHandlerStr(var message);
  679. begin
  680. end;
  681. procedure TObject.CleanupInstance;
  682. var
  683. vmt : PVmt;
  684. temp : pointer;
  685. begin
  686. vmt := PVmt(ClassType);
  687. while vmt<>nil do
  688. begin
  689. Temp:= vmt^.vInitTable;
  690. {$ifdef FPC_HAS_FEATURE_RTTI}
  691. { The RTTI format matches one for records, except the type is tkClass.
  692. Since RecordRTTI does not check the type, calling it yields the desired result. }
  693. if Assigned(Temp) then
  694. RecordRTTI(Self,aligntoqword(Temp+2+PByte(Temp)[1]),@int_finalize);
  695. {$endif def FPC_HAS_FEATURE_RTTI}
  696. vmt:= vmt^.vParent;
  697. end;
  698. {$IFDEF SYSTEM_HAS_FEATURE_MONITOR}
  699. if Assigned(_MonitorData) then
  700. TMonitor.FreeMonitorData(_MonitorData);
  701. {$ENDIF}
  702. end;
  703. procedure TObject.AfterConstruction;
  704. begin
  705. end;
  706. procedure TObject.BeforeDestruction;
  707. begin
  708. end;
  709. function IsGUIDEqual(const guid1, guid2: tguid): boolean;
  710. begin
  711. IsGUIDEqual:=
  712. (guid1.D1=guid2.D1) and
  713. (PDWORD(@guid1.D2)^=PDWORD(@guid2.D2)^) and
  714. (PDWORD(@guid1.D4[0])^=PDWORD(@guid2.D4[0])^) and
  715. (PDWORD(@guid1.D4[4])^=PDWORD(@guid2.D4[4])^);
  716. end;
  717. // Use of managed types should be avoided here; implicit _Addref/_Release
  718. // will end up in unpredictable behaviour if called on CORBA interfaces.
  719. type
  720. TInterfaceGetter = procedure(out Obj) of object;
  721. TClassGetter = function: TObject of object;
  722. function GetInterfaceByEntry(Instance: pointer; IEntry: pinterfaceentry; out obj): boolean;
  723. var
  724. Getter: TMethod;
  725. begin
  726. Pointer(Obj) := nil;
  727. Getter.Data := Instance;
  728. if Assigned(IEntry) and Assigned(Instance) then
  729. begin
  730. case IEntry^.IType of
  731. etStandard:
  732. Pointer(Obj) := PByte(instance)+IEntry^.IOffset;
  733. etFieldValue, etFieldValueClass:
  734. Pointer(obj) := PPointer(PByte(Instance)+IEntry^.IOffset)^;
  735. etVirtualMethodResult:
  736. begin
  737. // IOffset is relative to the VMT, not to instance.
  738. Getter.code := PCodePointer(PByte(PPointer(Instance)^) + IEntry^.IOffset)^;
  739. TInterfaceGetter(Getter)(obj);
  740. end;
  741. etVirtualMethodClass:
  742. begin
  743. // IOffset is relative to the VMT, not to instance.
  744. Getter.code := PCodePointer(PByte(PPointer(Instance)^) + IEntry^.IOffset)^;
  745. TObject(obj) := TClassGetter(Getter)();
  746. end;
  747. etStaticMethodResult:
  748. begin
  749. Getter.code := IEntry^.IOffsetAsCodePtr;
  750. TInterfaceGetter(Getter)(obj);
  751. end;
  752. etStaticMethodClass:
  753. begin
  754. Getter.code := IEntry^.IOffsetAsCodePtr;
  755. TObject(obj) := TClassGetter(Getter)();
  756. end;
  757. end;
  758. end;
  759. result := assigned(pointer(obj));
  760. end;
  761. function TObject.GetInterface(const iid : tguid;out obj) : boolean;
  762. var
  763. IEntry: PInterfaceEntry;
  764. Instance: TObject;
  765. begin
  766. if IsGUIDEqual(IObjectInstance,iid) then
  767. begin
  768. TObject(Obj) := Self;
  769. Result := True;
  770. Exit;
  771. end;
  772. Instance := self;
  773. repeat
  774. IEntry := Instance.GetInterfaceEntry(iid);
  775. result := GetInterfaceByEntry(Instance, IEntry, obj);
  776. if (not result) or
  777. (IEntry^.IType in [etStandard, etFieldValue,
  778. etStaticMethodResult, etVirtualMethodResult]) then
  779. Break;
  780. { if interface is implemented by a class-type property or field,
  781. continue search }
  782. Instance := TObject(obj);
  783. until False;
  784. { Getter function will normally AddRef, so adding another reference here
  785. will cause memleak. }
  786. if result and (IEntry^.IType in [etStandard, etFieldValue]) then
  787. IInterface(obj)._AddRef;
  788. end;
  789. function TObject.GetInterfaceWeak(const iid : tguid; out obj) : boolean;
  790. var
  791. IEntry: PInterfaceEntry;
  792. Instance: TObject;
  793. begin
  794. if IsGUIDEqual(IObjectInstance,iid) then
  795. begin
  796. TObject(Obj) := Self;
  797. Result := True;
  798. Exit;
  799. end;
  800. Instance := self;
  801. repeat
  802. IEntry := Instance.GetInterfaceEntry(iid);
  803. result := GetInterfaceByEntry(Instance, IEntry, obj);
  804. if (not result) or
  805. (IEntry^.IType in [etStandard, etFieldValue,
  806. etStaticMethodResult, etVirtualMethodResult]) then
  807. Break;
  808. { if interface is implemented by a class-type property or field,
  809. continue search }
  810. Instance := TObject(obj);
  811. until False;
  812. { Getter function will normally AddRef, so we have to release it,
  813. else the ref is not weak. }
  814. if result and not (IEntry^.IType in [etStandard, etFieldValue]) then
  815. IInterface(obj)._Release;
  816. end;
  817. function TObject.GetInterfaceByStr(const iidstr : shortstring;out obj) : boolean;
  818. var
  819. IEntry: PInterfaceEntry;
  820. Instance: TObject;
  821. begin
  822. Instance := self;
  823. repeat
  824. IEntry := Instance.GetInterfaceEntryByStr(iidstr);
  825. result := GetInterfaceByEntry(Instance, IEntry, obj);
  826. if (not result) or
  827. (IEntry^.IType in [etStandard, etFieldValue,
  828. etStaticMethodResult, etVirtualMethodResult]) then
  829. Break;
  830. { if interface is implemented by a class-type property or field,
  831. continue search }
  832. Instance := TObject(obj);
  833. until False;
  834. { Getter function will normally AddRef, so adding another reference here
  835. will cause memleak. (com interfaces only!) }
  836. if result and Assigned(IEntry^.IID) and (IEntry^.IType in [etStandard, etFieldValue]) then
  837. IInterface(obj)._AddRef;
  838. end;
  839. function TObject.GetInterface(const iidstr : shortstring;out obj) : boolean;
  840. begin
  841. Result := GetInterfaceByStr(iidstr,obj);
  842. end;
  843. class function TObject.GetInterfaceEntry(const iid : tguid) : pinterfaceentry;
  844. var
  845. i: longint;
  846. intftable: pinterfacetable;
  847. ovmt: PVmt;
  848. begin
  849. ovmt := PVmt(Self);
  850. while Assigned(ovmt) and Assigned(ovmt^.vIntftable) do
  851. begin
  852. intftable:=ovmt^.vIntfTable;
  853. for i:=0 to intftable^.EntryCount-1 do
  854. begin
  855. result:=@intftable^.Entries[i];
  856. if assigned(Result^.iid) and IsGUIDEqual(Result^.iid^,iid) then
  857. Exit;
  858. end;
  859. ovmt := ovmt^.vParent;
  860. end;
  861. result := nil;
  862. end;
  863. class function TObject.GetInterfaceEntryByStr(const iidstr : shortstring) : pinterfaceentry;
  864. var
  865. i: longint;
  866. intftable: pinterfacetable;
  867. ovmt: PVmt;
  868. begin
  869. ovmt := PVmt(Self);
  870. while Assigned(ovmt) and Assigned(ovmt^.vIntfTable) do
  871. begin
  872. intftable:=ovmt^.vIntfTable;
  873. for i:=0 to intftable^.EntryCount-1 do
  874. begin
  875. result:=@intftable^.Entries[i];
  876. if assigned(result^.iidstr) and (result^.iidstr^ = iidstr) then
  877. Exit;
  878. end;
  879. ovmt := ovmt^.vParent;
  880. end;
  881. result:=nil;
  882. end;
  883. class function TObject.GetInterfaceTable : pinterfacetable;
  884. begin
  885. getinterfacetable:=PVmt(Self)^.vIntfTable;
  886. end;
  887. class function TObject.UnitName : RTLString;
  888. {$ifdef FPC_HAS_FEATURE_RTTI}
  889. type
  890. TClassTypeInfo = {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
  891. Attributes: Pointer;
  892. case TTypeKind of
  893. tkClass: (
  894. ClassType: TClass;
  895. ParentInfo: Pointer;
  896. PropCount: SmallInt;
  897. UnitName: ShortString;
  898. );
  899. { include for proper alignment }
  900. tkInt64: (
  901. Dummy: Int64;
  902. );
  903. end;
  904. PClassTypeInfo = ^TClassTypeInfo;
  905. var
  906. classtypeinfo: PClassTypeInfo;
  907. begin
  908. classtypeinfo:=ClassInfo;
  909. if Assigned(classtypeinfo) then
  910. begin
  911. // offset PTypeInfo by Length(Name) + 2 (ShortString length byte + SizeOf(Kind))
  912. inc(Pointer(classtypeinfo), PByte(Pointer(classtypeinfo)+1)^ + 2);
  913. {$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
  914. classtypeinfo:=aligntoqword(classtypeinfo);
  915. {$endif}
  916. result:=classtypeinfo^.UnitName;
  917. end
  918. else
  919. result:='';
  920. end;
  921. {$else not FPC_HAS_FEATURE_RTTI}
  922. begin
  923. result:='';
  924. end;
  925. {$endif ndef FPC_HAS_FEATURE_RTTI}
  926. class function TObject.QualifiedClassName: RTLString;
  927. var
  928. uname: RTLString;
  929. begin
  930. uname := UnitName; //TODO: change 'UnitName' to 'UnitScope' as soon as RTL implement it
  931. if uname='' then
  932. result:=ClassName
  933. else
  934. result:=Concat(uname, '.', ClassName);
  935. end;
  936. function TObject.Equals(Obj: TObject) : boolean;
  937. begin
  938. result:=Obj=Self;
  939. end;
  940. function TObject.GetHashCode: PtrInt;
  941. begin
  942. result:=PtrInt(Self);
  943. end;
  944. function TObject.ToString: RTLString;
  945. begin
  946. result:=ClassName;
  947. end;
  948. procedure TObject.DisposeOf;
  949. begin
  950. Free;
  951. end;
  952. function TObject.GetDisposed : Boolean;
  953. begin
  954. Result:=False;
  955. end;
  956. procedure TObject.CheckDisposed;
  957. begin
  958. // Do nothing since we have no reference count.
  959. end;
  960. {$IFDEF SYSTEM_HAS_FEATURE_MONITOR}
  961. function TObject.SetMonitorData(aData,aCheckOld : Pointer) : Pointer;
  962. begin
  963. Result:=InterlockedCompareExchange(_MonitorData,aData,aCheckOld);
  964. end;
  965. function TObject.GetMonitorData: Pointer;
  966. begin
  967. Result:=_MonitorData;
  968. end;
  969. {$ENDIF}
  970. {****************************************************************************
  971. TINTERFACEDOBJECT
  972. ****************************************************************************}
  973. function TInterfacedObject.QueryInterface(
  974. {$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} iid : tguid;out obj) : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  975. begin
  976. if getinterface(iid,obj) then
  977. result:=S_OK
  978. else
  979. result:=longint(E_NOINTERFACE);
  980. end;
  981. function TInterfacedObject._AddRef : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  982. begin
  983. _addref:=frefcount;
  984. if _addref<>0 then
  985. begin
  986. if _addref>0 then
  987. _addref:=interlockedincrement(frefcount);
  988. exit;
  989. end;
  990. frefcount:=1; { Work non-atomically in the common case of refcount = 0 (typical state after the complete object construction). }
  991. _addref:=1;
  992. end;
  993. function TInterfacedObject._Release : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  994. begin
  995. _Release:=frefcount;
  996. if _Release<>1 then
  997. begin
  998. if _Release<=0 then { -1 means recursive call from destructor... 0 is impossible. }
  999. exit;
  1000. _Release:=interlockeddecrement(frefcount);
  1001. if _Release>0 then
  1002. exit;
  1003. end
  1004. else
  1005. _Release:=0; { Work non-atomically in the common case of refcount = 1 (typical state for the last owner... which is often the only owner). }
  1006. frefcount:=-1; { Prevent recursive _Release from destroying twice (bug 32168). }
  1007. self.destroy;
  1008. end;
  1009. destructor TInterfacedObject.Destroy;
  1010. begin
  1011. // We must explicitly reset. Bug ID 32353
  1012. FRefCount:=0;
  1013. inherited Destroy;
  1014. end;
  1015. procedure TInterfacedObject.AfterConstruction;
  1016. begin
  1017. { we need to fix the refcount we forced in newinstance }
  1018. { further, it must be done in a thread safe way }
  1019. if frefcount=1 then
  1020. frefcount:=0 { Work non-atomically in the common case of refcount = 1 (usual state before AfterConstruction). }
  1021. else
  1022. declocked(frefcount);
  1023. end;
  1024. procedure TInterfacedObject.BeforeDestruction;
  1025. begin
  1026. if frefcount>0 then { Legitimate values: -1 if destroying by _Release, 0 if destroying manually. }
  1027. HandleError(204);
  1028. end;
  1029. class function TInterfacedObject.NewInstance : TObject;
  1030. begin
  1031. NewInstance:=inherited NewInstance;
  1032. if NewInstance<>nil then
  1033. TInterfacedObject(NewInstance).frefcount:=1;
  1034. end;
  1035. {****************************************************************************
  1036. TAGGREGATEDOBJECT
  1037. ****************************************************************************}
  1038. constructor TAggregatedObject.Create(const aController: IUnknown);
  1039. begin
  1040. inherited Create;
  1041. { do not keep a counted reference to the controller! }
  1042. fcontroller := Pointer(aController);
  1043. end;
  1044. function TAggregatedObject.QueryInterface(
  1045. {$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} iid : tguid;out obj) : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1046. begin
  1047. Result := IUnknown(fcontroller).QueryInterface(iid, obj);
  1048. end;
  1049. function TAggregatedObject._AddRef : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1050. begin
  1051. Result := IUnknown(fcontroller)._AddRef;
  1052. end;
  1053. function TAggregatedObject._Release : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1054. begin
  1055. Result := IUnknown(fcontroller)._Release;
  1056. end;
  1057. function TAggregatedObject.GetController : IUnknown;
  1058. begin
  1059. Result := IUnknown(fcontroller);
  1060. end;
  1061. {****************************************************************************
  1062. TContainedOBJECT
  1063. ****************************************************************************}
  1064. function TContainedObject.QueryInterface(
  1065. {$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} iid : tguid;out obj) : longint; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1066. begin
  1067. if getinterface(iid,obj) then
  1068. result:=S_OK
  1069. else
  1070. result:=longint(E_NOINTERFACE);
  1071. end;
  1072. {****************************************************************************
  1073. TNoRefCountObject
  1074. ****************************************************************************}
  1075. function TNoRefCountObject.QueryInterface( {$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} iid : tguid;out obj) : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1076. begin
  1077. if getinterface(iid,obj) then
  1078. result:=S_OK
  1079. else
  1080. result:=longint(E_NOINTERFACE);
  1081. end;
  1082. function TNoRefCountObject._AddRef : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1083. begin
  1084. Result:=-1;
  1085. end;
  1086. function TNoRefCountObject._Release : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1087. begin
  1088. Result:=-1;
  1089. end;
  1090. {****************************************************************************
  1091. TCustomAttribute
  1092. ****************************************************************************}
  1093. constructor TCustomAttribute.Create;
  1094. begin
  1095. inherited;
  1096. end;
  1097. {****************************************************************************
  1098. TUnimplementedAttribute
  1099. ****************************************************************************}
  1100. constructor TUnimplementedAttribute.Create;
  1101. begin
  1102. inherited;
  1103. end;
  1104. {****************************************************************************
  1105. TCustomStoredAttribute
  1106. ****************************************************************************}
  1107. constructor StoredAttribute.Create;
  1108. begin
  1109. end;
  1110. constructor StoredAttribute.Create(Const aFlag : Boolean);
  1111. begin
  1112. FFlag:=aFlag;
  1113. end;
  1114. constructor StoredAttribute.Create(Const aName : shortstring);
  1115. begin
  1116. FName:=aName;
  1117. end;
  1118. {****************************************************************************
  1119. TInterfaceThunk
  1120. ****************************************************************************}
  1121. Constructor TInterfaceThunk.Create(aCallback : TThunkCallback);
  1122. begin
  1123. FCallBack:=aCallBack;
  1124. end;
  1125. Procedure TInterfaceThunk.Thunk(aMethod: Longint; aCount : Longint; aData : PArgData);
  1126. begin
  1127. if Assigned(FCallBack) then
  1128. FCallBack(Self,aMethod,aCount,aData);
  1129. end;
  1130. function TInterfaceThunk.QueryInterface({$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} iid : tguid;out obj) : longint;{$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
  1131. begin
  1132. result:=longint(E_NOINTERFACE);
  1133. if (TMethod(FCallBack).Data<>Nil) then
  1134. // Query the object that created us, this is normally TVirtualInterface
  1135. // Take care: do not call QueryInterface, that would create a never-ending loop !!
  1136. if TObject(TMethod(FCallBack).Data).GetInterface(iid,obj) then
  1137. result:=S_OK;
  1138. if (Result<>S_OK) then
  1139. Result:=Inherited QueryInterface(iid,obj);
  1140. if (Result<>S_OK) and Assigned(OnQueryInterface) then
  1141. OnQueryInterface(iid,Result,obj);
  1142. end;
  1143. function TInterfaceThunk.InterfaceVMTOffset : word;
  1144. begin
  1145. Result:=0;
  1146. end;
  1147. {****************************************************************************
  1148. Exception Support
  1149. ****************************************************************************}
  1150. {$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
  1151. {$if defined(FPC_WASM_NATIVE_EXCEPTIONS)}
  1152. {$I except_native.inc}
  1153. {$elseif defined(FPC_WASM_BRANCHFUL_EXCEPTIONS)}
  1154. {$I except_branchful.inc}
  1155. {$else}
  1156. {$i except.inc}
  1157. {$endif}
  1158. {$endif FPC_HAS_FEATURE_EXCEPTIONS}
  1159. {****************************************************************************
  1160. Various Delphi constructs
  1161. ****************************************************************************}
  1162. class operator TMethod.=(const aLeft, aRight: TMethod): Boolean; inline;
  1163. begin
  1164. Result:=(aLeft.Data=aRight.Data) and (aLeft.Code=aRight.Code);
  1165. end;
  1166. class operator TMethod.<>(const aLeft, aRight: TMethod): Boolean; inline;
  1167. begin
  1168. Result:=(aLeft.Data<>aRight.Data) or (aLeft.Code<>aRight.Code);
  1169. end;
  1170. class operator TMethod.>(const aLeft, aRight: TMethod): Boolean; inline;
  1171. begin
  1172. Result:=(PtrUInt(aLeft.Data)>PtrUInt(aRight.Data))
  1173. or
  1174. ((aLeft.Data=aRight.Data) and (PtrUInt(aLeft.Code)>PtrUint(aRight.Code)));
  1175. end;
  1176. class operator TMethod.>=(const aLeft, aRight: TMethod): Boolean; inline;
  1177. begin
  1178. Result:=(aLeft>aRight) or (aLeft=aRight);
  1179. end;
  1180. class operator TMethod.<(const aLeft, aRight: TMethod): Boolean; inline;
  1181. begin
  1182. Result:=(PtrUInt(aLeft.Data)<PtrUInt(aRight.Data))
  1183. or
  1184. ((aLeft.Data=aRight.Data) and (PtrUInt(aLeft.Code)<PtrUint(aRight.Code)));
  1185. end;
  1186. class operator TMethod.<=(const aLeft, aRight: TMethod): Boolean; inline;
  1187. begin
  1188. Result:=(aLeft<aRight) or (aLeft=aRight);
  1189. end;
  1190. function TPtrWrapper.ToPointer: Pointer;
  1191. begin
  1192. Result:=FValue;
  1193. end;
  1194. class function TPtrWrapper.GetNilValue: TPtrWrapper;
  1195. begin
  1196. Result.FValue:=Nil;
  1197. end;
  1198. constructor TPtrWrapper.Create(AValue: PtrInt);
  1199. begin
  1200. FValue:=Pointer(aValue);
  1201. end;
  1202. constructor TPtrWrapper.Create(AValue: Pointer);
  1203. begin
  1204. FValue:=aValue;
  1205. end;
  1206. function TPtrWrapper.ToInteger: PtrInt;
  1207. begin
  1208. Result:=PtrInt(FValue);
  1209. end;
  1210. class operator TPtrWrapper.=(Left, Right: TPtrWrapper): Boolean;
  1211. begin
  1212. Result:=Left.FValue=Right.FValue;
  1213. end;
  1214. constructor TMarshal.Create;
  1215. begin
  1216. System.Error(reInvalidPtr);
  1217. end;
  1218. class function TMarshal.AllocMem(Size: SizeInt): TPtrWrapper;
  1219. begin
  1220. Result.Value := System.AllocMem(Size);
  1221. end;
  1222. class function TMarshal.ReallocMem(OldPtr: TPtrWrapper; NewSize: SizeInt): TPtrWrapper;
  1223. var
  1224. P: Pointer;
  1225. begin
  1226. P := OldPtr.Value;
  1227. Result.Value := System.ReallocMem(P, NewSize);
  1228. end;
  1229. class procedure TMarshal.FreeMem(Ptr: TPtrWrapper);
  1230. begin
  1231. System.FreeMem(Ptr.Value);
  1232. end;
  1233. class procedure TMarshal.Move(Src, Dest: TPtrWrapper; Count: SizeInt); static;
  1234. begin
  1235. System.Move(Src.Value^, Dest.Value^, Count);
  1236. end;
  1237. class function TMarshal.UnsafeAddrOf(var Value): TPtrWrapper;
  1238. begin
  1239. Result.Value := @Value;
  1240. end;
  1241. class procedure TMarshal.Copy(const Src: TUint8Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1242. begin
  1243. System.Move(PUInt8(Src)[StartIndex], Dest.Value^, Count * SizeOf(UInt8));
  1244. end;
  1245. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TUint8Array; StartIndex: SizeInt; Count: SizeInt);
  1246. begin
  1247. System.Move(Src.Value^, PUInt8(Dest)[StartIndex], Count * SizeOf(UInt8));
  1248. end;
  1249. class procedure TMarshal.Copy(const Src: TInt8Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1250. begin
  1251. System.Move(PInt8(Src)[StartIndex], Dest.Value^, Count * SizeOf(Int8));
  1252. end;
  1253. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TInt8Array ; StartIndex: SizeInt; Count: SizeInt);
  1254. begin
  1255. System.Move(Src.Value^, PInt8(Dest)[StartIndex], Count * SizeOf(Int8));
  1256. end;
  1257. class procedure TMarshal.Copy(const Src: TUInt16Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1258. begin
  1259. System.Move(PUInt16(Src)[StartIndex], Dest.Value^, Count * SizeOf(UInt16));
  1260. end;
  1261. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TUint16Array; StartIndex: SizeInt; Count: SizeInt);
  1262. begin
  1263. System.Move(Src.Value^, PUInt16(Dest)[StartIndex], Count * SizeOf(UInt16));
  1264. end;
  1265. class procedure TMarshal.Copy(const Src: TInt16Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1266. begin
  1267. System.Move(PInt16(Src)[StartIndex], Dest.Value^, Count * SizeOf(Int16));
  1268. end;
  1269. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TInt16Array; StartIndex: SizeInt; Count: SizeInt);
  1270. begin
  1271. System.Move(Src.Value^, PInt16(Dest)[StartIndex], Count * SizeOf(Int16));
  1272. end;
  1273. class procedure TMarshal.Copy(const Src: TInt32Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1274. begin
  1275. System.Move(PInt32(Src)[StartIndex], Dest.Value^, Count * SizeOf(Int32));
  1276. end;
  1277. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TInt32Array; StartIndex: SizeInt; Count: SizeInt);
  1278. begin
  1279. System.Move(Src.Value^, PInt32(Dest)[StartIndex], Count * SizeOf(Int32));
  1280. end;
  1281. class procedure TMarshal.Copy(const Src: TInt64Array; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1282. begin
  1283. System.Move(PInt64(Src)[StartIndex], Dest.Value^, Count * SizeOf(Int64));
  1284. end;
  1285. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TInt64Array; StartIndex: SizeInt; Count: SizeInt);
  1286. begin
  1287. System.Move(Src.Value^, PInt64(Dest)[StartIndex], Count * SizeOf(Int64));
  1288. end;
  1289. class procedure TMarshal.Copy(const Src: TPtrWrapperArray; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1290. begin
  1291. System.Move(PPointer(Src)[StartIndex], Dest.Value^, Count * SizeOf(TPtrWrapper));
  1292. end;
  1293. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TPtrWrapperArray; StartIndex: SizeInt; Count: SizeInt);
  1294. begin
  1295. System.Move(Src.Value^, PPointer(Dest)[StartIndex], Count * SizeOf(TPtrWrapper));
  1296. end;
  1297. generic class function TMarshal.FixArray<T>(const Arr: specialize TArray<T>): TPtrWrapper;
  1298. begin
  1299. Result.Value := nil;
  1300. specialize TArray<T>(Result) := Arr;
  1301. end;
  1302. generic class procedure TMarshal.UnfixArray<T>(ArrPtr: TPtrWrapper);
  1303. begin
  1304. Finalize(specialize TArray<T>(ArrPtr));
  1305. end;
  1306. class function TMarshal.ReadByte(Ptr: TPtrWrapper; Ofs: SizeInt = 0): Byte;
  1307. begin
  1308. Result := PByte(Ptr.Value + Ofs)^;
  1309. end;
  1310. class procedure TMarshal.WriteByte(Ptr: TPtrWrapper; Ofs: SizeInt; Value: Byte);
  1311. begin
  1312. PByte(Ptr.Value + Ofs)^ := Value;
  1313. end;
  1314. class procedure TMarshal.WriteByte(Ptr: TPtrWrapper; Value: Byte);
  1315. begin
  1316. PByte(Ptr.Value)^ := Value;
  1317. end;
  1318. class function TMarshal.ReadInt16(Ptr: TPtrWrapper; Ofs: SizeInt = 0): Int16;
  1319. begin
  1320. Result := PInt16(Ptr.Value + Ofs)^;
  1321. end;
  1322. class procedure TMarshal.WriteInt16(Ptr: TPtrWrapper; Ofs: SizeInt; Value: Int16);
  1323. begin
  1324. PInt16(Ptr.Value + Ofs)^ := Value;
  1325. end;
  1326. class procedure TMarshal.WriteInt16(Ptr: TPtrWrapper; Value: Int16);
  1327. begin
  1328. PInt16(Ptr.Value)^ := Value;
  1329. end;
  1330. class function TMarshal.ReadInt32(Ptr: TPtrWrapper; Ofs: SizeInt = 0): Int32;
  1331. begin
  1332. Result := PInt32(Ptr.Value + Ofs)^;
  1333. end;
  1334. class procedure TMarshal.WriteInt32(Ptr: TPtrWrapper; Ofs: SizeInt; Value: Int32);
  1335. begin
  1336. PInt32(Ptr.Value + Ofs)^ := Value;
  1337. end;
  1338. class procedure TMarshal.WriteInt32(Ptr: TPtrWrapper; Value: Int32);
  1339. begin
  1340. PInt32(Ptr.Value)^ := Value;
  1341. end;
  1342. class function TMarshal.ReadInt64(Ptr: TPtrWrapper; Ofs: SizeInt = 0): Int64;
  1343. begin
  1344. Result := PInt64(Ptr.Value + Ofs)^;
  1345. end;
  1346. class procedure TMarshal.WriteInt64(Ptr: TPtrWrapper; Ofs: SizeInt; Value: Int64);
  1347. begin
  1348. PInt64(Ptr.Value + Ofs)^ := Value;
  1349. end;
  1350. class procedure TMarshal.WriteInt64(Ptr: TPtrWrapper; Value: Int64);
  1351. begin
  1352. PInt64(Ptr.Value)^ := Value;
  1353. end;
  1354. class function TMarshal.ReadPtr(Ptr: TPtrWrapper; Ofs: SizeInt = 0): TPtrWrapper;
  1355. begin
  1356. Result.Value := PPointer(Ptr.Value + Ofs)^;
  1357. end;
  1358. class procedure TMarshal.WritePtr(Ptr: TPtrWrapper; Ofs: SizeInt; Value: TPtrWrapper);
  1359. begin
  1360. PPointer(Ptr.Value + Ofs)^ := Value.Value;
  1361. end;
  1362. class procedure TMarshal.WritePtr(Ptr, Value: TPtrWrapper);
  1363. begin
  1364. PPointer(Ptr.Value)^ := Value.Value;
  1365. end;
  1366. {$IFDEF FPC_HAS_FEATURE_UNICODESTRINGS}
  1367. class function TMarshal.AsAnsi(const S: UnicodeString): AnsiString;
  1368. begin
  1369. Result := AnsiString(S);
  1370. end;
  1371. class function TMarshal.AsAnsi(S: PUnicodeChar): AnsiString;
  1372. begin
  1373. result := AnsiString(S);
  1374. end;
  1375. class function TMarshal.InOutString(const S: UnicodeString): PUnicodeChar;
  1376. begin
  1377. Result := PUnicodeChar(S);
  1378. end;
  1379. class function TMarshal.InString(const S: UnicodeString): PUnicodeChar;
  1380. begin
  1381. Result := PUnicodeChar(S);
  1382. end;
  1383. class function TMarshal.OutString(const S: UnicodeString): PUnicodeChar;
  1384. begin
  1385. Result := PUnicodeChar(S);
  1386. end;
  1387. class function TMarshal.FixString(var Str: UnicodeString): TPtrWrapper;
  1388. begin
  1389. UniqueString(Str);
  1390. Result := UnsafeFixString(Str);
  1391. end;
  1392. class procedure TMarshal.UnfixString(Ptr: TPtrWrapper);
  1393. begin
  1394. if Ptr.Value <> PUnicodeChar('') then
  1395. Finalize(UnicodeString(Ptr));
  1396. end;
  1397. class function TMarshal.UnsafeFixString(const Str: UnicodeString): TPtrWrapper;
  1398. begin
  1399. if Length(Str) = 0 then
  1400. begin
  1401. Result.Value := PUnicodeChar('');
  1402. Exit;
  1403. end;
  1404. Result.Value := nil;
  1405. UnicodeString(Result) := Str;
  1406. end;
  1407. class function TMarshal.AllocStringAsAnsi(const Str: UnicodeString): TPtrWrapper;
  1408. begin
  1409. Result := AllocStringAsAnsi(PUnicodeChar(Pointer(Str)), Length(Str), DefaultSystemCodePage);
  1410. end;
  1411. class function TMarshal.AllocStringAsAnsi(const Str: UnicodeString; CodePage: Word): TPtrWrapper;
  1412. begin
  1413. Result := AllocStringAsAnsi(PUnicodeChar(Pointer(Str)), Length(Str), CodePage);
  1414. end;
  1415. class function TMarshal.AllocStringAsAnsi(S: PUnicodeChar): TPtrWrapper;
  1416. begin
  1417. Result := AllocStringAsAnsi(S, Length(S), DefaultSystemCodePage);
  1418. end;
  1419. class function TMarshal.AllocStringAsAnsi(S: PUnicodeChar; CodePage: Word): TPtrWrapper;
  1420. begin
  1421. Result := AllocStringAsAnsi(S, Length(S), CodePage);
  1422. end;
  1423. class function TMarshal.AllocStringAsUnicode(const Str: UnicodeString): TPtrWrapper;
  1424. var
  1425. NBytes: SizeUint;
  1426. begin
  1427. NBytes := (Length(Str) + 1) * SizeOf(UnicodeChar);
  1428. Result.Value := System.GetMem(NBytes);
  1429. System.Move(PUnicodeChar(Str)^, Result.Value^, NBytes);
  1430. end;
  1431. class function TMarshal.AllocStringAsUtf8(const Str: UnicodeString): TPtrWrapper;
  1432. begin
  1433. Result := AllocStringAsAnsi(PUnicodeChar(Pointer(Str)), Length(Str), CP_UTF8);
  1434. end;
  1435. class function TMarshal.AllocStringAsUtf8(S: PUnicodeChar): TPtrWrapper;
  1436. begin
  1437. Result := AllocStringAsAnsi(S, Length(S), CP_UTF8);
  1438. end;
  1439. class function TMarshal.AllocStringAsAnsi(S: PUnicodeChar; Len: SizeInt; CodePage: Word): TPtrWrapper;
  1440. var
  1441. U2ARes: AnsiString;
  1442. NBytes: SizeInt;
  1443. begin
  1444. U2ARes := ''; { Suppress warning. }
  1445. WideStringManager.Unicode2AnsiMoveProc(S, U2ARes, CodePage, Len);
  1446. if Length(U2ARes) = 0 then
  1447. begin
  1448. Result.Value := nil;
  1449. Exit;
  1450. end;
  1451. { Could instead avoid the second allocation, assuming U2ARes.RefCount = 1:
  1452. System.Move(Pointer(U2ARes)^, (Pointer(U2ARes) - AnsiStringHeaderSize)^, (Length(U2ARes) + 1) * SizeOf(AnsiChar));
  1453. Result.FValue := Pointer(U2ARes) - AnsiStringHeaderSize;
  1454. Pointer(U2ARes) := nil; }
  1455. NBytes := (Length(U2ARes) + 1) * SizeOf(AnsiChar);
  1456. Result.Value := System.GetMem(NBytes);
  1457. System.Move(PAnsiChar(U2ARes)^, Result.Value^, NBytes);
  1458. end;
  1459. class procedure TMarshal.Copy(const Src: TUnicodeCharArray; StartIndex: SizeInt; Dest: TPtrWrapper; Count: SizeInt);
  1460. begin
  1461. System.Move(PUnicodeChar(Src)[StartIndex], Dest.Value^, Count * SizeOf(UnicodeChar));
  1462. end;
  1463. class procedure TMarshal.Copy(Src: TPtrWrapper; var Dest: TUnicodeCharArray; StartIndex: SizeInt; Count: SizeInt);
  1464. begin
  1465. System.Move(Src.Value^, PUnicodeChar(Dest)[StartIndex], Count * SizeOf(UnicodeChar));
  1466. end;
  1467. class function TMarshal.ReadStringAsAnsi(Ptr: TPtrWrapper; Len: SizeInt = -1): UnicodeString;
  1468. begin
  1469. Result := ReadStringAsAnsi(DefaultSystemCodePage, Ptr, Len);
  1470. end;
  1471. class function TMarshal.ReadStringAsAnsi(CodePage: Word; Ptr: TPtrWrapper; Len: SizeInt = -1): UnicodeString;
  1472. begin
  1473. { Here and below, IndexByte/Word assume that, when Len >= 0, either:
  1474. - Up to Len characters are accessible in Ptr;
  1475. - IndexByte/Word cannot access invalid memory past the searched character
  1476. (e.g. i386.inc and x86_64.inc IndexByte/Word versions are specifically designed not to). }
  1477. if Len < 0 then
  1478. Len := IndexByte(Ptr.Value^, Len, 0);
  1479. Result := ''; { Suppress warning. }
  1480. WideStringManager.Ansi2UnicodeMoveProc(Ptr.Value, CodePage, Result, Len);
  1481. end;
  1482. class function TMarshal.ReadStringAsAnsiUpTo(CodePage: Word; Ptr: TPtrWrapper; MaxLen: SizeInt): UnicodeString;
  1483. var
  1484. Len: SizeInt;
  1485. begin
  1486. Len := IndexByte(Ptr.Value^, MaxLen, 0);
  1487. if Len < 0 then
  1488. Len := MaxLen;
  1489. Result := ReadStringAsAnsi(CodePage, Ptr, Len);
  1490. end;
  1491. class procedure TMarshal.WriteStringAsAnsi(Ptr: TPtrWrapper; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1492. begin
  1493. WriteStringAsAnsi(Ptr, 0, Value, MaxCharsIncNull, DefaultSystemCodePage);
  1494. end;
  1495. class procedure TMarshal.WriteStringAsAnsi(Ptr: TPtrWrapper; const Value: UnicodeString; MaxCharsIncNull: SizeInt; CodePage: Word);
  1496. begin
  1497. WriteStringAsAnsi(Ptr, 0, Value, MaxCharsIncNull, CodePage);
  1498. end;
  1499. class procedure TMarshal.WriteStringAsAnsi(Ptr: TPtrWrapper; Ofs: SizeInt; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1500. begin
  1501. WriteStringAsAnsi(Ptr, Ofs, Value, MaxCharsIncNull, DefaultSystemCodePage);
  1502. end;
  1503. class procedure TMarshal.WriteStringAsAnsi(Ptr: TPtrWrapper; Ofs: SizeInt; const Value: UnicodeString; MaxCharsIncNull: SizeInt; CodePage: Word);
  1504. var
  1505. U2ARes: AnsiString;
  1506. ValueLen, U2AResLen: SizeInt;
  1507. begin
  1508. U2ARes := ''; { Suppress warning. }
  1509. ValueLen := Length(Value);
  1510. { Delphi null-terminates iff MaxCharsIncNull < 0, so MaxCharsIncNull is actually just MaxChars. }
  1511. if (MaxCharsIncNull > 0) and (MaxCharsIncNull < ValueLen) then
  1512. ValueLen := MaxCharsIncNull; { UTF-16 → ANSI should never shrink element count, so limit the number of characters analyzed. }
  1513. WideStringManager.Unicode2AnsiMoveProc(PUnicodeChar(Pointer(Value)), U2ARes, CodePage, ValueLen);
  1514. U2AResLen := Length(U2ARes);
  1515. if (MaxCharsIncNull >= 0) and (MaxCharsIncNull < U2AResLen) then
  1516. U2AResLen := MaxCharsIncNull;
  1517. System.Move(PAnsiChar(Pointer(U2ARes))^, (Ptr.Value + Ofs)^, U2AResLen * SizeOf(AnsiChar));
  1518. if MaxCharsIncNull < 0 then
  1519. PAnsiChar(Ptr.Value + Ofs)[U2AResLen] := #0;
  1520. end;
  1521. class function TMarshal.ReadStringAsUnicode(Ptr: TPtrWrapper; Len: SizeInt = -1): UnicodeString;
  1522. begin
  1523. if Len < 0 then
  1524. Len := Length(PUnicodeChar(Ptr.Value));
  1525. Result := ''; { Suppress warning. }
  1526. SetLength(Result, Len);
  1527. System.Move(Ptr.Value^, Pointer(Result)^, Len * SizeOf(UnicodeChar));
  1528. end;
  1529. class function TMarshal.ReadStringAsUnicodeUpTo(Ptr: TPtrWrapper; MaxLen: SizeInt): UnicodeString;
  1530. var
  1531. Len: SizeInt;
  1532. begin
  1533. Len := IndexWord(Ptr.Value^, MaxLen, 0);
  1534. if Len < 0 then
  1535. Len := MaxLen;
  1536. Result := ReadStringAsUnicode(Ptr, Len);
  1537. end;
  1538. class procedure TMarshal.WriteStringAsUnicode(Ptr: TPtrWrapper; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1539. begin
  1540. WriteStringAsUnicode(Ptr, 0, Value, MaxCharsIncNull);
  1541. end;
  1542. class procedure TMarshal.WriteStringAsUnicode(Ptr: TPtrWrapper; Ofs: SizeInt; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1543. var
  1544. Len: SizeInt;
  1545. begin
  1546. { Again, Delphi null-terminates iff MaxCharsIncNull < 0, so MaxCharsIncNull is actually just MaxChars. }
  1547. Len := Length(Value);
  1548. if (MaxCharsIncNull >= 0) and (MaxCharsIncNull < Len) then
  1549. Len := MaxCharsIncNull;
  1550. System.Move(Pointer(Value)^, (Ptr.Value + Ofs)^, Len * SizeOf(UnicodeChar));
  1551. if MaxCharsIncNull < 0 then
  1552. PUnicodeChar(Ptr.Value + Ofs)[Len] := #0;
  1553. end;
  1554. class function TMarshal.ReadStringAsUtf8(Ptr: TPtrWrapper; Len: SizeInt = -1): UnicodeString;
  1555. begin
  1556. Result := ReadStringAsAnsi(CP_UTF8, Ptr, Len);
  1557. end;
  1558. class function TMarshal.ReadStringAsUtf8UpTo(Ptr: TPtrWrapper; MaxLen: SizeInt): UnicodeString;
  1559. begin
  1560. Result := ReadStringAsAnsiUpTo(CP_UTF8, Ptr, MaxLen);
  1561. end;
  1562. class procedure TMarshal.WriteStringAsUtf8(Ptr: TPtrWrapper; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1563. begin
  1564. WriteStringAsAnsi(Ptr, Value, MaxCharsIncNull, CP_UTF8);
  1565. end;
  1566. class procedure TMarshal.WriteStringAsUtf8(Ptr: TPtrWrapper; Ofs: SizeInt; const Value: UnicodeString; MaxCharsIncNull: SizeInt);
  1567. begin
  1568. WriteStringAsAnsi(Ptr, Ofs, Value, MaxCharsIncNull, CP_UTF8);
  1569. end;
  1570. {$ENDIF}
  1571. {$IFDEF SYSTEM_HAS_FEATURE_MONITOR}
  1572. {$i monitor.inc}
  1573. {$ENDIF}