| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by the Free Pascal development team    This unit makes Free Pascal as much as possible Delphi compatible    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}    procedure DoDispCallByIDError(res : Pointer; const disp : IDispatch;desc : PDispDesc; params : Pointer);      begin        handleerrorframe(RuntimeErrorExitCodes[reVarDispatch],get_frame);      end;    procedure fpc_dispatch_by_id(Result: Pointer; const Dispatch: pointer;      DispDesc: Pointer; Params: Pointer); compilerproc;      type        TDispProc = procedure(res : Pointer; const disp : IDispatch;desc : PDispDesc; params : Pointer);      begin        TDispProc(DispCallByIDProc)(Result,IDispatch(Dispatch),DispDesc,Params);      end;{****************************************************************************                  Internal Routines called from the Compiler****************************************************************************}    { the reverse order of the parameters make code generation easier }    function fpc_do_is(aclass : tclass;aobject : tobject) : boolean;[public,alias: 'FPC_DO_IS']; compilerproc;      begin         fpc_do_is:=assigned(aobject) and assigned(aclass) and           aobject.inheritsfrom(aclass);      end;    { the reverse order of the parameters make code generation easier }    function fpc_do_as(aclass : tclass;aobject : tobject): tobject;[public,alias: 'FPC_DO_AS']; compilerproc;      begin         if assigned(aobject) and not(aobject.inheritsfrom(aclass)) then           handleerrorframe(219,get_frame);         result := aobject;      end;    { interface helpers }    procedure fpc_intf_decr_ref(var i: pointer);[public,alias: 'FPC_INTF_DECR_REF']; compilerproc;      begin        if assigned(i) then          begin            IUnknown(i)._Release;            i:=nil;          end;      end;    { local declaration for intf_decr_ref for local access }    procedure intf_decr_ref(var i: pointer); [external name 'FPC_INTF_DECR_REF'];    procedure fpc_intf_incr_ref(i: pointer);[public,alias: 'FPC_INTF_INCR_REF']; compilerproc;      begin         if assigned(i) then           IUnknown(i)._AddRef;      end;    { local declaration of intf_incr_ref for local access }    procedure intf_incr_ref(i: pointer); [external name 'FPC_INTF_INCR_REF'];    procedure fpc_intf_assign(var D: pointer; const S: pointer);[public,alias: 'FPC_INTF_ASSIGN']; compilerproc;      begin         if assigned(S) then           IUnknown(S)._AddRef;         if assigned(D) then           IUnknown(D)._Release;         D:=S;      end;    procedure fpc_intf_assign_by_iid(var D: pointer; const S: pointer; const iid: TGUID);[public,alias: 'FPC_INTF_ASSIGN2']; compilerproc;      var        tmp : pointer;      begin         if assigned(S) then           begin             if IUnknown(S).QueryInterface(iid,tmp)<>S_OK then               handleerror(219);             if assigned(D) then               IUnknown(D)._Release;             D:=tmp;           end         else           begin             if assigned(D) then               IUnknown(D)._Release;             D:=nil;           end;      end;    function fpc_intf_as(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_INTF_AS']; compilerproc;      var        tmpi: pointer; // _AddRef before _Release      begin        if assigned(S) then          begin             if IUnknown(S).QueryInterface(iid,tmpi)<>S_OK then               handleerror(219);             pointer(fpc_intf_as):=tmpi;          end        else          fpc_intf_as:=nil;      end;    function fpc_class_as_intf(const S: pointer; const iid: TGUID): IInterface;[public,alias: 'FPC_CLASS_AS_INTF']; compilerproc;      var        tmpi: pointer; // _AddRef before _Release      begin        if assigned(S) then          begin             if not TObject(S).GetInterface(iid,tmpi) then               handleerror(219);             pointer(fpc_class_as_intf):=tmpi;          end        else          fpc_class_as_intf:=nil;      end;{****************************************************************************                               TOBJECT****************************************************************************}      constructor TObject.Create;        begin        end;      destructor TObject.Destroy;        begin        end;      procedure TObject.Free;        begin           // the call via self avoids a warning           if self<>nil then             self.destroy;        end;      class function TObject.InstanceSize : SizeInt;        begin           InstanceSize:=pSizeInt(pointer(self)+vmtInstanceSize)^;        end;      procedure InitInterfacePointers(objclass: tclass;instance : pointer);        var          i: integer;          intftable: pinterfacetable;          Res: pinterfaceentry;        begin          while assigned(objclass) do            begin              intftable:=pinterfacetable((pointer(objclass)+vmtIntfTable)^);              if assigned(intftable) then              begin                i:=intftable^.EntryCount;                Res:=@intftable^.Entries[0];                while i>0 do begin                  if Res^.IType = etStandard then                    ppointer(@(pbyte(instance)[Res^.IOffset]))^:=                      pointer(Res^.VTable);                  inc(Res);                  dec(i);                end;              end;              objclass:=pclass(pointer(objclass)+vmtParent)^;            end;        end;      class function TObject.InitInstance(instance : pointer) : tobject; {$ifdef SYSTEMINLINE} inline; {$ENDIF}        begin           { the size is saved at offset 0 }           fillchar(instance^, InstanceSize, 0);           { insert VMT pointer into the new created memory area }           { (in class methods self contains the VMT!)           }           ppointer(instance)^:=pointer(self);           InitInterfacePointers(self,instance);           InitInstance:=TObject(Instance);        end;      class function TObject.ClassParent : tclass;        begin           { type of self is class of tobject => it points to the vmt }           { the parent vmt is saved at offset vmtParent              }           classparent:=pclass(pointer(self)+vmtParent)^;        end;      class function TObject.NewInstance : tobject;        var           p : pointer;        begin           getmem(p, InstanceSize);           if p <> nil then              InitInstance(p);           NewInstance:=TObject(p);        end;      procedure TObject.FreeInstance;        begin           CleanupInstance;           FreeMem(Pointer(Self));        end;      class function TObject.ClassType : TClass;        begin           ClassType:=TClass(Pointer(Self))        end;      type         tmethodnamerec = packed record            name : pshortstring;            addr : pointer;         end;         tmethodnametable = packed record           count : dword;           entries : packed array[0..0] of tmethodnamerec;         end;         pmethodnametable =  ^tmethodnametable;      class function TObject.MethodAddress(const name : shortstring) : pointer;        var           UName : ShortString;           methodtable : pmethodnametable;           i : dword;           vmt : tclass;        begin           UName := UpCase(name);           vmt:=self;           while assigned(vmt) do             begin                methodtable:=pmethodnametable((Pointer(vmt)+vmtMethodTable)^);                if assigned(methodtable) then                  begin                     for i:=0 to methodtable^.count-1 do                       if UpCase(methodtable^.entries[i].name^)=UName then                         begin                            MethodAddress:=methodtable^.entries[i].addr;                            exit;                         end;                  end;                vmt:=pclass(pointer(vmt)+vmtParent)^;             end;           MethodAddress:=nil;        end;      class function TObject.MethodName(address : pointer) : shortstring;        var           methodtable : pmethodnametable;           i : dword;           vmt : tclass;        begin           vmt:=self;           while assigned(vmt) do             begin                methodtable:=pmethodnametable((Pointer(vmt)+vmtMethodTable)^);                if assigned(methodtable) then                  begin                     for i:=0 to methodtable^.count-1 do                       if methodtable^.entries[i].addr=address then                         begin                            MethodName:=methodtable^.entries[i].name^;                            exit;                         end;                  end;                vmt:=pclass(pointer(vmt)+vmtParent)^;             end;           MethodName:='';        end;      function TObject.FieldAddress(const name : shortstring) : pointer;        type           PFieldInfo = ^TFieldInfo;           TFieldInfo ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}           packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}           record             FieldOffset: PtrUInt;             ClassTypeIndex: Word;             Name: ShortString;           end;           PFieldTable = ^TFieldTable;           TFieldTable ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}           packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}           record             FieldCount: Word;             ClassTable: Pointer;             { should be array[Word] of TFieldInfo;  but               Elements have variant size! force at least proper alignment }             Fields: array[0..0] of TFieldInfo           end;        var           UName: ShortString;           CurClassType: TClass;           FieldTable: PFieldTable;           FieldInfo: PFieldInfo;           i: Integer;        begin           if Length(name) > 0 then           begin             UName := UpCase(name);             CurClassType := ClassType;             while CurClassType <> nil do             begin               FieldTable := PFieldTable((Pointer(CurClassType) + vmtFieldTable)^);               if FieldTable <> nil then               begin                 FieldInfo := @FieldTable^.Fields[0];                 for i := 0 to FieldTable^.FieldCount - 1 do                 begin                   if UpCase(FieldInfo^.Name) = UName then                   begin                     fieldaddress := Pointer(Self) + FieldInfo^.FieldOffset;                     exit;                   end;                   FieldInfo := PFieldInfo(PByte(@FieldInfo^.Name) + 1 + Length(FieldInfo^.Name));{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}                   { align to largest field of TFieldInfo }                   FieldInfo := Align(FieldInfo, SizeOf(PtrUInt));{$endif FPC_REQUIRES_PROPER_ALIGNMENT}                 end;               end;               { Try again with the parent class type }               CurClassType:=pclass(pointer(CurClassType)+vmtParent)^;             end;           end;           fieldaddress:=nil;        end;      function TObject.SafeCallException(exceptobject : tobject;        exceptaddr : pointer) : longint;        begin           safecallexception:=0;        end;      class function TObject.ClassInfo : pointer;        begin           ClassInfo:=ppointer(Pointer(self)+vmtTypeInfo)^;        end;      class function TObject.ClassName : ShortString;        begin           ClassName:=PShortString((Pointer(Self)+vmtClassName)^)^;        end;      class function TObject.ClassNameIs(const name : string) : boolean;        begin           ClassNameIs:=Upcase(ClassName)=Upcase(name);        end;      class function TObject.InheritsFrom(aclass : TClass) : Boolean;        var           vmt : tclass;        begin           vmt:=self;           while assigned(vmt) do             begin                if vmt=aclass then                  begin                     InheritsFrom:=true;                     exit;                  end;                vmt:=pclass(pointer(vmt)+vmtParent)^;             end;           InheritsFrom:=false;        end;      class function TObject.stringmessagetable : pstringmessagetable;        begin           stringmessagetable:=pstringmessagetable((pointer(Self)+vmtMsgStrPtr)^);        end;      type         tmessagehandler = procedure(var msg) of object;         tmessagehandlerrec = packed record            proc : pointer;            obj : pointer;         end;      procedure TObject.Dispatch(var message);        type           tmsgtable = packed record              index : dword;              method : pointer;           end;           pmsgtable = ^tmsgtable;        var           index : dword;           count,i : longint;           msgtable : pmsgtable;           p : pointer;           vmt : tclass;           msghandler : tmessagehandler;        begin           index:=dword(message);           vmt:=ClassType;           while assigned(vmt) do             begin                // See if we have messages at all in this class.                p:=pointer(vmt)+vmtDynamicTable;                If assigned(PPointer(p)^) then                  begin                     msgtable:=pmsgtable(Pointer(p^)+4);                     count:=pdword(p^)^;                  end                else                  Count:=0;                { later, we can implement a binary search here }                for i:=0 to count-1 do                  begin                     if index=msgtable[i].index then                       begin                          p:=msgtable[i].method;                          tmessagehandlerrec(msghandler).proc:=p;                          tmessagehandlerrec(msghandler).obj:=self;                          msghandler(message);                          exit;                       end;                  end;                vmt:=pclass(pointer(vmt)+vmtParent)^;             end;           DefaultHandler(message);        end;      procedure TObject.DispatchStr(var message);        type           PSizeUInt = ^SizeUInt;        var           name : shortstring;           count,i : longint;           msgstrtable : pmsgstrtable;           p : pointer;           vmt : tclass;           msghandler : tmessagehandler;        begin           name:=pshortstring(@message)^;           vmt:=ClassType;           while assigned(vmt) do             begin                p:=(pointer(vmt)+vmtMsgStrPtr);                If (P<>Nil) and (PPtruInt(P)^<>0) then                  begin                  count:=Pptruint(PSizeUInt(p)^)^;                  msgstrtable:=pmsgstrtable(PSizeUInt(P)^+sizeof(ptruint));                  end                else                  Count:=0;                { later, we can implement a binary search here }                for i:=0 to count-1 do                  begin                     if name=msgstrtable[i].name^ then                       begin                          p:=msgstrtable[i].method;                          tmessagehandlerrec(msghandler).proc:=p;                          tmessagehandlerrec(msghandler).obj:=self;                          msghandler(message);                          exit;                       end;                  end;                vmt:=pclass(pointer(vmt)+vmtParent)^;             end;           DefaultHandlerStr(message);        end;      procedure TObject.DefaultHandler(var message);        begin        end;      procedure TObject.DefaultHandlerStr(var message);        begin        end;      procedure TObject.CleanupInstance;        Type          TRecElem = packed Record            Info : Pointer;            Offset : Longint;          end;          TRecElemArray = packed array[1..Maxint] of TRecElem;          PRecRec = ^TRecRec;          TRecRec = record            Size,Count : Longint;            Elements : TRecElemArray;          end;        var           vmt  : tclass;           temp : pbyte;           count,           i    : longint;{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}           recelem  : TRecElem;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}        begin           vmt:=ClassType;           while vmt<>nil do             begin               { This need to be included here, because Finalize()                 has should support for tkClass }               Temp:=Pointer((Pointer(vmt)+vmtInitTable)^);               if Assigned(Temp) then                 begin                   inc(Temp);                   I:=Temp^;                   inc(temp,I+1);                // skip name string;                   temp:=aligntoptr(temp);{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}                   move(PRecRec(Temp)^.Count,Count,sizeof(Count));{$else FPC_REQUIRES_PROPER_ALIGNMENT}                   Count:=PRecRec(Temp)^.Count;  // get element Count{$endif FPC_REQUIRES_PROPER_ALIGNMENT}                   For I:=1 to count do{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}                     begin                       move(PRecRec(Temp)^.elements[I],RecElem,sizeof(RecElem));                       With RecElem do                         int_Finalize (pointer(self)+Offset,Info);                     end;{$else FPC_REQUIRES_PROPER_ALIGNMENT}                     With PRecRec(Temp)^.elements[I] do                       int_Finalize (pointer(self)+Offset,Info);{$endif FPC_REQUIRES_PROPER_ALIGNMENT}                 end;               vmt:=pclass(pointer(vmt)+vmtParent)^;             end;        end;      procedure TObject.AfterConstruction;        begin        end;      procedure TObject.BeforeDestruction;        begin        end;      function IsGUIDEqual(const guid1, guid2: tguid): boolean;        begin          IsGUIDEqual:=            (guid1.D1=guid2.D1) and            (PDWORD(@guid1.D2)^=PDWORD(@guid2.D2)^) and            (PDWORD(@guid1.D4[0])^=PDWORD(@guid2.D4[0])^) and            (PDWORD(@guid1.D4[4])^=PDWORD(@guid2.D4[4])^);        end;      function getinterfacebyentry(Instance: pointer; IEntry: pinterfaceentry; out obj): boolean;        var          Getter: function: IInterface of object;        begin          Pointer(Obj) := nil;          if Assigned(IEntry) and Assigned(Instance) then          begin            case IEntry^.IType of              etStandard:                begin                  //writeln('Doing etStandard cast of ', TObject(Instance).classname(), ' with self = ', ptruint(Instance), ' and offset = ', IEntry^.IOffset);                  Pbyte(Obj):=Pbyte(instance)+IEntry^.IOffset;                end;              etFieldValue:                begin                  //writeln('Doing etFieldValue cast of ', TObject(Instance).classname(), ' with offset = ', IEntry^.IOffset);                  Pointer(obj) := ppointer(Pbyte(Instance)+IEntry^.IOffset)^;                end;              etVirtualMethodResult:                begin                  //writeln('Doing etVirtualMethodResult cast of ', TObject(Instance).classname());                  TMethod(Getter).data := Instance;                  TMethod(Getter).code := ppointer(Pbyte(Instance) + IEntry^.IOffset)^;                  Pointer(obj) := Pointer(Getter());                end;              etStaticMethodResult:                begin                  //writeln('Doing etStaticMethodResult cast of ', TObject(Instance).classname());                  TMethod(Getter).data := Instance;                  TMethod(Getter).code := pointer(IEntry^.IOffset);                  Pointer(obj) := Pointer(Getter());                end;            end;          end;          result := assigned(pointer(obj));          if result then            IInterface(obj)._AddRef;        end;      function TObject.getinterface(const iid : tguid;out obj) : boolean;        begin          Result := getinterfacebyentry(self, getinterfaceentry(iid), obj);        end;      function TObject.getinterfacebystr(const iidstr : string;out obj) : boolean;        begin          Result := getinterfacebyentry(self, getinterfaceentrybystr(iidstr), obj);        end;      class function TObject.getinterfaceentry(const iid : tguid) : pinterfaceentry;        var          i: integer;          intftable: pinterfacetable;          Res: pinterfaceentry;        begin          getinterfaceentry:=nil;          intftable:=pinterfacetable((pointer(Self)+vmtIntfTable)^);          if assigned(intftable) then begin            i:=intftable^.EntryCount;            Res:=@intftable^.Entries[0];            while (i>0) and               not (assigned(Res^.iid) and IsGUIDEqual(Res^.iid^,iid)) do begin              inc(Res);              dec(i);            end;            if (i>0) then              getinterfaceentry:=Res;          end;          if (getinterfaceentry=nil)and not(classparent=nil) then            getinterfaceentry:=classparent.getinterfaceentry(iid)        end;      class function TObject.getinterfaceentrybystr(const iidstr : string) : pinterfaceentry;        var          i: integer;          intftable: pinterfacetable;          Res: pinterfaceentry;        begin          getinterfaceentrybystr:=nil;          intftable:=pinterfacetable((pointer(Self)+vmtIntfTable)^);          if assigned(intftable) then begin            i:=intftable^.EntryCount;            Res:=@intftable^.Entries[0];            while (i>0) and (Res^.iidstr^<>iidstr) do begin              inc(Res);              dec(i);            end;            if (i>0) then              getinterfaceentrybystr:=Res;          end;          if (getinterfaceentrybystr=nil) and not(classparent=nil) then            getinterfaceentrybystr:=classparent.getinterfaceentrybystr(iidstr)        end;      class function TObject.getinterfacetable : pinterfacetable;        begin          getinterfacetable:=pinterfacetable((pointer(Self)+vmtIntfTable)^);        end;{****************************************************************************                               TINTERFACEDOBJECT****************************************************************************}    function TInterfacedObject.QueryInterface(      const iid : tguid;out obj) : longint;stdcall;      begin         if getinterface(iid,obj) then           result:=0         else           result:=longint(E_NOINTERFACE);      end;    function TInterfacedObject._AddRef : longint;stdcall;      begin         _addref:=interlockedincrement(frefcount);      end;    function TInterfacedObject._Release : longint;stdcall;      begin         _Release:=interlockeddecrement(frefcount);         if _Release=0 then           self.destroy;      end;    procedure TInterfacedObject.AfterConstruction;      begin         { we need to fix the refcount we forced in newinstance }         { further, it must be done in a thread safe way        }         declocked(frefcount);      end;    procedure TInterfacedObject.BeforeDestruction;      begin         if frefcount<>0 then           HandleError(204);      end;    class function TInterfacedObject.NewInstance : TObject;      begin         NewInstance:=inherited NewInstance;         if NewInstance<>nil then           TInterfacedObject(NewInstance).frefcount:=1;      end;{****************************************************************************                               TAGGREGATEDOBJECT****************************************************************************}    constructor TAggregatedObject.Create(const aController: IUnknown);      begin        inherited Create;        { do not keep a counted reference to the controller! }        fcontroller := Pointer(aController);      end;    function TAggregatedObject.QueryInterface(      const iid : tguid;out obj) : longint;stdcall;      begin         Result := IUnknown(fcontroller).QueryInterface(iid, obj);      end;    function TAggregatedObject._AddRef : longint;stdcall;      begin         Result := IUnknown(fcontroller)._AddRef;      end;    function TAggregatedObject._Release : longint;stdcall;      begin         Result := IUnknown(fcontroller)._Release;      end;    function TAggregatedObject.GetController : IUnknown;      begin         Result := IUnknown(fcontroller);      end;{****************************************************************************                             Exception Support****************************************************************************}{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}{$i except.inc}{$endif FPC_HAS_FEATURE_EXCEPTIONS}
 |