mshtml: Fix loading pages in containers without document.
[wine] / dlls / mshtml / nsio.c
1 /*
2  * Copyright 2006 Jacek Caban for CodeWeavers
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  */
18
19 #include "config.h"
20
21 #include <stdarg.h>
22
23 #define COBJMACROS
24
25 #include "windef.h"
26 #include "winbase.h"
27 #include "winuser.h"
28 #include "ole2.h"
29 #include "shlguid.h"
30
31 #include "wine/debug.h"
32 #include "wine/unicode.h"
33
34 #include "mshtml_private.h"
35
36 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
37
38 #define LOAD_INITIAL_DOCUMENT_URI 0x80000
39
40 #define NS_IOSERVICE_CLASSNAME "nsIOService"
41 #define NS_IOSERVICE_CONTRACTID "@mozilla.org/network/io-service;1"
42
43 static const IID NS_IOSERVICE_CID =
44     {0x9ac9e770, 0x18bc, 0x11d3, {0x93, 0x37, 0x00, 0x10, 0x4b, 0xa0, 0xfd, 0x40}};
45
46 static nsIIOService *nsio = NULL;
47
48 typedef struct {
49     const nsIWineURIVtbl *lpWineURIVtbl;
50
51     LONG ref;
52
53     nsIURI *uri;
54     NSContainer *container;
55     IMoniker *mon;
56     LPSTR spec;
57 } nsURI;
58
59 #define NSURI(x)         ((nsIURI*)            &(x)->lpWineURIVtbl)
60
61 static nsresult create_uri(nsIURI*,NSContainer*,nsIURI**);
62
63 static BOOL exec_shldocvw_67(HTMLDocument *doc, LPCWSTR url)
64 {
65     IOleCommandTarget *cmdtrg = NULL;
66     HRESULT hres;
67
68     hres = IOleClientSite_QueryInterface(doc->client, &IID_IOleCommandTarget,
69                                          (void**)&cmdtrg);
70     if(SUCCEEDED(hres)) {
71         VARIANT varUrl, varRes;
72
73         V_VT(&varUrl) = VT_BSTR;
74         V_BSTR(&varUrl) = SysAllocString(url);
75         V_VT(&varRes) = VT_BOOL;
76
77         hres = IOleCommandTarget_Exec(cmdtrg, &CGID_ShellDocView, 67, 0, &varUrl, &varRes);
78
79         IOleCommandTarget_Release(cmdtrg);
80         SysFreeString(V_BSTR(&varUrl));
81
82         if(SUCCEEDED(hres) && !V_BOOL(&varRes)) {
83             TRACE("got VARIANT_FALSE, do not load\n");
84             return FALSE;
85         }
86     }
87
88     return TRUE;
89 }
90
91 static BOOL handle_uri(NSContainer *container, nsChannel *channel, LPCWSTR uri)
92 {
93     IServiceProvider *service_provider;
94     HTMLDocument *doc = container->doc;
95     DWORD hlnf = 0;
96     HRESULT hres;
97
98     if(!doc) {
99         NSContainer *container_iter = container;
100
101         hlnf = HLNF_OPENINNEWWINDOW;
102         while(!container_iter->doc)
103             container_iter = container_iter->parent;
104         doc = container_iter->doc;
105     }
106
107     if(!hlnf && !exec_shldocvw_67(doc, uri))
108         return FALSE;
109
110     hres = IOleClientSite_QueryInterface(doc->client, &IID_IServiceProvider,
111                                          (void**)&service_provider);
112     if(SUCCEEDED(hres)) {
113         IHlinkFrame *hlink_frame;
114
115         hres = IServiceProvider_QueryService(service_provider, &IID_IHlinkFrame,
116                                              &IID_IHlinkFrame, (void**)&hlink_frame);
117         IServiceProvider_Release(service_provider);
118         if(SUCCEEDED(hres)) {
119             hlink_frame_navigate(doc, hlink_frame, uri, channel->post_data_stream, hlnf);
120             IHlinkFrame_Release(hlink_frame);
121
122             return FALSE;
123         }
124     }
125
126     return TRUE;
127 }
128
129 static BOOL before_async_open(nsChannel *channel, NSContainer *container)
130 {
131     nsACString uri_str;
132     const char *uria;
133     LPWSTR uri;
134     DWORD len;
135     BOOL ret;
136
137     nsACString_Init(&uri_str, NULL);
138     nsIWineURI_GetSpec(channel->uri, &uri_str);
139     nsACString_GetData(&uri_str, &uria, NULL);
140     len = MultiByteToWideChar(CP_ACP, 0, uria, -1, NULL, 0);
141     uri = mshtml_alloc(len*sizeof(WCHAR));
142     MultiByteToWideChar(CP_ACP, 0, uria, -1, uri, len);
143     nsACString_Finish(&uri_str);
144
145     ret = handle_uri(container, channel, uri);
146
147     mshtml_free(uri);
148
149     return ret;
150 }
151
152 #define NSCHANNEL_THIS(iface) DEFINE_THIS(nsChannel, HttpChannel, iface)
153
154 static nsresult NSAPI nsChannel_QueryInterface(nsIHttpChannel *iface, nsIIDRef riid, nsQIResult result)
155 {
156     nsChannel *This = NSCHANNEL_THIS(iface);
157
158     *result = NULL;
159
160     if(IsEqualGUID(&IID_nsISupports, riid)) {
161         TRACE("(%p)->(IID_nsISupports %p)\n", This, result);
162         *result = NSCHANNEL(This);
163     }else if(IsEqualGUID(&IID_nsIRequest, riid)) {
164         TRACE("(%p)->(IID_nsIRequest %p)\n", This, result);
165         *result = NSCHANNEL(This);
166     }else if(IsEqualGUID(&IID_nsIChannel, riid)) {
167         TRACE("(%p)->(IID_nsIChannel %p)\n", This, result);
168         *result = NSCHANNEL(This);
169     }else if(This->http_channel && IsEqualGUID(&IID_nsIHttpChannel, riid)) {
170         TRACE("(%p)->(IID_nsIHttpChannel %p)\n", This, result);
171         *result = NSHTTPCHANNEL(This);
172     }else if(IsEqualGUID(&IID_nsIUploadChannel, riid)) {
173         TRACE("(%p)->(IID_nsIUploadChannel %p)\n", This, result);
174         *result = NSUPCHANNEL(This);
175     }
176
177     if(*result) {
178         nsIChannel_AddRef(NSCHANNEL(This));
179         return NS_OK;
180     }
181
182     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), result);
183
184     if(This->channel)
185         return nsIChannel_QueryInterface(This->channel, riid, result);
186     return NS_NOINTERFACE;
187 }
188
189 static nsrefcnt NSAPI nsChannel_AddRef(nsIHttpChannel *iface)
190 {
191     nsChannel *This = NSCHANNEL_THIS(iface);
192     nsrefcnt ref = InterlockedIncrement(&This->ref);
193
194     TRACE("(%p) ref=%d\n", This, ref);
195
196     return ref;
197 }
198
199 static nsrefcnt NSAPI nsChannel_Release(nsIHttpChannel *iface)
200 {
201     nsChannel *This = NSCHANNEL_THIS(iface);
202     LONG ref = InterlockedDecrement(&This->ref);
203
204     if(!ref) {
205         nsIWineURI_Release(This->uri);
206         if(This->channel)
207             nsIChannel_Release(This->channel);
208         if(This->http_channel)
209             nsIHttpChannel_Release(This->http_channel);
210         if(This->post_data_stream)
211             nsIInputStream_Release(This->post_data_stream);
212         if(This->load_group)
213             nsILoadGroup_Release(This->load_group);
214         if(This->notif_callback)
215             nsIInterfaceRequestor_Release(This->notif_callback);
216         if(This->original_uri)
217             nsIURI_Release(This->original_uri);
218         mshtml_free(This->content);
219         mshtml_free(This);
220     }
221
222     return ref;
223 }
224
225 static nsresult NSAPI nsChannel_GetName(nsIHttpChannel *iface, nsACString *aName)
226 {
227     nsChannel *This = NSCHANNEL_THIS(iface);
228
229     TRACE("(%p)->(%p)\n", This, aName);
230
231     if(This->channel)
232         return nsIChannel_GetName(This->channel, aName);
233
234     FIXME("default action not implemented\n");
235     return NS_ERROR_NOT_IMPLEMENTED;
236 }
237
238 static nsresult NSAPI nsChannel_IsPending(nsIHttpChannel *iface, PRBool *_retval)
239 {
240     nsChannel *This = NSCHANNEL_THIS(iface);
241
242     TRACE("(%p)->(%p)\n", This, _retval);
243
244     if(This->channel)
245         return nsIChannel_IsPending(This->channel, _retval);
246
247     FIXME("default action not implemented\n");
248     return NS_ERROR_NOT_IMPLEMENTED;
249 }
250
251 static nsresult NSAPI nsChannel_GetStatus(nsIHttpChannel *iface, nsresult *aStatus)
252 {
253     nsChannel *This = NSCHANNEL_THIS(iface);
254
255     TRACE("(%p)->(%p)\n", This, aStatus);
256
257     if(This->channel)
258         return nsIChannel_GetStatus(This->channel, aStatus);
259
260     TRACE("returning NS_OK\n");
261     return *aStatus = NS_OK;
262 }
263
264 static nsresult NSAPI nsChannel_Cancel(nsIHttpChannel *iface, nsresult aStatus)
265 {
266     nsChannel *This = NSCHANNEL_THIS(iface);
267
268     TRACE("(%p)->(%08x)\n", This, aStatus);
269
270     if(This->channel)
271         return nsIChannel_Cancel(This->channel, aStatus);
272
273     FIXME("default action not implemented\n");
274     return NS_ERROR_NOT_IMPLEMENTED;
275 }
276
277 static nsresult NSAPI nsChannel_Suspend(nsIHttpChannel *iface)
278 {
279     nsChannel *This = NSCHANNEL_THIS(iface);
280
281     TRACE("(%p)\n", This);
282
283     if(This->channel)
284         return nsIChannel_Suspend(This->channel);
285
286     FIXME("default action not implemented\n");
287     return NS_ERROR_NOT_IMPLEMENTED;
288 }
289
290 static nsresult NSAPI nsChannel_Resume(nsIHttpChannel *iface)
291 {
292     nsChannel *This = NSCHANNEL_THIS(iface);
293
294     TRACE("(%p)\n", This);
295
296     if(This->channel)
297         return nsIChannel_Resume(This->channel);
298
299     FIXME("default action not implemented\n");
300     return NS_ERROR_NOT_IMPLEMENTED;
301 }
302
303 static nsresult NSAPI nsChannel_GetLoadGroup(nsIHttpChannel *iface, nsILoadGroup **aLoadGroup)
304 {
305     nsChannel *This = NSCHANNEL_THIS(iface);
306
307     TRACE("(%p)->(%p)\n", This, aLoadGroup);
308
309     if(This->load_group)
310         nsILoadGroup_AddRef(This->load_group);
311
312     *aLoadGroup = This->load_group;
313     return NS_OK;
314 }
315
316 static nsresult NSAPI nsChannel_SetLoadGroup(nsIHttpChannel *iface, nsILoadGroup *aLoadGroup)
317 {
318     nsChannel *This = NSCHANNEL_THIS(iface);
319
320     TRACE("(%p)->(%p)\n", This, aLoadGroup);
321
322     if(This->load_group)
323         nsILoadGroup_Release(This->load_group);
324     if(aLoadGroup)
325         nsILoadGroup_AddRef(aLoadGroup);
326
327     This->load_group = aLoadGroup;
328
329     if(This->channel)
330         return nsIChannel_SetLoadGroup(This->channel, aLoadGroup);
331     return NS_OK;
332 }
333
334 static nsresult NSAPI nsChannel_GetLoadFlags(nsIHttpChannel *iface, nsLoadFlags *aLoadFlags)
335 {
336     nsChannel *This = NSCHANNEL_THIS(iface);
337
338     TRACE("(%p)->(%p)\n", This, aLoadFlags);
339
340     *aLoadFlags = This->load_flags;
341     return NS_OK;
342 }
343
344 static nsresult NSAPI nsChannel_SetLoadFlags(nsIHttpChannel *iface, nsLoadFlags aLoadFlags)
345 {
346     nsChannel *This = NSCHANNEL_THIS(iface);
347
348     TRACE("(%p)->(%08x)\n", This, aLoadFlags);
349
350     This->load_flags = aLoadFlags;
351
352     if(This->channel)
353         return nsIChannel_SetLoadFlags(This->channel, aLoadFlags);
354     return NS_OK;
355 }
356
357 static nsresult NSAPI nsChannel_GetOriginalURI(nsIHttpChannel *iface, nsIURI **aOriginalURI)
358 {
359     nsChannel *This = NSCHANNEL_THIS(iface);
360
361     TRACE("(%p)->(%p)\n", This, aOriginalURI);
362
363     if(This->original_uri)
364         nsIURI_AddRef(This->original_uri);
365
366     *aOriginalURI = This->original_uri;
367     return NS_OK;
368 }
369
370 static nsresult NSAPI nsChannel_SetOriginalURI(nsIHttpChannel *iface, nsIURI *aOriginalURI)
371 {
372     nsChannel *This = NSCHANNEL_THIS(iface);
373
374     TRACE("(%p)->(%p)\n", This, aOriginalURI);
375
376     if(This->original_uri)
377         nsIURI_Release(This->original_uri);
378
379     nsIURI_AddRef(aOriginalURI);
380     This->original_uri = aOriginalURI;
381
382     if(This->channel)
383         return nsIChannel_SetOriginalURI(This->channel, aOriginalURI);
384     return NS_OK;
385 }
386
387 static nsresult NSAPI nsChannel_GetURI(nsIHttpChannel *iface, nsIURI **aURI)
388 {
389     nsChannel *This = NSCHANNEL_THIS(iface);
390
391     TRACE("(%p)->(%p)\n", This, aURI);
392
393     nsIWineURI_AddRef(This->uri);
394     *aURI = (nsIURI*)This->uri;
395
396     return NS_OK;
397 }
398
399 static nsresult NSAPI nsChannel_GetOwner(nsIHttpChannel *iface, nsISupports **aOwner)
400 {
401     nsChannel *This = NSCHANNEL_THIS(iface);
402
403     TRACE("(%p)->(%p)\n", This, aOwner);
404
405     if(This->channel)
406         return nsIChannel_GetOwner(This->channel, aOwner);
407
408     FIXME("default action not implemented\n");
409     return NS_ERROR_NOT_IMPLEMENTED;
410 }
411
412 static nsresult NSAPI nsChannel_SetOwner(nsIHttpChannel *iface, nsISupports *aOwner)
413 {
414     nsChannel *This = NSCHANNEL_THIS(iface);
415
416     TRACE("(%p)->(%p)\n", This, aOwner);
417
418     if(This->channel)
419         return nsIChannel_SetOwner(This->channel, aOwner);
420
421     FIXME("default action not implemented\n");
422     return NS_ERROR_NOT_IMPLEMENTED;
423 }
424
425 static nsresult NSAPI nsChannel_GetNotificationCallbacks(nsIHttpChannel *iface,
426         nsIInterfaceRequestor **aNotificationCallbacks)
427 {
428     nsChannel *This = NSCHANNEL_THIS(iface);
429
430     TRACE("(%p)->(%p)\n", This, aNotificationCallbacks);
431
432     if(This->notif_callback)
433         nsIInterfaceRequestor_AddRef(This->notif_callback);
434     *aNotificationCallbacks = This->notif_callback;
435
436     return NS_OK;
437 }
438
439 static nsresult NSAPI nsChannel_SetNotificationCallbacks(nsIHttpChannel *iface,
440         nsIInterfaceRequestor *aNotificationCallbacks)
441 {
442     nsChannel *This = NSCHANNEL_THIS(iface);
443
444     TRACE("(%p)->(%p)\n", This, aNotificationCallbacks);
445
446     if(This->notif_callback)
447         nsIInterfaceRequestor_Release(This->notif_callback);
448     if(aNotificationCallbacks)
449         nsIInterfaceRequestor_AddRef(aNotificationCallbacks);
450
451     This->notif_callback = aNotificationCallbacks;
452
453     if(This->channel)
454         return nsIChannel_SetNotificationCallbacks(This->channel, aNotificationCallbacks);
455     return NS_OK;
456 }
457
458 static nsresult NSAPI nsChannel_GetSecurityInfo(nsIHttpChannel *iface, nsISupports **aSecurityInfo)
459 {
460     nsChannel *This = NSCHANNEL_THIS(iface);
461
462     TRACE("(%p)->(%p)\n", This, aSecurityInfo);
463
464     if(This->channel)
465         return nsIChannel_GetSecurityInfo(This->channel, aSecurityInfo);
466
467     FIXME("default action not implemented\n");
468     return NS_ERROR_NOT_IMPLEMENTED;
469 }
470
471 static nsresult NSAPI nsChannel_GetContentType(nsIHttpChannel *iface, nsACString *aContentType)
472 {
473     nsChannel *This = NSCHANNEL_THIS(iface);
474
475     TRACE("(%p)->(%p)\n", This, aContentType);
476
477     if(This->content) {
478         nsACString_SetData(aContentType, This->content);
479         return S_OK;
480     }
481
482     if(This->channel)
483         return nsIChannel_GetContentType(This->channel, aContentType);
484
485     TRACE("returning default text/html\n");
486     nsACString_SetData(aContentType, "text/html");
487     return NS_OK;
488 }
489
490 static nsresult NSAPI nsChannel_SetContentType(nsIHttpChannel *iface,
491                                                const nsACString *aContentType)
492 {
493     nsChannel *This = NSCHANNEL_THIS(iface);
494
495     TRACE("(%p)->(%p)\n", This, aContentType);
496
497     if(This->channel)
498         return nsIChannel_SetContentType(This->channel, aContentType);
499
500     FIXME("default action not implemented\n");
501     return NS_ERROR_NOT_IMPLEMENTED;
502 }
503
504 static nsresult NSAPI nsChannel_GetContentCharset(nsIHttpChannel *iface,
505                                                   nsACString *aContentCharset)
506 {
507     nsChannel *This = NSCHANNEL_THIS(iface);
508
509     TRACE("(%p)->(%p)\n", This, aContentCharset);
510
511     if(This->channel)
512         return nsIChannel_GetContentCharset(This->channel, aContentCharset);
513
514     FIXME("default action not implemented\n");
515     return NS_ERROR_NOT_IMPLEMENTED;
516 }
517
518 static nsresult NSAPI nsChannel_SetContentCharset(nsIHttpChannel *iface,
519                                                   const nsACString *aContentCharset)
520 {
521     nsChannel *This = NSCHANNEL_THIS(iface);
522
523     TRACE("(%p)->(%p)\n", This, aContentCharset);
524
525     if(This->channel)
526         return nsIChannel_SetContentCharset(This->channel, aContentCharset);
527
528     FIXME("default action not implemented\n");
529     return NS_ERROR_NOT_IMPLEMENTED;
530 }
531
532 static nsresult NSAPI nsChannel_GetContentLength(nsIHttpChannel *iface, PRInt32 *aContentLength)
533 {
534     nsChannel *This = NSCHANNEL_THIS(iface);
535
536     TRACE("(%p)->(%p)\n", This, aContentLength);
537
538     if(This->channel)
539         return nsIChannel_GetContentLength(This->channel, aContentLength);
540
541     FIXME("default action not implemented\n");
542     return NS_ERROR_NOT_IMPLEMENTED;
543 }
544
545 static nsresult NSAPI nsChannel_SetContentLength(nsIHttpChannel *iface, PRInt32 aContentLength)
546 {
547     nsChannel *This = NSCHANNEL_THIS(iface);
548
549     TRACE("(%p)->(%d)\n", This, aContentLength);
550
551     if(This->channel)
552         return nsIChannel_SetContentLength(This->channel, aContentLength);
553
554     FIXME("default action not implemented\n");
555     return NS_ERROR_NOT_IMPLEMENTED;
556 }
557
558 static nsresult NSAPI nsChannel_Open(nsIHttpChannel *iface, nsIInputStream **_retval)
559 {
560     nsChannel *This = NSCHANNEL_THIS(iface);
561
562     TRACE("(%p)->(%p)\n", This, _retval);
563
564     if(This->channel)
565         return nsIChannel_Open(This->channel, _retval);
566
567     FIXME("default action not implemented\n");
568     return NS_ERROR_NOT_IMPLEMENTED;
569 }
570
571 static BOOL do_load_from_moniker_hack(nsChannel *This)
572 {
573     PRBool b = FALSE;
574
575     /* 
576      * We should always load the page from IMoniker, but Wine is not yet
577      * ready for this. This function is a heuristic, that decides which
578      * way of loading is better (Gecko implementation or IMoniker). The
579      * aim is to always return TRUE.
580      */
581
582     /* Load from moniker if there is no Gecko channel available */
583     if(!This->channel)
584         return TRUE;
585
586     /* Load about protocol from moniker */
587     nsIWineURI_SchemeIs(This->uri, "about", &b);
588     if(b)
589         return TRUE;
590
591     return FALSE;
592 }
593
594 static nsresult NSAPI nsChannel_AsyncOpen(nsIHttpChannel *iface, nsIStreamListener *aListener,
595                                           nsISupports *aContext)
596 {
597     nsChannel *This = NSCHANNEL_THIS(iface);
598     BSCallback *bscallback;
599     nsIWineURI *wine_uri;
600     IMoniker *mon;
601     nsresult nsres;
602
603     TRACE("(%p)->(%p %p)\n", This, aListener, aContext);
604
605     if(This->load_flags & LOAD_INITIAL_DOCUMENT_URI) {
606         NSContainer *container;
607
608         nsIWineURI_GetNSContainer(This->uri, &container);
609         if(!container) {
610             ERR("container = NULL\n");
611             return NS_ERROR_UNEXPECTED;
612         }
613
614         if(container->bscallback) {
615             nsIChannel_AddRef(NSCHANNEL(This));
616             container->bscallback->nschannel = This;
617
618             nsIStreamListener_AddRef(aListener);
619             container->bscallback->nslistener = aListener;
620
621             if(aContext) {
622                 nsISupports_AddRef(aContext);
623                 container->bscallback->nscontext = aContext;
624             }
625
626             nsIWebBrowserChrome_Release(NSWBCHROME(container));
627
628             if(do_load_from_moniker_hack(This)) {
629                 if(This->load_group) {
630                     nsres = nsILoadGroup_AddRequest(This->load_group,
631                                                     (nsIRequest*)NSCHANNEL(This), NULL);
632
633                     if(NS_FAILED(nsres))
634                         ERR("AddRequest failed:%08x\n", nsres);
635                 }
636
637                 return WINE_NS_LOAD_FROM_MONIKER;
638             }
639         }else if(container->doc) {
640             BOOL cont = before_async_open(This, container);
641             nsIWebBrowserChrome_Release(NSWBCHROME(container));
642
643             if(!cont) {
644                 TRACE("canceled\n");
645                 return NS_ERROR_UNEXPECTED;
646             }
647         }else {
648             nsIWebBrowserChrome_Release(NSWBCHROME(container));
649             return This->channel
650                 ?  nsIChannel_AsyncOpen(This->channel, aListener, aContext)
651                 : NS_ERROR_UNEXPECTED;
652         }
653     }
654
655     if(This->channel) {
656         if(This->post_data_stream) {
657             nsIUploadChannel *upload_channel;
658
659             nsres = nsIChannel_QueryInterface(This->channel, &IID_nsIUploadChannel,
660                                           (void**)&upload_channel);
661             if(NS_SUCCEEDED(nsres)) {
662                 nsACString empty_string;
663                 nsACString_Init(&empty_string, "");
664
665                 nsres = nsIUploadChannel_SetUploadStream(upload_channel, This->post_data_stream,
666                                                          &empty_string, -1);
667                 nsIUploadChannel_Release(upload_channel);
668                 if(NS_FAILED(nsres))
669                     WARN("SetUploadStream failed: %08x\n", nsres);
670
671                 nsACString_Finish(&empty_string);
672             }
673         }
674
675         nsres = nsIChannel_AsyncOpen(This->channel, aListener, aContext);
676
677         if(NS_FAILED(nsres) && (This->load_flags & LOAD_INITIAL_DOCUMENT_URI))
678             return WINE_NS_LOAD_FROM_MONIKER;
679         return nsres;
680     }
681
682     TRACE("channel == NULL\n");
683
684     if(!This->original_uri) {
685         ERR("original_uri == NULL\n");
686         return NS_ERROR_UNEXPECTED;
687     }
688
689     nsres = nsIURI_QueryInterface(This->original_uri, &IID_nsIWineURI, (void**)&wine_uri);
690     if(NS_FAILED(nsres)) {
691         ERR("Could not get nsIWineURI: %08x\n", nsres);
692         return NS_ERROR_UNEXPECTED;
693     }
694
695     nsIWineURI_GetMoniker(wine_uri, &mon);
696     nsIWineURI_Release(wine_uri);
697
698     if(!mon) {
699         WARN("mon == NULL\n");
700         return NS_ERROR_UNEXPECTED;
701     }
702
703     bscallback = create_bscallback(mon);
704     IMoniker_Release(mon);
705
706     nsIChannel_AddRef(NSCHANNEL(This));
707     bscallback->nschannel = This;
708
709     nsIStreamListener_AddRef(aListener);
710     bscallback->nslistener = aListener;
711
712     if(aContext) {
713         nsISupports_AddRef(aContext);
714         bscallback->nscontext = aContext;
715     }
716
717     if(This->load_group) {
718         nsres = nsILoadGroup_AddRequest(This->load_group,
719                 (nsIRequest*)NSCHANNEL(This), NULL);
720
721         if(NS_FAILED(nsres))
722             ERR("AddRequest failed:%08x\n", nsres);
723     }
724
725     start_binding(bscallback);
726     IBindStatusCallback_Release(STATUSCLB(bscallback));
727
728     return NS_OK;
729 }
730
731 static nsresult NSAPI nsChannel_GetRequestMethod(nsIHttpChannel *iface, nsACString *aRequestMethod)
732 {
733     nsChannel *This = NSCHANNEL_THIS(iface);
734
735     TRACE("(%p)->(%p)\n", This, aRequestMethod);
736
737     if(This->http_channel)
738         return nsIHttpChannel_GetRequestMethod(This->http_channel, aRequestMethod);
739
740     return NS_ERROR_NOT_IMPLEMENTED;
741 }
742
743 static nsresult NSAPI nsChannel_SetRequestMethod(nsIHttpChannel *iface,
744                                                  const nsACString *aRequestMethod)
745 {
746     nsChannel *This = NSCHANNEL_THIS(iface);
747
748     TRACE("(%p)->(%p)\n", This, aRequestMethod);
749
750     if(This->http_channel)
751         return nsIHttpChannel_SetRequestMethod(This->http_channel, aRequestMethod);
752
753     return NS_ERROR_NOT_IMPLEMENTED;
754 }
755
756 static nsresult NSAPI nsChannel_GetReferrer(nsIHttpChannel *iface, nsIURI **aReferrer)
757 {
758     nsChannel *This = NSCHANNEL_THIS(iface);
759
760     TRACE("(%p)->(%p)\n", This, aReferrer);
761
762     if(This->http_channel)
763         return nsIHttpChannel_GetReferrer(This->http_channel, aReferrer);
764
765     return NS_ERROR_NOT_IMPLEMENTED;
766 }
767
768 static nsresult NSAPI nsChannel_SetReferrer(nsIHttpChannel *iface, nsIURI *aReferrer)
769 {
770     nsChannel *This = NSCHANNEL_THIS(iface);
771
772     TRACE("(%p)->(%p)\n", This, aReferrer);
773
774     if(This->http_channel)
775         return nsIHttpChannel_SetReferrer(This->http_channel, aReferrer);
776
777     return NS_ERROR_NOT_IMPLEMENTED;
778 }
779
780 static nsresult NSAPI nsChannel_GetRequestHeader(nsIHttpChannel *iface,
781          const nsACString *aHeader, nsACString *_retval)
782 {
783     nsChannel *This = NSCHANNEL_THIS(iface);
784
785     TRACE("(%p)->(%p %p)\n", This, aHeader, _retval);
786
787     if(This->http_channel)
788         return nsIHttpChannel_GetRequestHeader(This->http_channel, aHeader, _retval);
789
790     return NS_ERROR_NOT_IMPLEMENTED;
791 }
792
793 static nsresult NSAPI nsChannel_SetRequestHeader(nsIHttpChannel *iface,
794          const nsACString *aHeader, const nsACString *aValue, PRBool aMerge)
795 {
796     nsChannel *This = NSCHANNEL_THIS(iface);
797
798     TRACE("(%p)->(%p %p %x)\n", This, aHeader, aValue, aMerge);
799
800     if(This->http_channel)
801         return nsIHttpChannel_SetRequestHeader(This->http_channel, aHeader, aValue, aMerge);
802
803     return NS_ERROR_NOT_IMPLEMENTED;
804 }
805
806 static nsresult NSAPI nsChannel_VisitRequestHeaders(nsIHttpChannel *iface,
807                                                     nsIHttpHeaderVisitor *aVisitor)
808 {
809     nsChannel *This = NSCHANNEL_THIS(iface);
810
811     TRACE("(%p)->(%p)\n", This, aVisitor);
812
813     if(This->http_channel)
814         return nsIHttpChannel_VisitRequestHeaders(This->http_channel, aVisitor);
815
816     return NS_ERROR_NOT_IMPLEMENTED;
817 }
818
819 static nsresult NSAPI nsChannel_GetAllowPipelining(nsIHttpChannel *iface, PRBool *aAllowPipelining)
820 {
821     nsChannel *This = NSCHANNEL_THIS(iface);
822
823     TRACE("(%p)->(%p)\n", This, aAllowPipelining);
824
825     if(This->http_channel)
826         return nsIHttpChannel_GetAllowPipelining(This->http_channel, aAllowPipelining);
827
828     return NS_ERROR_NOT_IMPLEMENTED;
829 }
830
831 static nsresult NSAPI nsChannel_SetAllowPipelining(nsIHttpChannel *iface, PRBool aAllowPipelining)
832 {
833     nsChannel *This = NSCHANNEL_THIS(iface);
834
835     TRACE("(%p)->(%x)\n", This, aAllowPipelining);
836
837     if(This->http_channel)
838         return nsIHttpChannel_SetAllowPipelining(This->http_channel, aAllowPipelining);
839
840     return NS_ERROR_NOT_IMPLEMENTED;
841 }
842
843 static nsresult NSAPI nsChannel_GetRedirectionLimit(nsIHttpChannel *iface, PRUint32 *aRedirectionLimit)
844 {
845     nsChannel *This = NSCHANNEL_THIS(iface);
846
847     TRACE("(%p)->(%p)\n", This, aRedirectionLimit);
848
849     if(This->http_channel)
850         return nsIHttpChannel_GetRedirectionLimit(This->http_channel, aRedirectionLimit);
851
852     return NS_ERROR_NOT_IMPLEMENTED;
853 }
854
855 static nsresult NSAPI nsChannel_SetRedirectionLimit(nsIHttpChannel *iface, PRUint32 aRedirectionLimit)
856 {
857     nsChannel *This = NSCHANNEL_THIS(iface);
858
859     TRACE("(%p)->(%u)\n", This, aRedirectionLimit);
860
861     if(This->http_channel)
862         return nsIHttpChannel_SetRedirectionLimit(This->http_channel, aRedirectionLimit);
863
864     return NS_ERROR_NOT_IMPLEMENTED;
865 }
866
867 static nsresult NSAPI nsChannel_GetResponseStatus(nsIHttpChannel *iface, PRUint32 *aResponseStatus)
868 {
869     nsChannel *This = NSCHANNEL_THIS(iface);
870
871     TRACE("(%p)->(%p)\n", This, aResponseStatus);
872
873     if(This->http_channel)
874         return nsIHttpChannel_GetResponseStatus(This->http_channel, aResponseStatus);
875
876     return NS_ERROR_NOT_IMPLEMENTED;
877 }
878
879 static nsresult NSAPI nsChannel_GetResponseStatusText(nsIHttpChannel *iface,
880                                                       nsACString *aResponseStatusText)
881 {
882     nsChannel *This = NSCHANNEL_THIS(iface);
883
884     TRACE("(%p)->(%p)\n", This, aResponseStatusText);
885
886     if(This->http_channel)
887         return nsIHttpChannel_GetResponseStatusText(This->http_channel, aResponseStatusText);
888
889     return NS_ERROR_NOT_IMPLEMENTED;
890 }
891
892 static nsresult NSAPI nsChannel_GetRequestSucceeded(nsIHttpChannel *iface,
893                                                     PRBool *aRequestSucceeded)
894 {
895     nsChannel *This = NSCHANNEL_THIS(iface);
896
897     TRACE("(%p)->(%p)\n", This, aRequestSucceeded);
898
899     if(This->http_channel)
900         return nsIHttpChannel_GetRequestSucceeded(This->http_channel, aRequestSucceeded);
901
902     return NS_ERROR_NOT_IMPLEMENTED;
903 }
904
905 static nsresult NSAPI nsChannel_GetResponseHeader(nsIHttpChannel *iface,
906          const nsACString *header, nsACString *_retval)
907 {
908     nsChannel *This = NSCHANNEL_THIS(iface);
909
910     TRACE("(%p)->(%p %p)\n", This, header, _retval);
911
912     if(This->http_channel)
913         return nsIHttpChannel_GetResponseHeader(This->http_channel, header, _retval);
914
915     return NS_ERROR_NOT_IMPLEMENTED;
916 }
917
918 static nsresult NSAPI nsChannel_SetResponseHeader(nsIHttpChannel *iface,
919         const nsACString *header, const nsACString *value, PRBool merge)
920 {
921     nsChannel *This = NSCHANNEL_THIS(iface);
922
923     TRACE("(%p)->(%p %p %x)\n", This, header, value, merge);
924
925     if(This->http_channel)
926         return nsIHttpChannel_SetResponseHeader(This->http_channel, header, value, merge);
927
928     return NS_ERROR_NOT_IMPLEMENTED;
929 }
930
931 static nsresult NSAPI nsChannel_VisitResponseHeaders(nsIHttpChannel *iface,
932         nsIHttpHeaderVisitor *aVisitor)
933 {
934     nsChannel *This = NSCHANNEL_THIS(iface);
935
936     TRACE("(%p)->(%p)\n", This, aVisitor);
937
938     if(This->http_channel)
939         return nsIHttpChannel_VisitResponseHeaders(This->http_channel, aVisitor);
940
941     return NS_ERROR_NOT_IMPLEMENTED;
942 }
943
944 static nsresult NSAPI nsChannel_IsNoStoreResponse(nsIHttpChannel *iface, PRBool *_retval)
945 {
946     nsChannel *This = NSCHANNEL_THIS(iface);
947
948     TRACE("(%p)->(%p)\n", This, _retval);
949
950     if(This->http_channel)
951         return nsIHttpChannel_IsNoStoreResponse(This->http_channel, _retval);
952
953     return NS_ERROR_NOT_IMPLEMENTED;
954 }
955
956 static nsresult NSAPI nsChannel_IsNoCacheResponse(nsIHttpChannel *iface, PRBool *_retval)
957 {
958     nsChannel *This = NSCHANNEL_THIS(iface);
959
960     TRACE("(%p)->(%p)\n", This, _retval);
961
962     if(This->http_channel)
963         return nsIHttpChannel_IsNoCacheResponse(This->http_channel, _retval);
964
965     return NS_ERROR_NOT_IMPLEMENTED;
966 }
967
968 #undef NSCHANNEL_THIS
969
970 static const nsIHttpChannelVtbl nsChannelVtbl = {
971     nsChannel_QueryInterface,
972     nsChannel_AddRef,
973     nsChannel_Release,
974     nsChannel_GetName,
975     nsChannel_IsPending,
976     nsChannel_GetStatus,
977     nsChannel_Cancel,
978     nsChannel_Suspend,
979     nsChannel_Resume,
980     nsChannel_GetLoadGroup,
981     nsChannel_SetLoadGroup,
982     nsChannel_GetLoadFlags,
983     nsChannel_SetLoadFlags,
984     nsChannel_GetOriginalURI,
985     nsChannel_SetOriginalURI,
986     nsChannel_GetURI,
987     nsChannel_GetOwner,
988     nsChannel_SetOwner,
989     nsChannel_GetNotificationCallbacks,
990     nsChannel_SetNotificationCallbacks,
991     nsChannel_GetSecurityInfo,
992     nsChannel_GetContentType,
993     nsChannel_SetContentType,
994     nsChannel_GetContentCharset,
995     nsChannel_SetContentCharset,
996     nsChannel_GetContentLength,
997     nsChannel_SetContentLength,
998     nsChannel_Open,
999     nsChannel_AsyncOpen,
1000     nsChannel_GetRequestMethod,
1001     nsChannel_SetRequestMethod,
1002     nsChannel_GetReferrer,
1003     nsChannel_SetReferrer,
1004     nsChannel_GetRequestHeader,
1005     nsChannel_SetRequestHeader,
1006     nsChannel_VisitRequestHeaders,
1007     nsChannel_GetAllowPipelining,
1008     nsChannel_SetAllowPipelining,
1009     nsChannel_GetRedirectionLimit,
1010     nsChannel_SetRedirectionLimit,
1011     nsChannel_GetResponseStatus,
1012     nsChannel_GetResponseStatusText,
1013     nsChannel_GetRequestSucceeded,
1014     nsChannel_GetResponseHeader,
1015     nsChannel_SetResponseHeader,
1016     nsChannel_VisitResponseHeaders,
1017     nsChannel_IsNoStoreResponse,
1018     nsChannel_IsNoCacheResponse
1019 };
1020
1021 #define NSUPCHANNEL_THIS(iface) DEFINE_THIS(nsChannel, UploadChannel, iface)
1022
1023 static nsresult NSAPI nsUploadChannel_QueryInterface(nsIUploadChannel *iface, nsIIDRef riid,
1024                                                      nsQIResult result)
1025 {
1026     nsChannel *This = NSUPCHANNEL_THIS(iface);
1027     return nsIChannel_QueryInterface(NSCHANNEL(This), riid, result);
1028 }
1029
1030 static nsrefcnt NSAPI nsUploadChannel_AddRef(nsIUploadChannel *iface)
1031 {
1032     nsChannel *This = NSUPCHANNEL_THIS(iface);
1033     return nsIChannel_AddRef(NSCHANNEL(This));
1034 }
1035
1036 static nsrefcnt NSAPI nsUploadChannel_Release(nsIUploadChannel *iface)
1037 {
1038     nsChannel *This = NSUPCHANNEL_THIS(iface);
1039     return nsIChannel_Release(NSCHANNEL(This));
1040 }
1041
1042 static nsresult NSAPI nsUploadChannel_SetUploadStream(nsIUploadChannel *iface,
1043         nsIInputStream *aStream, const nsACString *aContentType, PRInt32 aContentLength)
1044 {
1045     nsChannel *This = NSUPCHANNEL_THIS(iface);
1046     const char *content_type;
1047
1048     TRACE("(%p)->(%p %p %d)\n", This, aStream, aContentType, aContentLength);
1049
1050     if(This->post_data_stream)
1051         nsIInputStream_Release(This->post_data_stream);
1052
1053     if(aContentType) {
1054         nsACString_GetData(aContentType, &content_type, NULL);
1055         if(*content_type)
1056             FIXME("Unsupported aContentType argument: %s\n", debugstr_a(content_type));
1057     }
1058
1059     if(aContentLength != -1)
1060         FIXME("Unsupported acontentLength = %d\n", aContentLength);
1061
1062     if(aStream)
1063         nsIInputStream_AddRef(aStream);
1064     This->post_data_stream = aStream;
1065
1066     return NS_OK;
1067 }
1068
1069 static nsresult NSAPI nsUploadChannel_GetUploadStream(nsIUploadChannel *iface,
1070         nsIInputStream **aUploadStream)
1071 {
1072     nsChannel *This = NSUPCHANNEL_THIS(iface);
1073
1074     TRACE("(%p)->(%p)\n", This, aUploadStream);
1075
1076     if(This->post_data_stream)
1077         nsIInputStream_AddRef(This->post_data_stream);
1078
1079     *aUploadStream = This->post_data_stream;
1080     return NS_OK;
1081 }
1082
1083 #undef NSUPCHANNEL_THIS
1084
1085 static const nsIUploadChannelVtbl nsUploadChannelVtbl = {
1086     nsUploadChannel_QueryInterface,
1087     nsUploadChannel_AddRef,
1088     nsUploadChannel_Release,
1089     nsUploadChannel_SetUploadStream,
1090     nsUploadChannel_GetUploadStream
1091 };
1092
1093 #define NSURI_THIS(iface) DEFINE_THIS(nsURI, WineURI, iface)
1094
1095 static nsresult NSAPI nsURI_QueryInterface(nsIWineURI *iface, nsIIDRef riid, nsQIResult result)
1096 {
1097     nsURI *This = NSURI_THIS(iface);
1098
1099     *result = NULL;
1100
1101     if(IsEqualGUID(&IID_nsISupports, riid)) {
1102         TRACE("(%p)->(IID_nsISupports %p)\n", This, result);
1103         *result = NSURI(This);
1104     }else if(IsEqualGUID(&IID_nsIURI, riid)) {
1105         TRACE("(%p)->(IID_nsIURI %p)\n", This, result);
1106         *result = NSURI(This);
1107     }else if(IsEqualGUID(&IID_nsIWineURI, riid)) {
1108         TRACE("(%p)->(IID_nsIWineURI %p)\n", This, result);
1109         *result = NSURI(This);
1110     }
1111
1112     if(*result) {
1113         nsIURI_AddRef(NSURI(This));
1114         return NS_OK;
1115     }
1116
1117     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), result);
1118     return This->uri ? nsIURI_QueryInterface(This->uri, riid, result) : NS_NOINTERFACE;
1119 }
1120
1121 static nsrefcnt NSAPI nsURI_AddRef(nsIWineURI *iface)
1122 {
1123     nsURI *This = NSURI_THIS(iface);
1124     LONG ref = InterlockedIncrement(&This->ref);
1125
1126     TRACE("(%p) ref=%d\n", This, ref);
1127
1128     return ref;
1129 }
1130
1131 static nsrefcnt NSAPI nsURI_Release(nsIWineURI *iface)
1132 {
1133     nsURI *This = NSURI_THIS(iface);
1134     LONG ref = InterlockedDecrement(&This->ref);
1135
1136     TRACE("(%p) ref=%d\n", This, ref);
1137
1138     if(!ref) {
1139         if(This->container)
1140             nsIWebBrowserChrome_Release(NSWBCHROME(This->container));
1141         if(This->uri)
1142             nsIURI_Release(This->uri);
1143         if(This->mon)
1144             IMoniker_Release(This->mon);
1145         mshtml_free(This->spec);
1146         mshtml_free(This);
1147     }
1148
1149     return ref;
1150 }
1151
1152 static nsresult NSAPI nsURI_GetSpec(nsIWineURI *iface, nsACString *aSpec)
1153 {
1154     nsURI *This = NSURI_THIS(iface);
1155
1156     TRACE("(%p)->(%p)\n", This, aSpec);
1157
1158     if(This->uri)
1159         return nsIURI_GetSpec(This->uri, aSpec);
1160
1161     if(This->spec) {
1162         nsACString_SetData(aSpec, This->spec);
1163         return NS_OK;
1164     }
1165
1166     WARN("mon and uri are NULL\n");
1167     return NS_ERROR_NOT_IMPLEMENTED;
1168
1169 }
1170
1171 static nsresult NSAPI nsURI_SetSpec(nsIWineURI *iface, const nsACString *aSpec)
1172 {
1173     nsURI *This = NSURI_THIS(iface);
1174
1175     TRACE("(%p)->(%p)\n", This, aSpec);
1176
1177     if(This->uri)
1178         return nsIURI_SetSpec(This->uri, aSpec);
1179
1180     FIXME("default action not implemented\n");
1181     return NS_ERROR_NOT_IMPLEMENTED;
1182 }
1183
1184 static nsresult NSAPI nsURI_GetPrePath(nsIWineURI *iface, nsACString *aPrePath)
1185 {
1186     nsURI *This = NSURI_THIS(iface);
1187
1188     TRACE("(%p)->(%p)\n", This, aPrePath);
1189
1190     if(This->uri)
1191         return nsIURI_GetPrePath(This->uri, aPrePath);
1192
1193     FIXME("default action not implemented\n");
1194     return NS_ERROR_NOT_IMPLEMENTED;
1195 }
1196
1197 static nsresult NSAPI nsURI_GetScheme(nsIWineURI *iface, nsACString *aScheme)
1198 {
1199     nsURI *This = NSURI_THIS(iface);
1200
1201     TRACE("(%p)->(%p)\n", This, aScheme);
1202
1203     if(This->uri)
1204         return nsIURI_GetScheme(This->uri, aScheme);
1205
1206     FIXME("default action not implemented\n");
1207     return NS_ERROR_NOT_IMPLEMENTED;
1208 }
1209
1210 static nsresult NSAPI nsURI_SetScheme(nsIWineURI *iface, const nsACString *aScheme)
1211 {
1212     nsURI *This = NSURI_THIS(iface);
1213
1214     TRACE("(%p)->(%p)\n", This, aScheme);
1215
1216     if(This->uri)
1217         return nsIURI_SetScheme(This->uri, aScheme);
1218
1219     FIXME("default action not implemented\n");
1220     return NS_ERROR_NOT_IMPLEMENTED;
1221 }
1222
1223 static nsresult NSAPI nsURI_GetUserPass(nsIWineURI *iface, nsACString *aUserPass)
1224 {
1225     nsURI *This = NSURI_THIS(iface);
1226
1227     TRACE("(%p)->(%p)\n", This, aUserPass);
1228
1229     if(This->uri)
1230         return nsIURI_GetUserPass(This->uri, aUserPass);
1231
1232     FIXME("default action not implemented\n");
1233     return NS_ERROR_NOT_IMPLEMENTED;
1234 }
1235
1236 static nsresult NSAPI nsURI_SetUserPass(nsIWineURI *iface, const nsACString *aUserPass)
1237 {
1238     nsURI *This = NSURI_THIS(iface);
1239
1240     TRACE("(%p)->(%p)\n", This, aUserPass);
1241
1242     if(This->uri)
1243         return nsIURI_SetUserPass(This->uri, aUserPass);
1244
1245     FIXME("default action not implemented\n");
1246     return NS_ERROR_NOT_IMPLEMENTED;
1247 }
1248
1249 static nsresult NSAPI nsURI_GetUsername(nsIWineURI *iface, nsACString *aUsername)
1250 {
1251     nsURI *This = NSURI_THIS(iface);
1252
1253     TRACE("(%p)->(%p)\n", This, aUsername);
1254
1255     if(This->uri)
1256         return nsIURI_GetUsername(This->uri, aUsername);
1257
1258     FIXME("default action not implemented\n");
1259     return NS_ERROR_NOT_IMPLEMENTED;
1260 }
1261
1262 static nsresult NSAPI nsURI_SetUsername(nsIWineURI *iface, const nsACString *aUsername)
1263 {
1264     nsURI *This = NSURI_THIS(iface);
1265
1266     TRACE("(%p)->(%p)\n", This, aUsername);
1267
1268     if(This->uri)
1269         return nsIURI_SetUsername(This->uri, aUsername);
1270
1271     FIXME("default action not implemented\n");
1272     return NS_ERROR_NOT_IMPLEMENTED;
1273 }
1274
1275 static nsresult NSAPI nsURI_GetPassword(nsIWineURI *iface, nsACString *aPassword)
1276 {
1277     nsURI *This = NSURI_THIS(iface);
1278
1279     TRACE("(%p)->(%p)\n", This, aPassword);
1280
1281     if(This->uri)
1282         return nsIURI_GetPassword(This->uri, aPassword);
1283
1284     FIXME("default action not implemented\n");
1285     return NS_ERROR_NOT_IMPLEMENTED;
1286 }
1287
1288 static nsresult NSAPI nsURI_SetPassword(nsIWineURI *iface, const nsACString *aPassword)
1289 {
1290     nsURI *This = NSURI_THIS(iface);
1291
1292     TRACE("(%p)->(%p)\n", This, aPassword);
1293
1294     if(This->uri)
1295         return nsIURI_SetPassword(This->uri, aPassword);
1296
1297     FIXME("default action not implemented\n");
1298     return NS_ERROR_NOT_IMPLEMENTED;
1299 }
1300
1301 static nsresult NSAPI nsURI_GetHostPort(nsIWineURI *iface, nsACString *aHostPort)
1302 {
1303     nsURI *This = NSURI_THIS(iface);
1304
1305     TRACE("(%p)->(%p)\n", This, aHostPort);
1306
1307     if(This->uri)
1308         return nsIURI_GetHostPort(This->uri, aHostPort);
1309
1310     FIXME("default action not implemented\n");
1311     return NS_ERROR_NOT_IMPLEMENTED;
1312 }
1313
1314 static nsresult NSAPI nsURI_SetHostPort(nsIWineURI *iface, const nsACString *aHostPort)
1315 {
1316     nsURI *This = NSURI_THIS(iface);
1317
1318     TRACE("(%p)->(%p)\n", This, aHostPort);
1319
1320     if(This->uri)
1321         return nsIURI_SetHostPort(This->uri, aHostPort);
1322
1323     FIXME("default action not implemented\n");
1324     return NS_ERROR_NOT_IMPLEMENTED;
1325 }
1326
1327 static nsresult NSAPI nsURI_GetHost(nsIWineURI *iface, nsACString *aHost)
1328 {
1329     nsURI *This = NSURI_THIS(iface);
1330
1331     TRACE("(%p)->(%p)\n", This, aHost);
1332
1333     if(This->uri)
1334         return nsIURI_GetHost(This->uri, aHost);
1335
1336     FIXME("default action not implemented\n");
1337     return NS_ERROR_NOT_IMPLEMENTED;
1338 }
1339
1340 static nsresult NSAPI nsURI_SetHost(nsIWineURI *iface, const nsACString *aHost)
1341 {
1342     nsURI *This = NSURI_THIS(iface);
1343
1344     TRACE("(%p)->(%p)\n", This, aHost);
1345
1346     if(This->uri)
1347         return nsIURI_SetHost(This->uri, aHost);
1348
1349     FIXME("default action not implemented\n");
1350     return NS_ERROR_NOT_IMPLEMENTED;
1351 }
1352
1353 static nsresult NSAPI nsURI_GetPort(nsIWineURI *iface, PRInt32 *aPort)
1354 {
1355     nsURI *This = NSURI_THIS(iface);
1356
1357     TRACE("(%p)->(%p)\n", This, aPort);
1358
1359     if(This->uri)
1360         return nsIURI_GetPort(This->uri, aPort);
1361
1362     FIXME("default action not implemented\n");
1363     return NS_ERROR_NOT_IMPLEMENTED;
1364 }
1365
1366 static nsresult NSAPI nsURI_SetPort(nsIWineURI *iface, PRInt32 aPort)
1367 {
1368     nsURI *This = NSURI_THIS(iface);
1369
1370     TRACE("(%p)->(%d)\n", This, aPort);
1371
1372     if(This->uri)
1373         return nsIURI_SetPort(This->uri, aPort);
1374
1375     FIXME("default action not implemented\n");
1376     return NS_ERROR_NOT_IMPLEMENTED;
1377 }
1378
1379 static nsresult NSAPI nsURI_GetPath(nsIWineURI *iface, nsACString *aPath)
1380 {
1381     nsURI *This = NSURI_THIS(iface);
1382
1383     TRACE("(%p)->(%p)\n", This, aPath);
1384
1385     if(This->uri)
1386         return nsIURI_GetPath(This->uri, aPath);
1387
1388     FIXME("default action not implemented\n");
1389     return NS_ERROR_NOT_IMPLEMENTED;
1390 }
1391
1392 static nsresult NSAPI nsURI_SetPath(nsIWineURI *iface, const nsACString *aPath)
1393 {
1394     nsURI *This = NSURI_THIS(iface);
1395
1396     TRACE("(%p)->(%p)\n", This, aPath);
1397
1398     if(This->uri)
1399         return nsIURI_SetPath(This->uri, aPath);
1400
1401     FIXME("default action not implemented\n");
1402     return NS_ERROR_NOT_IMPLEMENTED;
1403 }
1404
1405 static nsresult NSAPI nsURI_Equals(nsIWineURI *iface, nsIURI *other, PRBool *_retval)
1406 {
1407     nsURI *This = NSURI_THIS(iface);
1408
1409     TRACE("(%p)->(%p %p)\n", This, other, _retval);
1410
1411     if(This->uri)
1412         return nsIURI_Equals(This->uri, other, _retval);
1413
1414     FIXME("default action not implemented\n");
1415     return NS_ERROR_NOT_IMPLEMENTED;
1416 }
1417
1418 static nsresult NSAPI nsURI_SchemeIs(nsIWineURI *iface, const char *scheme, PRBool *_retval)
1419 {
1420     nsURI *This = NSURI_THIS(iface);
1421
1422     TRACE("(%p)->(%s %p)\n", This, debugstr_a(scheme), _retval);
1423
1424     if(This->uri)
1425         return nsIURI_SchemeIs(This->uri, scheme, _retval);
1426
1427     FIXME("default action not implemented\n");
1428     return NS_ERROR_NOT_IMPLEMENTED;
1429 }
1430
1431 static nsresult NSAPI nsURI_Clone(nsIWineURI *iface, nsIURI **_retval)
1432 {
1433     nsURI *This = NSURI_THIS(iface);
1434
1435     TRACE("(%p)->(%p)\n", This, _retval);
1436
1437     if(This->uri) {
1438         nsIURI *uri;
1439         nsresult nsres;
1440
1441         nsres = nsIURI_Clone(This->uri, &uri);
1442         if(NS_FAILED(nsres)) {
1443             WARN("Clone failed: %08x\n", nsres);
1444             return nsres;
1445         }
1446
1447         return create_uri(uri, This->container, _retval);
1448     }
1449
1450     FIXME("default action not implemented\n");
1451     return NS_ERROR_NOT_IMPLEMENTED;
1452 }
1453
1454 static nsresult NSAPI nsURI_Resolve(nsIWineURI *iface, const nsACString *arelativePath,
1455         nsACString *_retval)
1456 {
1457     nsURI *This = NSURI_THIS(iface);
1458
1459     TRACE("(%p)->(%p %p)\n", This, arelativePath, _retval);
1460
1461     if(This->uri)
1462         return nsIURI_Resolve(This->uri, arelativePath, _retval);
1463
1464     FIXME("default action not implemented\n");
1465     return NS_ERROR_NOT_IMPLEMENTED;
1466 }
1467
1468 static nsresult NSAPI nsURI_GetAsciiSpec(nsIWineURI *iface, nsACString *aAsciiSpec)
1469 {
1470     nsURI *This = NSURI_THIS(iface);
1471
1472     TRACE("(%p)->(%p)\n", This, aAsciiSpec);
1473
1474     if(This->uri)
1475         return nsIURI_GetAsciiSpec(This->uri, aAsciiSpec);
1476
1477     FIXME("default action not implemented\n");
1478     return NS_ERROR_NOT_IMPLEMENTED;
1479 }
1480
1481 static nsresult NSAPI nsURI_GetAsciiHost(nsIWineURI *iface, nsACString *aAsciiHost)
1482 {
1483     nsURI *This = NSURI_THIS(iface);
1484
1485     TRACE("(%p)->(%p)\n", This, aAsciiHost);
1486
1487     if(This->uri)
1488         return nsIURI_GetAsciiHost(This->uri, aAsciiHost);
1489
1490     FIXME("default action not implemented\n");
1491     return NS_ERROR_NOT_IMPLEMENTED;
1492 }
1493
1494 static nsresult NSAPI nsURI_GetOriginCharset(nsIWineURI *iface, nsACString *aOriginCharset)
1495 {
1496     nsURI *This = NSURI_THIS(iface);
1497
1498     TRACE("(%p)->(%p)\n", This, aOriginCharset);
1499
1500     if(This->uri)
1501         return nsIURI_GetOriginCharset(This->uri, aOriginCharset);
1502
1503     FIXME("default action not implemented\n");
1504     return NS_ERROR_NOT_IMPLEMENTED;
1505 }
1506
1507 static nsresult NSAPI nsURI_GetNSContainer(nsIWineURI *iface, NSContainer **aContainer)
1508 {
1509     nsURI *This = NSURI_THIS(iface);
1510
1511     TRACE("(%p)->(%p)\n", This, aContainer);
1512
1513     if(This->container)
1514         nsIWebBrowserChrome_AddRef(NSWBCHROME(This->container));
1515     *aContainer = This->container;
1516
1517     return NS_OK;
1518 }
1519
1520 static nsresult NSAPI nsURI_SetNSContainer(nsIWineURI *iface, NSContainer *aContainer)
1521 {
1522     nsURI *This = NSURI_THIS(iface);
1523
1524     TRACE("(%p)->(%p)\n", This, aContainer);
1525
1526     if(This->container) {
1527         WARN("Container already set: %p\n", This->container);
1528         nsIWebBrowserChrome_Release(NSWBCHROME(This->container));
1529     }
1530
1531     if(aContainer)
1532         nsIWebBrowserChrome_AddRef(NSWBCHROME(aContainer));
1533     This->container = aContainer;
1534
1535     return NS_OK;
1536 }
1537
1538 static nsresult NSAPI nsURI_GetMoniker(nsIWineURI *iface, IMoniker **aMoniker)
1539 {
1540     nsURI *This = NSURI_THIS(iface);
1541
1542     TRACE("(%p)->(%p)\n", This, aMoniker);
1543
1544     if(This->mon)
1545         IMoniker_AddRef(This->mon);
1546     *aMoniker = This->mon;
1547
1548     return NS_OK;
1549 }
1550
1551 static nsresult NSAPI nsURI_SetMoniker(nsIWineURI *iface, IMoniker *aMoniker)
1552 {
1553     nsURI *This = NSURI_THIS(iface);
1554
1555     TRACE("(%p)->(%p)\n", This, aMoniker);
1556
1557     if(This->mon) {
1558         WARN("Moniker already set: %p\n", This->container);
1559         IMoniker_Release(This->mon);
1560
1561         mshtml_free(This->spec);
1562         This->spec = NULL;
1563     }
1564
1565     if(aMoniker) {
1566         LPWSTR url = NULL;
1567         HRESULT hres;
1568
1569         hres = IMoniker_GetDisplayName(aMoniker, NULL, NULL, &url);
1570         if(SUCCEEDED(hres)) {
1571             DWORD len;
1572
1573             len = WideCharToMultiByte(CP_ACP, 0, url, -1, NULL, 0, NULL, NULL);
1574             This->spec = mshtml_alloc(len*sizeof(WCHAR));
1575             WideCharToMultiByte(CP_ACP, 0, url, -1, This->spec, -1, NULL, NULL);
1576             CoTaskMemFree(url);
1577
1578             TRACE("spec %s\n", debugstr_a(This->spec));
1579         }else {
1580             ERR("GetDisplayName failed: %08x\n", hres);
1581         }
1582
1583         IMoniker_AddRef(aMoniker);
1584     }
1585     This->mon = aMoniker;
1586
1587     return NS_OK;
1588 }
1589
1590 #undef NSURI_THIS
1591
1592 static const nsIWineURIVtbl nsWineURIVtbl = {
1593     nsURI_QueryInterface,
1594     nsURI_AddRef,
1595     nsURI_Release,
1596     nsURI_GetSpec,
1597     nsURI_SetSpec,
1598     nsURI_GetPrePath,
1599     nsURI_GetScheme,
1600     nsURI_SetScheme,
1601     nsURI_GetUserPass,
1602     nsURI_SetUserPass,
1603     nsURI_GetUsername,
1604     nsURI_SetUsername,
1605     nsURI_GetPassword,
1606     nsURI_SetPassword,
1607     nsURI_GetHostPort,
1608     nsURI_SetHostPort,
1609     nsURI_GetHost,
1610     nsURI_SetHost,
1611     nsURI_GetPort,
1612     nsURI_SetPort,
1613     nsURI_GetPath,
1614     nsURI_SetPath,
1615     nsURI_Equals,
1616     nsURI_SchemeIs,
1617     nsURI_Clone,
1618     nsURI_Resolve,
1619     nsURI_GetAsciiSpec,
1620     nsURI_GetAsciiHost,
1621     nsURI_GetOriginCharset,
1622     nsURI_GetNSContainer,
1623     nsURI_SetNSContainer,
1624     nsURI_GetMoniker,
1625     nsURI_SetMoniker
1626 };
1627
1628 static nsresult create_uri(nsIURI *uri, NSContainer *container, nsIURI **_retval)
1629 {
1630     nsURI *ret = mshtml_alloc(sizeof(nsURI));
1631
1632     ret->lpWineURIVtbl = &nsWineURIVtbl;
1633     ret->ref = 1;
1634     ret->uri = uri;
1635     ret->container = container;
1636     ret->mon = NULL;
1637     ret->spec = NULL;
1638
1639     if(container)
1640         nsIWebBrowserChrome_AddRef(NSWBCHROME(container));
1641
1642     TRACE("retval=%p\n", ret);
1643     *_retval = NSURI(ret);
1644     return NS_OK;
1645 }
1646
1647 static nsresult NSAPI nsIOService_QueryInterface(nsIIOService *iface, nsIIDRef riid,
1648                                                  nsQIResult result)
1649 {
1650     *result = NULL;
1651
1652     if(IsEqualGUID(&IID_nsISupports, riid)) {
1653         TRACE("(IID_nsISupports %p)\n", result);
1654         *result = iface;
1655     }else if(IsEqualGUID(&IID_nsIIOService, riid)) {
1656         TRACE("(IID_nsIIOService %p)\n", result);
1657         *result = iface;
1658     }
1659
1660     if(*result) {
1661         nsIIOService_AddRef(iface);
1662         return S_OK;
1663     }
1664
1665     WARN("(%s %p)\n", debugstr_guid(riid), result);
1666     return NS_NOINTERFACE;
1667 }
1668
1669 static nsrefcnt NSAPI nsIOService_AddRef(nsIIOService *iface)
1670 {
1671     return 2;
1672 }
1673
1674 static nsrefcnt NSAPI nsIOService_Release(nsIIOService *iface)
1675 {
1676     return 1;
1677 }
1678
1679 static nsresult NSAPI nsIOService_GetProtocolHandler(nsIIOService *iface, const char *aScheme,
1680                                                      nsIProtocolHandler **_retval)
1681 {
1682     TRACE("(%s %p)\n", debugstr_a(aScheme), _retval);
1683     return nsIIOService_GetProtocolHandler(nsio, aScheme, _retval);
1684 }
1685
1686 static nsresult NSAPI nsIOService_GetProtocolFlags(nsIIOService *iface, const char *aScheme,
1687                                                     PRUint32 *_retval)
1688 {
1689     TRACE("(%s %p)\n", debugstr_a(aScheme), _retval);
1690     return nsIIOService_GetProtocolFlags(nsio, aScheme, _retval);
1691 }
1692
1693 static nsresult NSAPI nsIOService_NewURI(nsIIOService *iface, const nsACString *aSpec,
1694         const char *aOriginCharset, nsIURI *aBaseURI, nsIURI **_retval)
1695 {
1696     const char *spec = NULL;
1697     NSContainer *nscontainer = NULL;
1698     nsIURI *uri = NULL;
1699     PRBool is_javascript = FALSE;
1700     IMoniker *base_mon = NULL;
1701     nsresult nsres;
1702
1703     nsACString_GetData(aSpec, &spec, NULL);
1704
1705     TRACE("(%p(%s) %s %p %p)\n", aSpec, debugstr_a(spec), debugstr_a(aOriginCharset),
1706           aBaseURI, _retval);
1707
1708     if(aBaseURI) {
1709         nsACString base_uri_str;
1710         const char *base_uri = NULL;
1711
1712         nsACString_Init(&base_uri_str, NULL);
1713
1714         nsres = nsIURI_GetSpec(aBaseURI, &base_uri_str);
1715         if(NS_SUCCEEDED(nsres)) {
1716             nsACString_GetData(&base_uri_str, &base_uri, NULL);
1717             TRACE("uri=%s\n", debugstr_a(base_uri));
1718         }else {
1719             ERR("GetSpec failed: %08x\n", nsres);
1720         }
1721
1722         nsACString_Finish(&base_uri_str);
1723     }
1724
1725     nsres = nsIIOService_NewURI(nsio, aSpec, aOriginCharset, aBaseURI, &uri);
1726     if(NS_FAILED(nsres))
1727         TRACE("NewURI failed: %08x\n", nsres);
1728
1729     if(uri) {
1730         nsIURI_SchemeIs(uri, "javascript", &is_javascript);
1731         if(is_javascript) {
1732             TRACE("returning javascript uri: %p\n", uri);
1733             *_retval = uri;
1734             return NS_OK;
1735         }
1736     }
1737
1738     if(aBaseURI) {
1739         nsIWineURI *wine_uri;
1740
1741         nsres = nsIURI_QueryInterface(aBaseURI, &IID_nsIWineURI, (void**)&wine_uri);
1742         if(NS_SUCCEEDED(nsres)) {
1743             nsIWineURI_GetNSContainer(wine_uri, &nscontainer);
1744             nsIWineURI_GetMoniker(wine_uri, &base_mon);
1745             nsIWineURI_Release(wine_uri);
1746         }else {
1747             ERR("Could not get nsIWineURI: %08x\n", nsres);
1748         }
1749     }
1750
1751     nsres = create_uri(uri, nscontainer, _retval);
1752
1753     if(nscontainer)
1754         nsIWebBrowserChrome_Release(NSWBCHROME(nscontainer));
1755
1756     if(base_mon) {
1757         LPWSTR url;
1758         IMoniker *mon;
1759         DWORD len;
1760         HRESULT hres;
1761
1762         len = MultiByteToWideChar(CP_ACP, 0, spec, -1, NULL, 0);
1763         url = HeapAlloc(GetProcessHeap(), 0, len*sizeof(WCHAR));
1764         MultiByteToWideChar(CP_ACP, 0, spec, -1, url, -1);
1765
1766         hres = CreateURLMoniker(base_mon, url, &mon);
1767         HeapFree(GetProcessHeap(), 0, url);
1768         if(SUCCEEDED(hres)) {
1769             nsIWineURI_SetMoniker((nsIWineURI*)*_retval, mon);
1770             IMoniker_Release(mon);
1771         }else {
1772             WARN("CreateURLMoniker failed: %08x\n", hres);
1773         }
1774     }
1775
1776     return nsres;
1777 }
1778
1779 static nsresult NSAPI nsIOService_NewFileURI(nsIIOService *iface, nsIFile *aFile,
1780                                              nsIURI **_retval)
1781 {
1782     TRACE("(%p %p)\n", aFile, _retval);
1783     return nsIIOService_NewFileURI(nsio, aFile, _retval);
1784 }
1785
1786 static nsresult NSAPI nsIOService_NewChannelFromURI(nsIIOService *iface, nsIURI *aURI,
1787                                                      nsIChannel **_retval)
1788 {
1789     nsIChannel *channel = NULL;
1790     nsChannel *ret;
1791     nsIWineURI *wine_uri;
1792     nsresult nsres;
1793
1794     TRACE("(%p %p)\n", aURI, _retval);
1795
1796     nsres = nsIIOService_NewChannelFromURI(nsio, aURI, &channel);
1797     if(NS_FAILED(nsres) && nsres != NS_ERROR_UNKNOWN_PROTOCOL) {
1798         WARN("NewChannelFromURI failed: %08x\n", nsres);
1799         *_retval = channel;
1800         return nsres;
1801     }
1802
1803     nsres = nsIURI_QueryInterface(aURI, &IID_nsIWineURI, (void**)&wine_uri);
1804     if(NS_FAILED(nsres)) {
1805         WARN("Could not get nsIWineURI: %08x\n", nsres);
1806         *_retval = channel;
1807         return channel ? NS_OK : NS_ERROR_UNEXPECTED;
1808     }
1809
1810     ret = mshtml_alloc(sizeof(nsChannel));
1811
1812     ret->lpHttpChannelVtbl = &nsChannelVtbl;
1813     ret->lpUploadChannelVtbl = &nsUploadChannelVtbl;
1814     ret->ref = 1;
1815     ret->channel = channel;
1816     ret->http_channel = NULL;
1817     ret->uri = wine_uri;
1818     ret->post_data_stream = NULL;
1819     ret->load_group = NULL;
1820     ret->notif_callback = NULL;
1821     ret->load_flags = 0;
1822     ret->content = NULL;
1823
1824     nsIURI_AddRef(aURI);
1825     ret->original_uri = aURI;
1826
1827     if(channel)
1828         nsIChannel_QueryInterface(channel, &IID_nsIHttpChannel, (void**)&ret->http_channel);
1829
1830     *_retval = NSCHANNEL(ret);
1831     return NS_OK;
1832 }
1833
1834 static nsresult NSAPI nsIOService_NewChannel(nsIIOService *iface, const nsACString *aSpec,
1835         const char *aOriginCharset, nsIURI *aBaseURI, nsIChannel **_retval)
1836 {
1837     TRACE("(%p %s %p %p)\n", aSpec, debugstr_a(aOriginCharset), aBaseURI, _retval);
1838     return nsIIOService_NewChannel(nsio, aSpec, aOriginCharset, aBaseURI, _retval);
1839 }
1840
1841 static nsresult NSAPI nsIOService_GetOffline(nsIIOService *iface, PRBool *aOffline)
1842 {
1843     TRACE("(%p)\n", aOffline);
1844     return nsIIOService_GetOffline(nsio, aOffline);
1845 }
1846
1847 static nsresult NSAPI nsIOService_SetOffline(nsIIOService *iface, PRBool aOffline)
1848 {
1849     TRACE("(%x)\n", aOffline);
1850     return nsIIOService_SetOffline(nsio, aOffline);
1851 }
1852
1853 static nsresult NSAPI nsIOService_AllowPort(nsIIOService *iface, PRInt32 aPort,
1854                                              const char *aScheme, PRBool *_retval)
1855 {
1856     TRACE("(%d %s %p)\n", aPort, debugstr_a(aScheme), _retval);
1857     return nsIIOService_AllowPort(nsio, aPort, debugstr_a(aScheme), _retval);
1858 }
1859
1860 static nsresult NSAPI nsIOService_ExtractScheme(nsIIOService *iface, const nsACString *urlString,
1861                                                  nsACString * _retval)
1862 {
1863     TRACE("(%p %p)\n", urlString, _retval);
1864     return nsIIOService_ExtractScheme(nsio, urlString, _retval);
1865 }
1866
1867 static const nsIIOServiceVtbl nsIOServiceVtbl = {
1868     nsIOService_QueryInterface,
1869     nsIOService_AddRef,
1870     nsIOService_Release,
1871     nsIOService_GetProtocolHandler,
1872     nsIOService_GetProtocolFlags,
1873     nsIOService_NewURI,
1874     nsIOService_NewFileURI,
1875     nsIOService_NewChannelFromURI,
1876     nsIOService_NewChannel,
1877     nsIOService_GetOffline,
1878     nsIOService_SetOffline,
1879     nsIOService_AllowPort,
1880     nsIOService_ExtractScheme
1881 };
1882
1883 static nsIIOService nsIOService = { &nsIOServiceVtbl };
1884
1885 static nsresult NSAPI nsIOServiceFactory_QueryInterface(nsIFactory *iface, nsIIDRef riid,
1886                                                         nsQIResult result)
1887 {
1888     *result = NULL;
1889
1890     if(IsEqualGUID(&IID_nsISupports, riid)) {
1891         TRACE("(IID_nsISupoprts %p)\n", result);
1892         *result = iface;
1893     }else if(IsEqualGUID(&IID_nsIFactory, riid)) {
1894         TRACE("(IID_nsIFactory %p)\n", result);
1895         *result = iface;
1896     }
1897
1898     if(*result) {
1899         nsIFactory_AddRef(iface);
1900         return NS_OK;
1901     }
1902
1903     WARN("(%s %p)\n", debugstr_guid(riid), result);
1904     return NS_NOINTERFACE;
1905 }
1906
1907 static nsrefcnt NSAPI nsIOServiceFactory_AddRef(nsIFactory *iface)
1908 {
1909     return 2;
1910 }
1911
1912 static nsrefcnt NSAPI nsIOServiceFactory_Release(nsIFactory *iface)
1913 {
1914     return 1;
1915 }
1916
1917 static nsresult NSAPI nsIOServiceFactory_CreateInstance(nsIFactory *iface,
1918         nsISupports *aOuter, const nsIID *iid, void **result)
1919 {
1920     return nsIIOService_QueryInterface(&nsIOService, iid, result);
1921 }
1922
1923 static nsresult NSAPI nsIOServiceFactory_LockFactory(nsIFactory *iface, PRBool lock)
1924 {
1925     WARN("(%x)\n", lock);
1926     return NS_OK;
1927 }
1928
1929 static const nsIFactoryVtbl nsIOServiceFactoryVtbl = {
1930     nsIOServiceFactory_QueryInterface,
1931     nsIOServiceFactory_AddRef,
1932     nsIOServiceFactory_Release,
1933     nsIOServiceFactory_CreateInstance,
1934     nsIOServiceFactory_LockFactory
1935 };
1936
1937 static nsIFactory nsIOServiceFactory = { &nsIOServiceFactoryVtbl };
1938
1939 void init_nsio(nsIComponentManager *component_manager, nsIComponentRegistrar *registrar)
1940 {
1941     nsIFactory *old_factory = NULL;
1942     nsresult nsres;
1943
1944     nsres = nsIComponentManager_GetClassObject(component_manager, &NS_IOSERVICE_CID,
1945                                                &IID_nsIFactory, (void**)&old_factory);
1946     if(NS_FAILED(nsres)) {
1947         ERR("Could not get factory: %08x\n", nsres);
1948         nsIFactory_Release(old_factory);
1949         return;
1950     }
1951
1952     nsres = nsIFactory_CreateInstance(old_factory, NULL, &IID_nsIIOService, (void**)&nsio);
1953     if(NS_FAILED(nsres)) {
1954         ERR("Couldn not create nsIOService instance %08x\n", nsres);
1955         nsIFactory_Release(old_factory);
1956         return;
1957     }
1958
1959     nsres = nsIComponentRegistrar_UnregisterFactory(registrar, &NS_IOSERVICE_CID, old_factory);
1960     nsIFactory_Release(old_factory);
1961     if(NS_FAILED(nsres))
1962         ERR("UnregisterFactory failed: %08x\n", nsres);
1963
1964     nsres = nsIComponentRegistrar_RegisterFactory(registrar, &NS_IOSERVICE_CID,
1965             NS_IOSERVICE_CLASSNAME, NS_IOSERVICE_CONTRACTID, &nsIOServiceFactory);
1966     if(NS_FAILED(nsres))
1967         ERR("RegisterFactory failed: %08x\n", nsres);
1968 }