| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870 | {    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. **********************************************************************}{$ifdef FPC_HAS_FEATURE_VARIANTS}    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;{$endif FPC_HAS_FEATURE_VARIANTS}{****************************************************************************                  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;    function fpc_class_as_corbaintf(const S: pointer; const iid: Shortstring): Pointer;[public,alias: 'FPC_CLASS_AS_CORBAINTF']; compilerproc;      var        tmpi: pointer; // _AddRef before _Release      begin        if assigned(S) then          begin             if not TObject(S).GetInterface(iid,tmpi) then               handleerror(219);             fpc_class_as_corbaintf:=tmpi;          end        else          fpc_class_as_corbaintf:=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 := PVmt(Self)^.vInstanceSize;        end;      var        emptyintf: ptruint; public name 'FPC_EMPTYINTF';      procedure InitInterfacePointers(objclass: tclass;instance : pointer);        var          ovmt: PVmt;          i: longint;          intftable: pinterfacetable;          Res: pinterfaceentry;        begin          ovmt := PVmt(objclass);          while assigned(ovmt) and (ovmt^.vIntfTable <> @emptyintf) do            begin              intftable:=ovmt^.vIntfTable;              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;              ovmt:=ovmt^.vParent;            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);           if PVmt(self)^.vIntfTable <> @emptyintf then             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:=tclass(PVmt(Self)^.vParent);        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           methodtable : pmethodnametable;           i : dword;           ovmt : PVmt;        begin           ovmt:=PVmt(self);           while assigned(ovmt) do             begin                methodtable:=pmethodnametable(ovmt^.vMethodTable);                if assigned(methodtable) then                  begin                     for i:=0 to methodtable^.count-1 do                       if ShortCompareText(methodtable^.entries[i].name^, name)=0 then                         begin                            MethodAddress:=methodtable^.entries[i].addr;                            exit;                         end;                  end;                ovmt := ovmt^.vParent;             end;           MethodAddress:=nil;        end;      class function TObject.MethodName(address : pointer) : shortstring;        var           methodtable : pmethodnametable;           i : dword;           ovmt : PVmt;        begin           ovmt:=PVmt(self);           while assigned(ovmt) do             begin                methodtable:=pmethodnametable(ovmt^.vMethodTable);                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;                ovmt := ovmt^.vParent;             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           ovmt: PVmt;           FieldTable: PFieldTable;           FieldInfo: PFieldInfo;           i: longint;        begin           if Length(name) > 0 then           begin             ovmt := PVmt(ClassType);             while ovmt <> nil do             begin               FieldTable := PFieldTable(ovmt^.vFieldTable);               if FieldTable <> nil then               begin                 FieldInfo := @FieldTable^.Fields[0];                 for i := 0 to FieldTable^.FieldCount - 1 do                 begin                   if ShortCompareText(FieldInfo^.Name, name) = 0 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 }               ovmt:=ovmt^.vParent;             end;           end;           fieldaddress:=nil;        end;      function TObject.SafeCallException(exceptobject : tobject;        exceptaddr : pointer) : HResult;        begin           safecallexception:=E_UNEXPECTED;        end;      class function TObject.ClassInfo : pointer;        begin          ClassInfo := PVmt(Self)^.vTypeInfo;        end;      class function TObject.ClassName : ShortString;        begin          ClassName := PVmt(Self)^.vClassName^;        end;      class function TObject.ClassNameIs(const name : string) : boolean;        begin        // call to ClassName inlined here, this eliminates stack and string copying.           ClassNameIs:=ShortCompareText(PVmt(Self)^.vClassName^, name) = 0;        end;      class function TObject.InheritsFrom(aclass : TClass) : Boolean;        var           vmt: PVmt;        begin           if assigned(aclass) then             begin               vmt:=PVmt(self);               while assigned(vmt) and (vmt <> PVmt(aclass)) do                 vmt := vmt^.vParent;               InheritsFrom := (vmt = PVmt(aclass));             end           else             inheritsFrom := False;        end;      class function TObject.stringmessagetable : pstringmessagetable;        begin          stringmessagetable:=PVmt(Self)^.vMsgStrPtr;        end;      type         tmessagehandler = procedure(var msg) of object;      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;           ovmt : PVmt;           msghandler : tmessagehandler;        begin           index:=dword(message);           ovmt := PVmt(ClassType);           while assigned(ovmt) do             begin                // See if we have messages at all in this class.                p:=ovmt^.vDynamicTable;                If Assigned(p) then                  begin                     msgtable:=pmsgtable(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                          TMethod(msghandler).Code:=msgtable[i].method;                          TMethod(msghandler).Data:=self;                          msghandler(message);                          exit;                       end;                  end;                ovmt:=ovmt^.vParent;             end;           DefaultHandler(message);        end;      procedure TObject.DispatchStr(var message);        type           PSizeUInt = ^SizeUInt;        var           name : shortstring;           count,i : longint;           msgstrtable : pmsgstrtable;           p: pstringmessagetable;           ovmt : PVmt;           msghandler : tmessagehandler;        begin           name:=pshortstring(@message)^;           ovmt:=PVmt(ClassType);           while assigned(ovmt) do           begin                p := ovmt^.vMsgStrPtr;                if (P<>Nil) and (p^.count<>0) then                  begin                  count:=p^.count;                  msgstrtable:=@p^.msgstrtable;                  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                          TMethod(msghandler).Code:=msgstrtable[i].method;                          TMethod(msghandler).Data:=self;                          msghandler(message);                          exit;                       end;                  end;                ovmt:=ovmt^.vParent;           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;{$ifdef CPU16}          TRecElemArray = packed array[1..Maxint div sizeof(TRecElem)-1] of TRecElem;{$else CPU16}          TRecElemArray = packed array[1..Maxint] of TRecElem;{$endif CPU16}          PRecRec = ^TRecRec;          TRecRec = record            Size,Count : Longint;            Elements : TRecElemArray;          end;        var           vmt  : PVmt;           temp : pbyte;           count,           i    : longint;{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}           recelem  : TRecElem;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}        begin           vmt := PVmt(ClassType);           while vmt<>nil do             begin               { This need to be included here, because Finalize()                 has should support for tkClass }               Temp:= vmt^.vInitTable;               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:= vmt^.vParent;             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; Corba: Boolean; 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 and not Corba then            IInterface(obj)._AddRef;        end;      function TObject.getinterface(const iid : tguid;out obj) : boolean;        begin          Result := getinterfacebyentry(self, getinterfaceentry(iid), false, obj);        end;      function TObject.getinterfacebystr(const iidstr : shortstring;out obj) : boolean;        begin          Result := getinterfacebyentry(self, getinterfaceentrybystr(iidstr), true, obj);        end;      function TObject.getinterface(const iidstr : shortstring;out obj) : boolean;        begin          Result := getinterfacebystr(iidstr,obj);        end;      class function TObject.getinterfaceentry(const iid : tguid) : pinterfaceentry;        var          i: longint;          intftable: pinterfacetable;          ovmt: PVmt;        begin          ovmt := PVmt(Self);          while Assigned(ovmt) and (ovmt^.vIntfTable <> @emptyintf) do          begin            intftable:=ovmt^.vIntfTable;            if assigned(intftable) then            begin              for i:=0 to intftable^.EntryCount-1 do              begin                result:=@intftable^.Entries[i];                if assigned(Result^.iid) and IsGUIDEqual(Result^.iid^,iid) then                  Exit;              end;            end;            ovmt := ovmt^.vParent;          end;          result := nil;        end;      class function TObject.getinterfaceentrybystr(const iidstr : shortstring) : pinterfaceentry;        var          i: longint;          intftable: pinterfacetable;          ovmt: PVmt;        begin          ovmt := PVmt(Self);          while Assigned(ovmt) and (ovmt^.vIntfTable <> @emptyintf) do          begin            intftable:=ovmt^.vIntfTable;            if assigned(intftable) then            begin              for i:=0 to intftable^.EntryCount-1 do              begin                result:=@intftable^.Entries[i];                if assigned(result^.iidstr) and (result^.iidstr^ = iidstr) then                  Exit;              end;            end;            ovmt := ovmt^.vParent;          end;          result:=nil;        end;      class function TObject.getinterfacetable : pinterfacetable;        begin          getinterfacetable:=PVmt(Self)^.vIntfTable;        end;      class function TObject.UnitName : ansistring;        type          // from the typinfo unit          TClassTypeInfo = {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record            ClassType: TClass;            ParentInfo: Pointer;            PropCount: SmallInt;            UnitName: ShortString;          end;          PClassTypeInfo = ^TClassTypeInfo;        var          classtypeinfo: PClassTypeInfo;        begin          classtypeinfo:=ClassInfo;          if Assigned(classtypeinfo) then          begin            // offset PTypeInfo by Length(Name) + 2 (ShortString length byte + SizeOf(Kind))            inc(Pointer(classtypeinfo), PByte(Pointer(classtypeinfo)+1)^ + 2);            {$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}            classtypeinfo:=align(classtypeinfo,sizeof(classtypeinfo));            {$endif}            result:=classtypeinfo^.UnitName;          end          else            result:='';        end;      function TObject.Equals(Obj: TObject) : boolean;        begin          result:=Obj=Self;        end;      function TObject.GetHashCode: PtrInt;        begin          result:=PtrInt(Self);        end;      function TObject.ToString: ansistring;        begin          result:=ClassName;        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;{****************************************************************************                               TContainedOBJECT****************************************************************************}    function TContainedObject.QueryInterface(            const iid : tguid;out obj) : longint; stdcall;    begin      if getinterface(iid,obj) then        result:=0      else        result:=longint(E_NOINTERFACE);    end;{****************************************************************************                             Exception Support****************************************************************************}{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}{$i except.inc}{$endif FPC_HAS_FEATURE_EXCEPTIONS}
 |