urlmon: Added PI_MIMEVERIFICATION support.

This commit is contained in:
Jacek Caban 2009-05-14 00:49:01 +02:00 committed by Alexandre Julliard
parent daab4e2721
commit 299ce6c18e
1 changed files with 132 additions and 45 deletions

View File

@ -50,6 +50,7 @@ struct BindProtocol {
LONG priority; LONG priority;
BOOL reported_result; BOOL reported_result;
BOOL reported_mime;
BOOL from_urlmon; BOOL from_urlmon;
DWORD pi; DWORD pi;
@ -59,6 +60,11 @@ struct BindProtocol {
CRITICAL_SECTION section; CRITICAL_SECTION section;
task_header_t *task_queue_head, *task_queue_tail; task_header_t *task_queue_head, *task_queue_tail;
BYTE *buf;
DWORD buf_size;
LPWSTR mime;
LPWSTR url;
}; };
#define PROTOCOL(x) ((IInternetProtocol*) &(x)->lpInternetProtocolVtbl) #define PROTOCOL(x) ((IInternetProtocol*) &(x)->lpInternetProtocolVtbl)
@ -67,6 +73,9 @@ struct BindProtocol {
#define SERVPROV(x) ((IServiceProvider*) &(x)->lpServiceProviderVtbl) #define SERVPROV(x) ((IServiceProvider*) &(x)->lpServiceProviderVtbl)
#define PROTSINK(x) ((IInternetProtocolSink*) &(x)->lpInternetProtocolSinkVtbl) #define PROTSINK(x) ((IInternetProtocolSink*) &(x)->lpInternetProtocolSinkVtbl)
#define BUFFER_SIZE 2048
#define MIME_TEST_SIZE 255
#define WM_MK_CONTINUE (WM_USER+101) #define WM_MK_CONTINUE (WM_USER+101)
#define WM_MK_RELEASE (WM_USER+102) #define WM_MK_RELEASE (WM_USER+102)
@ -272,6 +281,9 @@ static ULONG WINAPI BindProtocol_Release(IInternetProtocol *iface)
if(This->notif_hwnd) if(This->notif_hwnd)
release_notif_hwnd(This->notif_hwnd); release_notif_hwnd(This->notif_hwnd);
DeleteCriticalSection(&This->section); DeleteCriticalSection(&This->section);
heap_free(This->mime);
heap_free(This->url);
heap_free(This); heap_free(This);
URLMON_UnlockModule(); URLMON_UnlockModule();
@ -300,6 +312,7 @@ static HRESULT WINAPI BindProtocol_Start(IInternetProtocol *iface, LPCWSTR szUrl
return E_INVALIDARG; return E_INVALIDARG;
This->pi = grfPI; This->pi = grfPI;
This->url = heap_strdupW(szUrl);
hres = IInternetProtocolSink_QueryInterface(pOIProtSink, &IID_IServiceProvider, hres = IInternetProtocolSink_QueryInterface(pOIProtSink, &IID_IServiceProvider,
(void**)&service_provider); (void**)&service_provider);
@ -417,11 +430,30 @@ static HRESULT WINAPI BindProtocol_Read(IInternetProtocol *iface, void *pv,
{ {
BindProtocol *This = PROTOCOL_THIS(iface); BindProtocol *This = PROTOCOL_THIS(iface);
ULONG read = 0; ULONG read = 0;
HRESULT hres; HRESULT hres = S_OK;
TRACE("(%p)->(%p %u %p)\n", This, pv, cb, pcbRead); TRACE("(%p)->(%p %u %p)\n", This, pv, cb, pcbRead);
hres = IInternetProtocol_Read(This->protocol, pv, cb, &read); if(This->buf) {
read = min(cb, This->buf_size);
memcpy(pv, This->buf, read);
if(read == This->buf_size) {
heap_free(This->buf);
This->buf = NULL;
}else {
memmove(This->buf, This->buf+cb, This->buf_size-cb);
}
This->buf_size -= read;
}
if(read < cb) {
ULONG cread = 0;
hres = IInternetProtocol_Read(This->protocol, (BYTE*)pv+read, cb, &cread);
read += cread;
}
*pcbRead = read; *pcbRead = read;
return hres; return hres;
@ -674,6 +706,47 @@ static HRESULT WINAPI BPInternetProtocolSink_Switch(IInternetProtocolSink *iface
return IInternetProtocolSink_Switch(This->protocol_sink, pProtocolData); return IInternetProtocolSink_Switch(This->protocol_sink, pProtocolData);
} }
static void report_progress(BindProtocol *This, ULONG status_code, LPCWSTR status_text)
{
switch(status_code) {
case BINDSTATUS_FINDINGRESOURCE:
case BINDSTATUS_CONNECTING:
case BINDSTATUS_BEGINDOWNLOADDATA:
case BINDSTATUS_SENDINGREQUEST:
case BINDSTATUS_CACHEFILENAMEAVAILABLE:
case BINDSTATUS_DIRECTBIND:
case BINDSTATUS_ACCEPTRANGES:
if(This->protocol_sink)
IInternetProtocolSink_ReportProgress(This->protocol_sink, status_code, status_text);
break;
case BINDSTATUS_MIMETYPEAVAILABLE:
if(!This->reported_mime) {
heap_free(This->mime);
This->mime = heap_strdupW(status_text);
}
if(This->protocol_sink && !(This->pi & PI_MIMEVERIFICATION))
IInternetProtocolSink_ReportProgress(This->protocol_sink, status_code, status_text);
break;
case BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE:
if(!This->reported_mime) {
heap_free(This->mime);
This->mime = heap_strdupW(status_text);
}
if(This->protocol_sink) {
This->reported_mime = TRUE;
IInternetProtocolSink_ReportProgress(This->protocol_sink, BINDSTATUS_MIMETYPEAVAILABLE, status_text);
}
break;
default:
FIXME("unsupported ulStatusCode %u\n", status_code);
}
}
typedef struct { typedef struct {
task_header_t header; task_header_t header;
@ -685,28 +758,12 @@ static void on_progress_proc(BindProtocol *This, task_header_t *t)
{ {
on_progress_task_t *task = (on_progress_task_t*)t; on_progress_task_t *task = (on_progress_task_t*)t;
IInternetProtocolSink_ReportProgress(This->protocol_sink, task->status_code, task->status_text); report_progress(This, task->status_code, task->status_text);
heap_free(task->status_text); heap_free(task->status_text);
heap_free(task); heap_free(task);
} }
static void report_progress(BindProtocol *This, ULONG status_code, LPCWSTR status_text)
{
if(do_direct_notif(This)) {
IInternetProtocolSink_ReportProgress(This->protocol_sink, status_code, status_text);
}else {
on_progress_task_t *task;
task = heap_alloc(sizeof(on_progress_task_t));
task->status_code = status_code;
task->status_text = heap_strdupW(status_text);
push_task(This, &task->header, on_progress_proc);
}
}
static HRESULT WINAPI BPInternetProtocolSink_ReportProgress(IInternetProtocolSink *iface, static HRESULT WINAPI BPInternetProtocolSink_ReportProgress(IInternetProtocolSink *iface,
ULONG ulStatusCode, LPCWSTR szStatusText) ULONG ulStatusCode, LPCWSTR szStatusText)
{ {
@ -714,36 +771,68 @@ static HRESULT WINAPI BPInternetProtocolSink_ReportProgress(IInternetProtocolSin
TRACE("(%p)->(%u %s)\n", This, ulStatusCode, debugstr_w(szStatusText)); TRACE("(%p)->(%u %s)\n", This, ulStatusCode, debugstr_w(szStatusText));
switch(ulStatusCode) { if(do_direct_notif(This)) {
case BINDSTATUS_FINDINGRESOURCE:
case BINDSTATUS_CONNECTING:
case BINDSTATUS_BEGINDOWNLOADDATA:
case BINDSTATUS_SENDINGREQUEST:
case BINDSTATUS_CACHEFILENAMEAVAILABLE:
case BINDSTATUS_DIRECTBIND:
case BINDSTATUS_ACCEPTRANGES:
case BINDSTATUS_MIMETYPEAVAILABLE:
if(!This->protocol_sink)
return S_OK;
report_progress(This, ulStatusCode, szStatusText); report_progress(This, ulStatusCode, szStatusText);
break; }else {
on_progress_task_t *task;
case BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE: task = heap_alloc(sizeof(on_progress_task_t));
if(!This->protocol_sink)
return S_OK;
report_progress(This,
This->from_urlmon ? BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE : BINDSTATUS_MIMETYPEAVAILABLE,
szStatusText);
break;
default: task->status_code = ulStatusCode;
FIXME("unsupported ulStatusCode %u\n", ulStatusCode); task->status_text = heap_strdupW(szStatusText);
return E_NOTIMPL;
push_task(This, &task->header, on_progress_proc);
} }
return S_OK; return S_OK;
} }
static HRESULT report_data(BindProtocol *This, DWORD bscf, ULONG progress, ULONG progress_max)
{
if(!This->protocol_sink)
return S_OK;
if((This->pi & PI_MIMEVERIFICATION) && !This->reported_mime) {
DWORD read = 0;
LPWSTR mime;
HRESULT hres;
if(!This->buf) {
This->buf = heap_alloc(BUFFER_SIZE);
if(!This->buf)
return E_OUTOFMEMORY;
}
do {
read = 0;
hres = IInternetProtocol_Read(This->protocol, This->buf+This->buf_size, BUFFER_SIZE-This->buf_size, &read);
if(hres != S_OK)
break;
This->buf_size += read;
}while(This->buf_size < MIME_TEST_SIZE);
if(FAILED(hres) && hres != E_PENDING)
return hres;
This->buf_size += read;
if(This->buf_size < MIME_TEST_SIZE && hres != S_FALSE)
return S_OK;
hres = FindMimeFromData(NULL, This->url, This->buf, min(This->buf_size, MIME_TEST_SIZE), This->mime, 0, &mime, 0);
if(FAILED(hres))
return hres;
heap_free(This->mime);
This->mime = heap_strdupW(mime);
CoTaskMemFree(mime);
This->reported_mime = TRUE;
IInternetProtocolSink_ReportProgress(This->protocol_sink, BINDSTATUS_MIMETYPEAVAILABLE, This->mime);
}
return IInternetProtocolSink_ReportData(This->protocol_sink, bscf, progress, progress_max);
}
typedef struct { typedef struct {
task_header_t header; task_header_t header;
DWORD bscf; DWORD bscf;
@ -755,9 +844,7 @@ static void report_data_proc(BindProtocol *This, task_header_t *t)
{ {
report_data_task_t *task = (report_data_task_t*)t; report_data_task_t *task = (report_data_task_t*)t;
if(This->protocol_sink) report_data(This, task->bscf, task->progress, task->progress_max);
IInternetProtocolSink_ReportData(This->protocol_sink, task->bscf, task->progress, task->progress_max);
heap_free(task); heap_free(task);
} }
@ -786,7 +873,7 @@ static HRESULT WINAPI BPInternetProtocolSink_ReportData(IInternetProtocolSink *i
return S_OK; return S_OK;
} }
return IInternetProtocolSink_ReportData(This->protocol_sink, grfBSCF, ulProgress, ulProgressMax); return report_data(This, grfBSCF, ulProgress, ulProgressMax);
} }
typedef struct { typedef struct {