Delphi -- Compiler helper for initializing/finalizing variable
1 it CompilerhelperForInitializingFinalizingVariable; 2 3 interface 4 5 { Compiler helper for initializing/finalizing variable } 6 7 procedure _Initialize(p : Pointer; typeInfo : Pointer); 8 procedure _InitializeArray(p : Pointer; typeInfo : Pointer; elemCount : NativeUInt); 9 procedure _InitializeRecord(p : Pointer; typeInfo : Pointer); 10 11 {$IF not defined(X86ASMRTL)} 12 // dcc64 generated code expects P to remain in RAX on exit from this function. 13 function _Finalize(P : Pointer; TypeInfo : Pointer): Pointer; 14 function _FinalizeArray(P : Pointer; TypeInfo : Pointer; ElemCount : NativeUInt): Pointer; 15 function _FinalizeRecord(P : Pointer; TypeInfo : Pointer): Pointer; 16 {$ELSE} 17 procedure _Finalize(p : Pointer; typeInfo : Pointer); 18 procedure _FinalizeArray(P : Pointer; TypeInfo : Pointer; ElemCount : NativeUInt); 19 procedure _FinalizeRecord(P : Pointer; TypeInfo : Pointer); 20 {$ENDIF} 21 22 procedure _CopyRecord(Dest, Source, TypeInfo : Pointer); 23 procedure _CopyObject(Dest, Source : Pointer; vmtPtrOffs : NativeInt; TypeInfo : Pointer); 24 procedure _CopyArray(Dest, Source, TypeInfo : Pointer; Count : NativeUInt); 25 26 procedure _AddRef(P : Pointer; TypeInfo : Pointer); 27 procedure _AddRefArray(P : Pointer; TypeInfo : Pointer; ElemCount : NativeUInt); 28 procedure _AddRefRecord(P : Pointer; TypeInfo : Pointer); 29 30 function _New(Size : NativeInt; TypeInfo : Pointer): Pointer; 31 procedure _Dispose(P : Pointer; TypeInfo : Pointer); 32 33 procedure InitializeArray(p : Pointer; typeInfo : Pointer; elemCount : NativeUInt); 34 procedure CopyArray(Dest, Source, TypeInfo : Pointer; Count : NativeInt); 35 procedure FinalizeArray(P : Pointer; TypeInfo : Pointer; Count : NativeUInt); 36 37 38 implementation 39 40 { =========================================================================== 41 InitializeRecord, InitializeArray, and Initialize are PIC safe even though 42 they alter EBX because they only call each other. They never call out to 43 other functions and they don t access global data. 44 45 FinalizeRecord, Finalize, and FinalizeArray are PIC safe because they call 46 Pascal routines which will have EBX fixup prologs. 47 ===========================================================================} 48 procedure _VarClr(var v : TVarData); 49 begin 50 if Assigned(VarClearProc) then 51 VarClearProc(v) 52 else 53 Error(reVarInvalidOp); 54 end; 55 56 procedure _VarCopy(var Dest : TVarData; const Src : TVarData); 57 begin 58 if Assigned(VarCopyProc) then 59 VarCopyProc(Dest, Src) 60 else 61 Error(reVarInvalidOp); 62 end; 63 64 procedure _VarAddRef(var v : TVarData); 65 begin 66 if Assigned(VarAddRefProc) then 67 VarAddRefProc(v) 68 else 69 Error(reVarInvalidOp); 70 end; 71 72 { =========================================================================== 73 InitializeRecord, InitializeArray, and Initialize are PIC safe even though 74 they alter EBX because they only call each other. They never call out to 75 other functions and they don t access global data. 76 77 FinalizeRecord, Finalize, and FinalizeArray are PIC safe because they call 78 Pascal routines which will have EBX fixup prologs. 79 ===========================================================================} 80 81 procedure _InitializeRecord(p : Pointer; typeInfo : Pointer); 82 var 83 FT : PFieldTable; 84 I : Cardinal; 85 begin 86 FT := PFieldTable(PByte(typeInfo) + Byte(PTypeInfo(typeInfo).Name[0])); 87 if FT.Count > 0 then 88 begin 89 for I := FT.Count - 1 downto 0 do 90 {$IFDEF WEAKREF} 91 if FT.Fields[I].TypeInfo <> nil then 92 {$ENDIF} 93 _InitializeArray(Pointer(PByte(P) + UIntPtr(FT.Fields[I].Offset)), 94 FT.Fields[I].TypeInfo^, 1); 95 end; 96 end; 97 98 function _FinalizeRecord(P : Pointer; TypeInfo : Pointer): Pointer; 99 var 100 FT : PFieldTable; 101 I : Cardinal; 102 {$IFDEF WEAKREF} 103 Weak : Boolean; 104 {$ENDIF} 105 begin 106 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 107 if FT.Count > 0 then 108 begin 109 {$IFDEF WEAKREF} 110 Weak := false; 111 {$ENDIF} 112 for I := 0 to FT.Count - 1 do 113 begin 114 {$IFDEF WEAKREF} 115 if FT.Fields[I].TypeInfo = nil then 116 begin 117 Weak := true; 118 Continue; 119 end; 120 if not Weak then 121 begin 122 {$ENDIF} 123 _FinalizeArray(Pointer(PByte(P) + IntPtr(FT.Fields[I].Offset)), 124 FT.Fields[I].TypeInfo^, 1); 125 {$IFDEF WEAKREF} 126 end 127 else 128 begin 129 case FT.Fields[I].TypeInfo^.Kind of 130 {$IFDEF WEAKINTFREF} 131 tkInterface: 132 _IntfWeakClear(IInterface(Pointer(PByte(P) + 133 IntPtr(FT.Fields[I].Offset))^)); 134 {$ENDIF} 135 {$IFDEF WEAKINSTREF} 136 tkClass: 137 _InstWeakClear(TObject(Pointer(PByte(P) + IntPtr(FT.Fields[I].Offset))^)); 138 {$ENDIF} 139 {$IFDEF WEAKREF} 140 tkMethod: 141 _ClosureRemoveWeakRef(TMethod(Pointer(PByte(P) + 142 IntPtr(FT.Fields[I].Offset))^)); 143 {$ENDIF} 144 else 145 Error(reInvalidPtr); 146 end; 147 end; 148 {$ENDIF} 149 end; 150 end; 151 Result := P; 152 end; 153 154 procedure _InitializeArray(p : Pointer; typeInfo : Pointer; elemCount : NativeUInt); 155 var 156 FT : PFieldTable; 157 I : Cardinal; 158 begin 159 if elemCount = 0 then 160 Exit; 161 case PTypeInfo(typeInfo).Kind of 162 {$IFDEF WEAKREF} 163 tkMethod: 164 while elemCount > 0 do 165 begin 166 TMethod(P^).Data := nil; 167 TMethod(P^).Code := nil; 168 Inc(PByte(P), SizeOf(TMethod)); 169 Dec(elemCount); 170 end; 171 {$ENDIF} 172 {$IFDEF AUTOREFCOUNT} 173 tkClass, 174 {$ENDIF} 175 tkLString, tkWString, tkInterface, tkDynArray, tkUString: 176 while elemCount > 0 do 177 begin 178 PPointer(P)^ := nil; 179 Inc(PByte(P), SizeOf(Pointer)); 180 Dec(elemCount); 181 end; 182 tkVariant: 183 while elemCount > 0 do 184 begin 185 with PVarData(P)^ do 186 for I := Low(RawData) to High(RawData) do 187 RawData[I] := 0; 188 Inc(PByte(P), SizeOf(TVarData)); 189 Dec(elemCount); 190 end; 191 tkArray: 192 begin 193 FT := PFieldTable(PByte(typeInfo) + Byte(PTypeInfo(typeInfo).Name[0])); 194 while elemCount > 0 do 195 begin 196 _InitializeArray(P, FT.Fields[0].TypeInfo^, FT.Count); 197 Inc(PByte(P), FT.Size); 198 Dec(elemCount); 199 end; 200 end; 201 tkRecord: 202 begin 203 FT := PFieldTable(PByte(typeInfo) + Byte(PTypeInfo(typeInfo).Name[0])); 204 while elemCount > 0 do 205 begin 206 _InitializeRecord(P, typeInfo); 207 Inc(PByte(P), FT.Size); 208 Dec(elemCount); 209 end; 210 end; 211 else 212 Error(reInvalidPtr); 213 end; 214 end; 215 216 function _FinalizeArray(P : Pointer; TypeInfo : Pointer; ElemCount : NativeUInt): Pointer; 217 var 218 FT : PFieldTable; 219 begin 220 Result := P; 221 if ElemCount = 0 then 222 Exit; 223 case PTypeInfo(TypeInfo).Kind of 224 {$IFDEF WEAKREF} 225 tkMethod: 226 while ElemCount > 0 do 227 begin 228 _ClosureRemoveWeakRef(TMethod(P^)); 229 Inc(PByte(P), SizeOf(TMethod)); 230 Dec(ElemCount); 231 end; 232 {$ENDIF} 233 {$IFDEF AUTOREFCOUNT} 234 tkClass: 235 while ElemCount > 0 do 236 begin 237 _InstClear(TObject(P^)); 238 Inc(PByte(P), SizeOf(Pointer)); 239 Dec(ElemCount); 240 end; 241 {$ENDIF} 242 tkLString: 243 _LStrArrayClr(P^, ElemCount); 244 tkWString: 245 _WStrArrayClr(P^, ElemCount); 246 tkUString: 247 _UStrArrayClr(P^, ElemCount); 248 tkVariant: 249 while ElemCount > 0 do 250 begin 251 _VarClr(PVarData(P)^); 252 Inc(PByte(P), SizeOf(TVarData)); 253 Dec(ElemCount); 254 end; 255 tkArray: 256 begin 257 FT := PFieldTable(PByte(typeInfo) + Byte(PTypeInfo(typeInfo).Name[0])); 258 while ElemCount > 0 do 259 begin 260 _FinalizeArray(P, FT.Fields[0].TypeInfo^, FT.Count); 261 Inc(PByte(P), FT.Size); 262 Dec(ElemCount); 263 end; 264 end; 265 tkRecord: 266 begin 267 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 268 while ElemCount > 0 do 269 begin 270 _FinalizeRecord(P, TypeInfo); 271 Inc(PByte(P), FT.Size); 272 Dec(ElemCount); 273 end; 274 end; 275 tkInterface: 276 while ElemCount > 0 do 277 begin 278 _IntfClear(IInterface(P^)); 279 Inc(PByte(P), SizeOf(Pointer)); 280 Dec(ElemCount); 281 end; 282 tkDynArray: 283 while ElemCount > 0 do 284 begin 285 { The cast and dereference of P here is to fake out the call to 286 _DynArrayClear. That function expects a var parameter. Our 287 declaration says we got a non-var parameter, but because of 288 the data type that got passed to us (tkDynArray), this isn t 289 strictly true. The compiler will have passed us a reference. } 290 _DynArrayClear(PPointer(P)^, typeInfo); 291 Inc(PByte(P), SizeOf(Pointer)); 292 Dec(ElemCount); 293 end; 294 else 295 Error(reInvalidPtr); 296 end; 297 end; 298 299 procedure _AddRefRecord(P : Pointer; TypeInfo : Pointer); 300 var 301 FT : PFieldTable; 302 I : Cardinal; 303 begin 304 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 305 if FT.Count > 0 then 306 begin 307 for I := 0 to FT.Count - 1 do 308 begin 309 {$IFDEF WEAKREF} 310 // Check for the sentinal indicating the following fields are weak references 311 // which don t need to be reference counted 312 if FT.Fields[I].TypeInfo = nil then 313 Break; 314 {$ENDIF} 315 _AddRefArray(Pointer(PByte(P) + UIntPtr(FT.Fields[I].Offset)), 316 FT.Fields[I].TypeInfo^, 1); 317 end; 318 end; 319 end; 320 321 procedure _AddRefArray(P : Pointer; TypeInfo : Pointer; ElemCount : NativeUInt); 322 var 323 FT : PFieldTable; 324 begin 325 if ElemCount = 0 then 326 Exit; 327 case PTypeInfo(TypeInfo).Kind of 328 {$IFDEF WEAKREF} 329 tkMethod: 330 while ElemCount > 0 do 331 begin 332 _ClosureAddWeakRef(TMethod(P^)); 333 Inc(PByte(P), SizeOf(TMethod)); 334 Dec(ElemCount); 335 end; 336 {$ENDIF} 337 {$IFDEF AUTOREFCOUNT} 338 tkClass: 339 while ElemCount > 0 do 340 begin 341 _InstAddRef(TObject(P^)); 342 Inc(PByte(P), SizeOf(Pointer)); 343 Dec(ElemCount); 344 end; 345 {$ENDIF} 346 tkLString: 347 while ElemCount > 0 do 348 begin 349 _LStrAddRef(PPointer(P)^); 350 Inc(PByte(P), SizeOf(Pointer)); 351 Dec(ElemCount); 352 end; 353 tkWString: 354 while ElemCount > 0 do 355 begin 356 {$IFDEF MSWINDOWS} 357 _WStrAddRef(PWideString(P)^); 358 {$ELSE} 359 _WStrAddRef(PPointer(P)^); 360 {$ENDIF} 361 Inc(PByte(P), SizeOf(Pointer)); 362 Dec(ElemCount); 363 end; 364 tkUString: 365 while ElemCount > 0 do 366 begin 367 _UStrAddRef(PPointer(P)^); 368 Inc(PByte(P), SizeOf(Pointer)); 369 Dec(ElemCount); 370 end; 371 tkVariant: 372 while ElemCount > 0 do 373 begin 374 _VarAddRef(PVarData(P)^); 375 Inc(PByte(P), SizeOf(TVarData)); 376 Dec(ElemCount); 377 end; 378 tkArray: 379 begin 380 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 381 while ElemCount > 0 do 382 begin 383 _AddRefArray(P, FT.Fields[0].TypeInfo^, FT.Count); 384 Inc(PByte(P), FT.Size); 385 Dec(ElemCount); 386 end; 387 end; 388 tkRecord: 389 begin 390 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 391 while ElemCount > 0 do 392 begin 393 _AddRefRecord(P, TypeInfo); 394 Inc(PByte(P), FT.Size); 395 Dec(ElemCount); 396 end; 397 end; 398 tkInterface: 399 while ElemCount > 0 do 400 begin 401 _IntfAddRef(IInterface(P^)); 402 Inc(PByte(P), SizeOf(Pointer)); 403 Dec(ElemCount); 404 end; 405 tkDynArray: 406 while ElemCount > 0 do 407 begin 408 _DynArrayAddRef(PPointer(P)^); 409 Inc(PByte(P), SizeOf(Pointer)); 410 Dec(ElemCount); 411 end; 412 else 413 Error(reInvalidPtr); 414 end; 415 end; 416 417 procedure _AddRef(P : Pointer; TypeInfo : Pointer); 418 begin 419 _AddRefArray(P, TypeInfo, 1); 420 end; 421 422 procedure _CopyRecord(Dest, Source, TypeInfo : Pointer); 423 var 424 FT, EFT : PFieldTable; 425 I, Count, L : Cardinal; 426 {$IFDEF WEAKREF} 427 J, K : Cardinal; 428 {$ENDIF} 429 Offset : UIntPtr; 430 FTypeInfo : PTypeInfo; 431 DestOff, SrcOff : Pointer; 432 begin 433 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 434 Offset := 0; 435 if FT.Count > 0 then 436 begin 437 Count := FT.Count; 438 {$IFDEF WEAKREF} 439 J := 0; 440 K := Count; 441 for I := Count - 1 downto 0 do 442 if FT.Fields[I].TypeInfo = nil then 443 begin 444 K := I + 1; // found the weak sentinal 445 Dec(Count); // remove the sentinal from consideration 446 Break; 447 end; 448 {$ENDIF} 449 for L := 0 to Count - 1 do 450 begin 451 {$IFDEF WEAKREF} 452 if (FT.Fields[J].TypeInfo <> nil) and 453 ((K = FT.Count) or (FT.Fields[J].Offset < FT.Fields[K].Offset)) then 454 begin 455 I := J; 456 Inc(J); 457 end 458 else 459 begin 460 I := K; 461 Inc(K); 462 end; 463 {$ELSE} 464 I := L; 465 {$ENDIF} 466 if FT.Fields[I].Offset > Offset then 467 Move(Pointer(PByte(Source) + Offset)^, 468 Pointer(PByte(Dest) + Offset)^, 469 FT.Fields[I].Offset - Offset); 470 Offset := FT.Fields[I].Offset; 471 FTypeInfo := FT.Fields[I].TypeInfo^; 472 DestOff := Pointer(PByte(Dest) + Offset); 473 SrcOff := Pointer(PByte(Source) + Offset); 474 case FTypeInfo.Kind of 475 {$IFDEF WEAKREF} 476 tkMethod: 477 begin 478 _CopyClosure(PMethod(DestOff)^, PMethod(SrcOff)^); 479 Inc(Offset, SizeOf(TMethod)); 480 end; 481 {$ENDIF} 482 {$IFDEF AUTOREFCOUNT} 483 tkClass: 484 begin 485 {$IFDEF WEAKINSTREF} 486 if I > J then 487 _InstWeakCopy(TObject(PPointer(DestOff)^), 488 TObject(PPointer(SrcOff)^)) 489 else 490 {$ENDIF} 491 _InstCopy(TObject(PPointer(DestOff)^), TObject(PPointer(SrcOff)^)); 492 Inc(Offset, SizeOf(Pointer)); 493 end; 494 {$ENDIF} 495 tkLString: 496 begin 497 _LStrAsg(_PAnsiStr(DestOff)^, _PAnsiStr(SrcOff)^); 498 Inc(Offset, SizeOf(Pointer)); 499 end; 500 tkWString: 501 begin 502 _WStrAsg(_PWideStr(DestOff)^, _PWideStr(SrcOff)^); 503 Inc(Offset, SizeOf(Pointer)); 504 end; 505 tkUString: 506 begin 507 _UStrAsg(PUnicodeString(DestOff)^, PUnicodeString(SrcOff)^); 508 Inc(Offset, SizeOf(Pointer)); 509 end; 510 tkVariant: 511 begin 512 _VarCopy(PVarData(DestOff)^, PVarData(SrcOff)^); 513 Inc(Offset, SizeOf(TVarData)); 514 end; 515 tkArray: 516 begin 517 EFT := 518 PFieldTable(PByte(FTypeInfo) + Byte(PTypeInfo(FTypeInfo).Name[0])); 519 _CopyArray(DestOff, SrcOff, EFT.Fields[0].TypeInfo^, EFT.Count); 520 Inc(Offset, EFT.Size); 521 end; 522 tkRecord: 523 begin 524 EFT := 525 PFieldTable(PByte(FTypeInfo) + Byte(PTypeInfo(FTypeInfo).Name[0])); 526 _CopyRecord(DestOff, SrcOff, FTypeInfo); 527 528 Inc(Offset, EFT.Size); 529 end; 530 tkInterface: 531 begin 532 {$IFDEF WEAKINTFREF} 533 if I > J then 534 _IntfWeakCopy(IInterface(PPointer(DestOff)^), 535 IInterface(PPointer(SrcOff)^)) 536 else 537 {$ENDIF} 538 _IntfCopy(IInterface(PPointer(DestOff)^), 539 IInterface(PPointer(SrcOff)^)); 540 Inc(Offset, SizeOf(Pointer)); 541 end; 542 tkDynArray: 543 begin 544 _DynArrayAsg(PPointer(DestOff)^, PPointer(SrcOff)^, FTypeInfo); 545 Inc(Offset, SizeOf(Pointer)); 546 end; 547 else 548 Error(reInvalidPtr); 549 end; 550 end; 551 end; 552 if FT.Size > Offset then 553 Move(Pointer(PByte(Source) + Offset)^, 554 Pointer(PByte(Dest) + Offset)^, 555 FT.Size - Offset); 556 end; 557 558 procedure _CopyObject(Dest, Source : Pointer; vmtPtrOffs : NativeInt; TypeInfo : Pointer); 559 var 560 SavedVmtPtr : Pointer; 561 begin 562 SavedVmtPtr := PPointer(PByte(Dest) + vmtPtrOffs)^; 563 _CopyRecord(Dest, Source, TypeInfo); 564 PPointer(PByte(Dest) + vmtPtrOffs)^ := SavedVmtPtr; 565 end; 566 567 procedure _CopyArray(Dest, Source, TypeInfo : Pointer; Count : NativeUInt); 568 var 569 FT : PFieldTable; 570 begin 571 if Count = 0 then 572 Exit; 573 case PTypeInfo(TypeInfo).Kind of 574 {$IFDEF WEAKREF} 575 tkMethod: 576 while Count > 0 do 577 begin 578 _CopyClosure(PMethod(Dest)^, PMethod(Source)^); 579 Inc(PByte(Dest), SizeOf(TMethod)); 580 Inc(PByte(Source), SizeOf(TMethod)); 581 Dec(Count); 582 end; 583 {$ENDIF} 584 {$IFDEF AUTOREFCOUNT} 585 tkClass: 586 while Count > 0 do 587 begin 588 _InstCopy(TObject(PPointer(Dest)^), TObject(PPointer(Source)^)); 589 Inc(PByte(Dest), SizeOf(Pointer)); 590 Inc(PByte(Source), SizeOf(Pointer)); 591 Dec(Count); 592 end; 593 {$ENDIF} 594 tkLString: 595 while Count > 0 do 596 begin 597 _LStrAsg(_PAnsiStr(Dest)^, _PAnsiStr(Source)^); 598 Inc(PByte(Dest), SizeOf(Pointer)); 599 Inc(PByte(Source), SizeOf(Pointer)); 600 Dec(Count); 601 end; 602 tkWString: 603 while Count > 0 do 604 begin 605 _WStrAsg(_PWideStr(Dest)^, _PWideStr(Source)^); 606 Inc(PByte(Dest), SizeOf(Pointer)); 607 Inc(PByte(Source), SizeOf(Pointer)); 608 Dec(Count); 609 end; 610 tkUString: 611 while Count > 0 do 612 begin 613 _UStrAsg(PUnicodeString(Dest)^, PUnicodeString(Source)^); 614 Inc(PByte(Dest), SizeOf(Pointer)); 615 Inc(PByte(Source), SizeOf(Pointer)); 616 Dec(Count); 617 end; 618 tkVariant: 619 while Count > 0 do 620 begin 621 _VarCopy(PVarData(Dest)^, PVarData(Source)^); 622 Inc(PByte(Dest), SizeOf(TVarData)); 623 Inc(PByte(Source), SizeOf(TVarData)); 624 Dec(Count); 625 end; 626 tkArray: 627 begin 628 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 629 while Count > 0 do 630 begin 631 _CopyArray(Pointer(Dest), Pointer(Source), 632 FT.Fields[0].TypeInfo^, FT.Count); 633 Inc(PByte(Dest), FT.Size); 634 Inc(PByte(Source), FT.Size); 635 Dec(Count); 636 end; 637 end; 638 tkRecord: 639 begin 640 FT := PFieldTable(PByte(TypeInfo) + Byte(PTypeInfo(TypeInfo).Name[0])); 641 while Count > 0 do 642 begin 643 _CopyRecord(Dest, Source, TypeInfo); 644 Inc(PByte(Dest), FT.Size); 645 Inc(PByte(Source), FT.Size); 646 Dec(Count); 647 end; 648 end; 649 tkInterface: 650 while Count > 0 do 651 begin 652 _IntfCopy(IInterface(PPointer(Dest)^), IInterface(PPointer(Source)^)); 653 Inc(PByte(Dest), SizeOf(Pointer)); 654 Inc(PByte(Source), SizeOf(Pointer)); 655 Dec(Count); 656 end; 657 tkDynArray: 658 while Count > 0 do 659 begin 660 _DynArrayAsg(PPointer(Dest)^, PPointer(Source)^, TypeInfo); 661 Inc(PByte(Dest), SizeOf(Pointer)); 662 Inc(PByte(Source), SizeOf(Pointer)); 663 Dec(Count); 664 end; 665 else 666 Error(reInvalidPtr); 667 end; 668 end; 669 670 procedure CopyArray(Dest, Source, TypeInfo : Pointer; Count : NativeInt); 671 begin 672 if Count > 0 then 673 _CopyArray(Dest, Source, TypeInfo, Count); 674 end; 675 676 procedure InitializeArray(p : Pointer; typeInfo : Pointer; elemCount : NativeUInt); 677 begin 678 _InitializeArray(p, typeInfo, elemCount); 679 end; 680 681 procedure FinalizeArray(P, TypeInfo : Pointer; Count : NativeUInt); 682 begin 683 _FinalizeArray(P, TypeInfo, Count); 684 end; 685 686 procedure _Initialize(p : Pointer; typeInfo : Pointer); 687 begin 688 _InitializeArray(p, typeInfo, 1); 689 end; 690 691 function _Finalize(p : Pointer; typeInfo : Pointer): Pointer; 692 begin 693 Result := _FinalizeArray(p, typeInfo, 1); 694 end; 695 696 function _New(Size : NativeInt; TypeInfo : Pointer): Pointer; 697 begin 698 GetMem(Result, Size); 699 if Result <> nil then 700 _Initialize(Result, TypeInfo); 701 end; 702 703 procedure _Dispose(P : Pointer; TypeInfo : Pointer); 704 begin 705 _Finalize(P, TypeInfo); 706 FreeMem(P); 707 end;