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