![]() |
COM-Objekte innerhalb von Threads
Hi,
der ganze OLE/COM-Kram treibt mich gerade zum Wahnsinn. Ich habe einen Thread, der im Execute Folgendes stehen hat:
Delphi-Quellcode:
Jetzt fliegt mir irgendwann eine EOleSysError-Exception (Erste Gelegenheit für Exception bei $7C812A5B. Exception-Klasse EOleSysError mit Meldung 'Starten des Servers fehlgeschlagen'. Prozess InfoTunes.exe (1708)) an den Kopf, die an folgender Stelle geraiset wird:
CoInitializeEx(nil, COINIT_MULTITHREADED);
TunesApp := CoiTunesApp.Create; try repeat {Tue was mit dem TunesApp-Interface} until false; finally TunesApp := nil; CoUninitialize; end;
Delphi-Quellcode:
Durch das unmoegliche Debugging von Threads kann ich absolut nicht einschaetzen, warum das alles so passiert und wann genau es zu dieser Exception kommt. Ich vermute, dass der COM-Server ist gar nicht mag, dass in jedem Thread eine neue Referenz erzeugt werden soll.
class function CoiTunesApp.Create: IiTunes;
begin Result := CreateComObject(CLASS_iTunesApp) as IiTunes; end; Weiss jemand, woran das liegen koennte und wie man das Problem loest? Bin fuer jeden Vorschlag dankbar. Das treibt mich schon den ganzen Tag in den Wahnsinn. Chris |
Re: COM-Objekte innerhalb von Threads
Du meinst du tust permanent diese Schleife ablaufen lassen?
Also entweder hat das ITunes-Interface irgendeine Quit/Exit-Methode die man Aufrufen muß bevor man das Interface freigibt oder ITunes hat Speicherlücken die vielen Anforderungen irgendeine Ressourcen (GDI, FileHandels, ...) volllaufen läuft. Für Fall a, einfach dieses Quit/Exit aufrufen. Für Fall b, Interface "cachen" und nicht permanent anfordern/freigeben oder auf fehlerbereinigte Version von ITunes warten. |
Re: COM-Objekte innerhalb von Threads
Hi Bernhard,
Zitat:
Zitat:
Und wie gesagt funktioniert es problemlos, wenn ich keinen Dienst verwende, sondern eine einfache VCL-Anwendung. :? Danke für deine Hilfe, Chris |
Re: COM-Objekte innerhalb von Threads
Rufe im Thread.Create
![]() ![]() Du kannst das natürlich auch an anderer Stelle tun, aber alle COM-Aufrufe und Objekte müssen zwischen Init und Uninit stattfinden. ...:cat:... |
Re: COM-Objekte innerhalb von Threads
Hi,
habe ich bereits drin. Habe es jetzt korrigiert und in den Service geschrieben:
Delphi-Quellcode:
Der Service startet immer noch nicht, d.h. er lädt und lädt (in der Systemsteuerung => Dienste) und gibt nach einiger Zeit folgende Fehlermeldung aus:
procedure TInfoTunesService.ServiceCreate(Sender: TObject);
begin CoInitialize(nil); FTunesInstance := CoiTunesApp.Create; end; procedure TInfoTunesService.ServiceDestroy(Sender: TObject); begin CoUninitialize; end; Zitat:
Chris |
Re: COM-Objekte innerhalb von Threads
Der Aufruf muss in jeden Thread, evtl. auch in den Service.
Wenn das alles nichts hilft, dan schaue Dir die madExcept Komponenten an - Tutorial in der Tut-Sparte. Für die private Nutzung sind die kostenlos. ...:cat:... |
Re: COM-Objekte innerhalb von Threads
Hi,
ah, eine Veränderung. Nachdem ich nun das Erstellen des COM-Objekts in das OnStart verlagert habe und CoInitializeEx mit COINIT_MULTITHREADED verwende, startet der Dienst zumindest:
Delphi-Quellcode:
Jedoch bekomme ich jetzt folgende Nachricht von Windows beim Starten des Dienstes:
procedure TInfoTunesService.ServiceCreate(Sender: TObject);
begin CoInitializeEx(nil, COINIT_MULTITHREADED); end; procedure TInfoTunesService.ServiceDestroy(Sender: TObject); begin CoUninitialize; end; procedure TInfoTunesService.ServiceStart(Sender: TService; var Started: Boolean); begin FTunesInstance := CoiTunesApp.Create; FTCPServer := TTCPDaemon.Create; FTCPServer.FTunesInstance := FTunesInstance; Started := true; end; Zitat:
Delphi-Quellcode:
Chris
procedure TInfoTunesService.ServiceExecute(Sender: TService);
begin // Endlosschleife ftw while not Terminated do begin ServiceThread.ProcessRequests(True); end; end; Edit: madExcept ist ein guter Tipp, danke. Dummerweise funktioniert das nicht, weil ich im Moment mit Turbo Delphi entwickle und er das Paket nicht laden kann (ich nehme an, weil man bekanntlich keine Dritt-Anbieter-Komponenten installieren kann). |
Re: COM-Objekte innerhalb von Threads
:wall: Man sollte gelegentlich auch mal in's Eventlog gucken:
Zitat:
Und es funktioniert immer noch problemlos ohne den Dienst, in einer normalen Anwendung. Das kann doch irgendwie nicht sein. :/ Chris |
Re: COM-Objekte innerhalb von Threads
Ich hab iTunes bisher auch nur aus einer normalen Applikation heraus benutzt, vielleicht mag iTunes es nicht, wenn der Aufrufer kein Window hat? :) Was auch noch wichtig ist: iTunes reagiert sehr träge auf Aufrufe, Services dürfen wenn ich mich recht entsinne nur kleine Timeslices in Anspruch nehmen, meine Testapplikation hängt z.B. mehrere Sekunden (singlethreaded, Aufruf im Mainthread, dient nur zum Testen) bevor das CoCreate fertig ist. Sowas könnte bei einem Service fatal sein.
|
Re: COM-Objekte innerhalb von Threads
Inzwischen vermute ich das auch. Dann muss ich wohl das ganze ein wenig umbasteln. :wall:
Na gut, danke für eure Hilfe und falls jemand doch ne Lösung kennt: immer her damit. Chris |
Re: COM-Objekte innerhalb von Threads
Ich würde das Interface für iTunes nur einmal erzeugen und zwar im Hauptthread.
Dann gibt man das Interface beim Erzeugen des Thread-Objekts "einfach" weiter. Wenn 5 Threads das Interface benützen, dann steht RefCount auf 6 (5 Thread + die Referenz im Hauptthread). Wenn Interface-Pointer von einem Thread an einen anderen Thread weitergereicht werden soll, dann muss man Marshaling anwenden. siehe: ![]() |
Re: COM-Objekte innerhalb von Threads
Hi shmia,
wie soll denn der Pointer weitergegeben werden? Reicht ein einfaches zuweisen, wie ich es oben schon probiert habe? Das Marshalling versuche ich nachher. Danke, Chris |
Re: COM-Objekte innerhalb von Threads
Zitat:
Auf der obigen Webseite werden 2 Techniken gezeigt; die 2. Technik mit dem Hilfs-Interface IGlobalInterfaceTable ist für deinen Zweck am Besten. |
Re: COM-Objekte innerhalb von Threads
Hi shmia,
ja, die Fehlermeldung hatte ich zwischen durch auch mal, aber wie gesagt, kommt jetzt im Moment der Fehler ja in erster Linie bei CoiTunesApp.Create und das passiert ja vor dem marshalen, oder? Chris |
Re: COM-Objekte innerhalb von Threads
Hi,
hab es jetzt auch so implementiert, wie es auf der Seite gezeigt wird. Der Fehler beim Erzeugen ist immer noch da. Und wie gesagt nehme ich an, dass iTunes es nicht mag, wenn man es über Dienste ansteuert. Chris |
Re: COM-Objekte innerhalb von Threads
Falls es nur am fehlenden Fenster liegt, kannst Du Dir ja ein unsichtbares Fenster mittels AllocateHwnd holen (das musst Du aber auch mit UnAllocateHwnd wieder freigeben!), nen Test kann ja nicht schaden.
|
Re: COM-Objekte innerhalb von Threads
Und was soll ich mit dem Fenster machen? Woher will iTunes denn wissen, ob ich ein Fenster habe oder nicht? Es wird nirgendwo ein Parameter gefordert, der so einen Schluss zulassen würde.
Chris Edit: Einfach so ein Handle zu erzeugen, hilft aber nicht. |
Re: COM-Objekte innerhalb von Threads
iTunes könnte z.B. sich das Top Level Window des Callers holen wollen (wozu auch immer, seitdem ich Entwicklerfeuerwehr spiel, halte ich keine Idee mehr für unmöglich *g*), dazu musst Du keinen Parameter übergeben.
Edit: Na jut, dann hat sich das wohl erledigt. Naja, ist nicht das erste Mal dass iTunes für graue Haare sorgt :) |
Alle Zeitangaben in WEZ +1. Es ist jetzt 14:39 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz