You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2147 lines
53 KiB
2147 lines
53 KiB
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1996.
|
|
//
|
|
// File: URLOSTRM.CXX
|
|
//
|
|
// Contents: Public interface and implementation of the URL
|
|
// Open Stream APIs
|
|
//
|
|
// Classes: many (see below)
|
|
//
|
|
// Functions:
|
|
//
|
|
// History: 04-22-96 VictorS Created
|
|
// 05-16-96 Jobi Modified
|
|
// 06-06-96 Jobi Modified
|
|
// 07-05-96 Ramesh Modified
|
|
//----------------------------------------------------------------------------
|
|
#ifndef unix
|
|
// For some reaon on unix this causes NOERROR compile errors
|
|
#include "winerror.h"
|
|
#endif /* unix */
|
|
#include "ocidl.h"
|
|
#include "servprov.h"
|
|
#include "tchar.h"
|
|
#include "wininet.h"
|
|
#include "urlmki.h"
|
|
#include "urlhlink.h"
|
|
#include <shlwapi.h>
|
|
#include <shlwapip.h>
|
|
|
|
#define URLOSTRM_DONOT_NOTIFY_ONDATA 0xFF
|
|
#define URLOSTRM_NOTIFY_ONDATA 0x00
|
|
|
|
//----------------------------------------------------------//
|
|
// //
|
|
// This file can never be compiled with the _UNICODE or //
|
|
// UNICODE macros defined. //
|
|
// //
|
|
//----------------------------------------------------------//
|
|
|
|
//----------------------------------------------------------//
|
|
// MACROS
|
|
//----------------------------------------------------------//
|
|
|
|
// These macros can go away when macros and implementation of
|
|
// the InetSDK has settled down...
|
|
|
|
#define IS_E_PENDING(x) (x == E_PENDING)
|
|
#define LPUOSCALLBACK LPBINDSTATUSCALLBACK
|
|
#define PUMPREAD(strm) \
|
|
{ \
|
|
DWORD dwSize = 0; \
|
|
char * x = new char[20]; \
|
|
hr = strm->Read(x, 20, &dwSize ); \
|
|
if( !IS_E_PENDING(hr) && (dwSize != 0) ) \
|
|
{ \
|
|
DPRINTF( ("Data on the over read! %d\n", dwSize) ); \
|
|
} \
|
|
delete x; \
|
|
}
|
|
|
|
#define HANDLE_ABORT(hr) \
|
|
{ if( hr == E_ABORT) \
|
|
{ m_bInAbort = 1; \
|
|
if( m_binding ) \
|
|
m_binding->Abort(); \
|
|
return(E_ABORT); \
|
|
} \
|
|
}
|
|
|
|
#define CHECK_MEMORY(ptr) \
|
|
{ if( !ptr ) \
|
|
{ DPRINTF( ("Failed to alloc memory") ); \
|
|
m_bInAbort = 1; \
|
|
if( m_binding ) \
|
|
m_binding->Abort(); \
|
|
return(E_OUTOFMEMORY); \
|
|
} \
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Refcount helper
|
|
//
|
|
|
|
// Standardized COM Ref counting. ASSUMES that class has a ULONG
|
|
// data member called 'm_ref'.
|
|
|
|
#define IMPLEMENT_REFCOUNT(clsname) \
|
|
STDMETHOD_(ULONG, AddRef)() \
|
|
{ CHECK_INTERFACE(this); \
|
|
DPRINTF( ("(%#08x) " #clsname "::Addref %d\n", this, m_ref+1) );\
|
|
return(++m_ref); }\
|
|
STDMETHOD_(ULONG, Release)()\
|
|
{ CHECK_INTERFACE(this); \
|
|
DPRINTF( ( "(%#08x) " #clsname "::Release : %d\n", this, m_ref-1) );\
|
|
if( !--m_ref )\
|
|
{ delete this; return 0; }\
|
|
return m_ref;\
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// DEBUG MACROS
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
#ifdef _DEBUG
|
|
|
|
// Check the validity of a pointer - use this for all allocated memory
|
|
|
|
#define CHECK_POINTER(val) \
|
|
if (!(val) || IsBadWritePtr((void *)(val), sizeof(void *))) \
|
|
{ DPRINTF( ("BAD POINTER: %s!\n", #val ) ); \
|
|
return E_POINTER; }
|
|
|
|
// Check the validity of an interface pointer. Use this for all pointers
|
|
// to C++ objects that are supposed to have vtables.
|
|
|
|
#define CHECK_INTERFACE(val) \
|
|
if (!(val) || IsBadWritePtr((void *)(val), sizeof(void *)) \
|
|
|| IsBadCodePtr( FARPROC( *((DWORD **)val)) ) ) \
|
|
{ DPRINTF( ("BAD INTERFACE: %s!\n", #val ) ); \
|
|
return E_POINTER; }
|
|
|
|
// Simple assert. Map this to whatever general
|
|
// framework assert you want.
|
|
|
|
#define UOSASSERT(x) { if(!(x)) dprintf( "Assert in URLOSTRM API: %s\n", #x ); }
|
|
|
|
#define DUOS OutputDebugString( "URLOSTRM API: " );
|
|
#define DPRINTF(x) DUOS dprintf x ;
|
|
|
|
#ifndef CHECK_METHOD
|
|
#define CHECK_METHOD( m, args ) DUOS dprintf( "(%#08x) %s(", this, #m ); dprintf args ; dprintf(")\n");
|
|
#endif
|
|
|
|
#ifndef MEMPRINTF
|
|
#define MEMPRINTF(x) DPRINTF(x)
|
|
#endif
|
|
|
|
void dprintf( char * format, ... )
|
|
{
|
|
char out[1024];
|
|
va_list marker;
|
|
va_start(marker, format);
|
|
wvsprintf(out, format, marker);
|
|
va_end(marker);
|
|
OutputDebugString( out );
|
|
}
|
|
|
|
|
|
#else
|
|
#define CHECK_POINTER(x)
|
|
#define CHECK_INTERFACE(x)
|
|
#define CHECK_METHOD( m, args )
|
|
#define UOSASSERT(x)
|
|
#define DPRINTF(x)
|
|
#define MEMPRINTF(x)
|
|
#endif
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// Local heap stuff
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
// Keeping this here makes this code portable to any .dll
|
|
static HANDLE g_hHeap;
|
|
|
|
#ifdef _DEBUG
|
|
// Uncomment the line below for Debug spew of memory stuff
|
|
//#define MONITER_MEMALLOC 1
|
|
#endif
|
|
|
|
#ifdef _DEBUG
|
|
static void * _cdecl
|
|
operator new( size_t size )
|
|
{
|
|
if( !g_hHeap )
|
|
g_hHeap = ::GetProcessHeap();
|
|
|
|
// Heap alloc is the fastest gun in the west
|
|
// for the type of allocations we do here.
|
|
void * p = HeapAlloc(g_hHeap, 0, size);
|
|
|
|
MEMPRINTF( ("operator new(%d) returns(%#08X)\n",size, DWORD(p)) );
|
|
|
|
return(p);
|
|
}
|
|
|
|
static void _cdecl
|
|
operator delete ( void *ptr)
|
|
{
|
|
MEMPRINTF( ("operator delete(%#08X)\n", DWORD(ptr) ) );
|
|
|
|
HeapFree(g_hHeap, 0, ptr);
|
|
}
|
|
#endif
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// class CBuffer
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
|
|
// Generic CBuffer class for quick and dirty mem allocs.
|
|
// Caller must check return results.
|
|
|
|
class CBuffer
|
|
{
|
|
public:
|
|
CBuffer(ULONG cBytes);
|
|
~CBuffer();
|
|
|
|
void *GetBuffer();
|
|
|
|
private:
|
|
void * m_pBuf;
|
|
// we'll use this temp buffer for small cases.
|
|
//
|
|
char m_szTmpBuf[120];
|
|
unsigned m_fHeapAlloc:1;
|
|
};
|
|
|
|
inline
|
|
CBuffer::CBuffer(ULONG cBytes)
|
|
{
|
|
if( !g_hHeap )
|
|
g_hHeap = ::GetProcessHeap();
|
|
|
|
m_pBuf = (cBytes <= 120) ? m_szTmpBuf : HeapAlloc(g_hHeap, 0, cBytes);
|
|
m_fHeapAlloc = (cBytes > 120);
|
|
}
|
|
|
|
inline
|
|
CBuffer::~CBuffer()
|
|
{
|
|
if (m_pBuf && m_fHeapAlloc) HeapFree(g_hHeap, 0, m_pBuf);
|
|
}
|
|
|
|
inline
|
|
void * CBuffer::GetBuffer()
|
|
{
|
|
return m_pBuf;
|
|
}
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// String ANSI <-> WIDE helper macros
|
|
//
|
|
// This stuff stolen from marcwan...
|
|
//
|
|
// allocates a temporary buffer that will disappear when it goes out of scope
|
|
// NOTE: be careful of that -- make sure you use the string in the same or
|
|
// nested scope in which you created this buffer. people should not use this
|
|
// class directly. use the macro(s) below.
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
#define MAKE_WIDE(ptrname) \
|
|
long __l##ptrname = (lstrlen(ptrname) + 1) * sizeof(WCHAR); \
|
|
CBuffer __CBuffer##ptrname(__l##ptrname); \
|
|
CHECK_POINTER(__CBuffer##ptrname.GetBuffer()); \
|
|
if( !__CBuffer##ptrname.GetBuffer()) \
|
|
return( E_OUTOFMEMORY ); \
|
|
MultiByteToWideChar(CP_ACP, 0, ptrname, -1, \
|
|
(LPWSTR)__CBuffer##ptrname.GetBuffer(), __l##ptrname); \
|
|
LPWSTR __w##ptrname = (LPWSTR)__CBuffer##ptrname.GetBuffer()
|
|
|
|
#define WIDE_NAME(ptrname) __w##ptrname
|
|
|
|
#define MAKE_ANSI(ptrname) \
|
|
long __l##ptrname = (lstrlenW(ptrname)*2 + 1) * sizeof(char); \
|
|
CBuffer __CBuffer##ptrname(__l##ptrname); \
|
|
CHECK_POINTER(__CBuffer##ptrname.GetBuffer()); \
|
|
if( !__CBuffer##ptrname.GetBuffer()) \
|
|
return( E_OUTOFMEMORY ); \
|
|
WideCharToMultiByte(CP_ACP, 0, ptrname, -1, \
|
|
(LPSTR)__CBuffer##ptrname.GetBuffer(), __l##ptrname, NULL, NULL); \
|
|
LPSTR __a##ptrname = (LPSTR)__CBuffer##ptrname.GetBuffer()
|
|
|
|
#define ANSI_NAME(ptrname) __a##ptrname
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// Misc helper functions
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
// These registry functions are here for support of backdoor
|
|
// flags and screamer features.
|
|
|
|
static HRESULT
|
|
GetRegDword( HKEY mainkey, LPCTSTR subkey, LPCTSTR valueName, DWORD * result )
|
|
{
|
|
HKEY hkey = 0;
|
|
DWORD dwDisposition;
|
|
|
|
LONG dwResult = RegCreateKeyEx(
|
|
mainkey, subkey,
|
|
0, // DWORD Reserved, // reserved
|
|
0, // LPTSTR lpClass, // address of class string
|
|
REG_OPTION_NON_VOLATILE, // DWORD dwOptions, // special options flag
|
|
KEY_ALL_ACCESS, // REGSAM samDesired, // desired security access
|
|
0, // LPSECURITY_ATTRIBUTES lpSecurityAttributes, // address of key security structure
|
|
&hkey, // PHKEY phkResult, // address of buffer for opened handle
|
|
&dwDisposition // LPDWORD lpdwDisposition // address of disposition value buffer
|
|
);
|
|
|
|
HRESULT hr = dwResult == ERROR_SUCCESS ? NOERROR : E_FAIL;
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
DWORD dwType;
|
|
DWORD dwSize = sizeof(DWORD);
|
|
DWORD dwSavedResult = *result;
|
|
|
|
dwResult = RegQueryValueEx(
|
|
hkey, // handle of key to query
|
|
valueName,
|
|
0, // LPDWORD lpReserved, // reserved
|
|
&dwType, // LPDWORD lpType, // address of buffer for value type
|
|
(LPBYTE)result, // LPBYTE lpData, // address of data buffer
|
|
&dwSize // LPDWORD lpcbData // address of data buffer size
|
|
);
|
|
|
|
hr = dwResult == ERROR_SUCCESS ? NOERROR : E_FAIL;
|
|
|
|
if( FAILED(hr) )
|
|
*result = dwSavedResult;
|
|
}
|
|
|
|
if( hkey )
|
|
RegCloseKey(hkey);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
static HRESULT
|
|
GetDLMRegDWord( LPCTSTR valueName, DWORD * result )
|
|
{
|
|
return(GetRegDword( HKEY_LOCAL_MACHINE,
|
|
_TEXT("Software\\Microsoft\\DownloadManager"),
|
|
valueName,
|
|
result ) );
|
|
}
|
|
|
|
|
|
static HRESULT
|
|
MyCreateFile( LPCWSTR filename, HANDLE & hfile )
|
|
{
|
|
// BUGBUG: in retrospect this should be a ansi function
|
|
// not a wide string one.
|
|
|
|
HRESULT hr = NOERROR;
|
|
|
|
/**********
|
|
MAKE_ANSI( filename );
|
|
|
|
hfile = ::CreateFileA(
|
|
ANSI_NAME(filename), // LPCTSTR lpFileName, // pointer to name of the file
|
|
GENERIC_WRITE, // DWORD dwDesiredAccess, // access (read-write) mode
|
|
0, // DWORD dwShareMode, // share mode
|
|
0, // LPSECURITY_ATTRIBUTES lpSecurityAttributes, // pointer to security descriptor
|
|
CREATE_ALWAYS, // DWORD dwCreationDistribution, // how to create
|
|
FILE_ATTRIBUTE_NORMAL, //DWORD dwFlagsAndAttributes, // file attributes
|
|
0 // HANDLE hTemplateFile // handle to file with attributes to copy
|
|
);
|
|
*************/
|
|
hfile = CreateFileWrapW(
|
|
filename,
|
|
GENERIC_WRITE,
|
|
0,
|
|
0,
|
|
CREATE_ALWAYS,
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
0
|
|
);
|
|
|
|
// Our code likes HRESULT style error handling
|
|
|
|
if( hfile == INVALID_HANDLE_VALUE )
|
|
hr = MK_E_CANTOPENFILE;
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// BindStatusCallback base class
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
//
|
|
// This is the base class for the download objects. It implements
|
|
// the url mon callback interface (IBindStatusCallback) and
|
|
// IServiceProvider -- which it delegates to the caller's IBSCB.
|
|
//
|
|
|
|
|
|
// State flags
|
|
|
|
class CBaseBSCB : public IBindStatusCallbackMsg,
|
|
public IServiceProvider
|
|
{
|
|
public:
|
|
|
|
CBaseBSCB( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback );
|
|
virtual ~CBaseBSCB();
|
|
|
|
STDMETHOD(KickOffDownload)( LPCWSTR szURL );
|
|
|
|
// IUnknown
|
|
STDMETHOD(QueryInterface)(REFIID riid, void **ppvObjOut);
|
|
|
|
IMPLEMENT_REFCOUNT(CBaseBSCB);
|
|
|
|
// IBindStatusCallback
|
|
|
|
STDMETHODIMP OnStartBinding(
|
|
DWORD grfBSCOption,
|
|
IBinding *pib);
|
|
|
|
STDMETHODIMP GetPriority(
|
|
LONG *pnPriority);
|
|
|
|
STDMETHODIMP OnLowResource(
|
|
DWORD reserved);
|
|
|
|
STDMETHODIMP OnProgress(
|
|
ULONG ulProgress,
|
|
ULONG ulProgressMax,
|
|
ULONG ulStatusCode,
|
|
LPCWSTR szStatusText);
|
|
|
|
STDMETHODIMP OnDataAvailable(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed);
|
|
|
|
STDMETHODIMP OnStopBinding(
|
|
HRESULT hresult,
|
|
LPCWSTR szError);
|
|
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
|
|
STDMETHODIMP OnObjectAvailable(
|
|
REFIID riid,
|
|
IUnknown *punk);
|
|
|
|
STDMETHODIMP MessagePending(
|
|
DWORD dwPendingType,
|
|
DWORD dwPendingRecursion,
|
|
DWORD dwReserved);
|
|
|
|
// IServiceProvider
|
|
|
|
STDMETHODIMP QueryService(
|
|
REFGUID rsid,
|
|
REFIID iid,
|
|
void **ppvObj);
|
|
|
|
|
|
// Local methods
|
|
|
|
void Abort();
|
|
BOOL IsAborted();
|
|
BOOL DownloadDone();
|
|
HRESULT FinalResult();
|
|
void SetEncodingFlags( ULONG flags );
|
|
|
|
IUnknown * Caller();
|
|
|
|
// I guess at one point I thought it would be cool
|
|
// to make all of these inlines and isolated from
|
|
// the core functionality.
|
|
|
|
HRESULT SignalOnData( DWORD flags, ULONG size, FORMATETC *pformatetc);
|
|
HRESULT SignalOnProgress( ULONG status, ULONG size, ULONG maxSize, LPCWSTR msg );
|
|
HRESULT SignalOnStopBinding( HRESULT hr, LPCWSTR msg );
|
|
|
|
HRESULT SignalOnStartBinding( DWORD grfBSCOption, IBinding *pib);
|
|
HRESULT SignalOnGetPriority(LONG*);
|
|
HRESULT SignalOnLowResource(DWORD);
|
|
HRESULT SignalGetBindInfo(DWORD *grfBINDF,BINDINFO *pbindinfo);
|
|
|
|
virtual void Neutralize();
|
|
|
|
ULONG m_ref;
|
|
LPUOSCALLBACK m_callback;
|
|
IUnknown * m_caller;
|
|
IBinding * m_binding;
|
|
IServiceProvider * m_callbackServiceProvider;
|
|
IBindStatusCallbackMsg *_pBSCBMsg;
|
|
BOOL m_bInAbort : 1;
|
|
BOOL m_bInCache : 1;
|
|
BOOL m_bCheckedForServiceProvider : 1;
|
|
DWORD m_readSoFar;
|
|
HRESULT m_finalResult;
|
|
|
|
// See notes above about IStream usage
|
|
|
|
IStream * m_UserStream;
|
|
|
|
ULONG m_maxSize;
|
|
DWORD m_bscoFlags;
|
|
UINT m_encoding;
|
|
char m_szCacheFileName[MAX_PATH];
|
|
};
|
|
|
|
|
|
/*---------------------*/
|
|
/* INLINES */
|
|
/*---------------------*/
|
|
|
|
inline void CBaseBSCB::Abort() { m_bInAbort = 1; }
|
|
inline BOOL CBaseBSCB::IsAborted() { return(m_bInAbort); }
|
|
inline HRESULT CBaseBSCB::FinalResult(){ return( m_finalResult ); }
|
|
inline IUnknown*CBaseBSCB::Caller() { return( m_caller ); }
|
|
inline void CBaseBSCB::SetEncodingFlags( ULONG flags ) { m_encoding = flags; }
|
|
|
|
inline HRESULT
|
|
CBaseBSCB::SignalOnData( DWORD flags, ULONG size, FORMATETC *pformatetc )
|
|
{
|
|
HRESULT hr=NOERROR;
|
|
|
|
if(m_bscoFlags!=URLOSTRM_NOTIFY_ONDATA)
|
|
return(hr);
|
|
|
|
STGMEDIUM stg;
|
|
|
|
stg.tymed = TYMED_ISTREAM;
|
|
stg.pstm = m_UserStream;
|
|
stg.pUnkForRelease = NULL;
|
|
|
|
if(m_callback)
|
|
hr=m_callback->OnDataAvailable(flags,size,pformatetc,&stg);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
inline HRESULT
|
|
CBaseBSCB::SignalOnProgress( ULONG status, ULONG size, ULONG maxSize, LPCWSTR msg )
|
|
{
|
|
if( !m_callback )
|
|
return(NOERROR);
|
|
|
|
if( size && !maxSize )
|
|
maxSize = size;
|
|
|
|
if( maxSize > m_maxSize )
|
|
m_maxSize = maxSize;
|
|
|
|
HRESULT hr = m_callback->OnProgress( size, m_maxSize, status, msg );
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
inline HRESULT
|
|
CBaseBSCB::SignalOnStopBinding( HRESULT hres, LPCWSTR msg )
|
|
{
|
|
if( !m_callback )
|
|
return(NOERROR);
|
|
|
|
HRESULT hr = m_callback->OnStopBinding( hres, msg );
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
|
|
inline HRESULT
|
|
CBaseBSCB::SignalOnStartBinding( DWORD grfBSCOption, IBinding *pib)
|
|
{
|
|
if( !m_callback )
|
|
return(NOERROR);
|
|
return( m_callback->OnStartBinding(grfBSCOption,pib) );
|
|
}
|
|
|
|
|
|
inline HRESULT
|
|
CBaseBSCB:: SignalOnGetPriority(LONG* lng)
|
|
{
|
|
if( !m_callback )
|
|
return(E_NOTIMPL);
|
|
return(m_callback->GetPriority(lng));
|
|
}
|
|
|
|
inline HRESULT
|
|
CBaseBSCB:: SignalOnLowResource(DWORD dw)
|
|
{
|
|
if( !m_callback )
|
|
return( NOERROR );
|
|
return( m_callback->OnLowResource(dw) );
|
|
}
|
|
|
|
inline HRESULT
|
|
CBaseBSCB::SignalGetBindInfo(DWORD *grfBINDF, BINDINFO * pbindinfo)
|
|
{
|
|
if( !m_callback )
|
|
return(E_NOTIMPL);
|
|
return( m_callback->GetBindInfo(grfBINDF, pbindinfo) );
|
|
}
|
|
|
|
|
|
/*---------------------*/
|
|
/* OUT-OF-LINES */
|
|
/*---------------------*/
|
|
|
|
|
|
// Do nothing CTOR
|
|
CBaseBSCB::CBaseBSCB
|
|
(
|
|
IUnknown * caller,
|
|
DWORD bscof,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
m_binding = 0;
|
|
m_ref = 0;
|
|
m_bInAbort = 0;
|
|
m_bCheckedForServiceProvider = 0;
|
|
m_bInCache = 0;
|
|
m_readSoFar = 0;
|
|
m_UserStream = 0;
|
|
m_encoding = 0;
|
|
m_bscoFlags = bscof;
|
|
m_callbackServiceProvider = 0;
|
|
m_szCacheFileName[0] = NULL;
|
|
m_finalResult = S_OK;
|
|
|
|
_pBSCBMsg = 0;
|
|
|
|
if( (m_callback = callback) != 0 )
|
|
m_callback->AddRef();
|
|
|
|
if( (m_caller = caller) != 0 )
|
|
caller->AddRef();
|
|
}
|
|
|
|
// Cleanup just call Neutralize();
|
|
CBaseBSCB::~CBaseBSCB()
|
|
{
|
|
Neutralize();
|
|
}
|
|
|
|
void
|
|
CBaseBSCB::Neutralize()
|
|
{
|
|
if( m_binding )
|
|
{
|
|
m_binding->Release();
|
|
m_binding = 0;
|
|
}
|
|
if( m_caller )
|
|
{
|
|
m_caller->Release();
|
|
m_caller = 0;
|
|
}
|
|
if( m_callback )
|
|
{
|
|
m_callback->Release();
|
|
m_callback = 0;
|
|
}
|
|
if( m_callbackServiceProvider )
|
|
{
|
|
m_callbackServiceProvider->Release();
|
|
m_callbackServiceProvider = 0;
|
|
}
|
|
if( m_UserStream )
|
|
{
|
|
m_UserStream->Release();
|
|
m_UserStream = 0;
|
|
}
|
|
if (_pBSCBMsg)
|
|
{
|
|
_pBSCBMsg->Release();
|
|
}
|
|
}
|
|
|
|
// IUnknown::QueryInterface
|
|
STDMETHODIMP
|
|
CBaseBSCB::QueryInterface
|
|
(
|
|
const GUID &iid,
|
|
void ** ppv
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::QueryInterface, ("") );
|
|
|
|
if (iid==IID_IUnknown || iid==IID_IBindStatusCallback)
|
|
{
|
|
*ppv =(IBindStatusCallback*)this;
|
|
AddRef();
|
|
return(NOERROR);
|
|
}
|
|
|
|
|
|
if( iid==IID_IServiceProvider)
|
|
{
|
|
*ppv =(IServiceProvider*)this;
|
|
AddRef();
|
|
return(NOERROR);
|
|
}
|
|
|
|
if (iid==IID_IBindStatusCallbackMsg)
|
|
{
|
|
*ppv =(IBindStatusCallbackMsg*)this;
|
|
AddRef();
|
|
return(NOERROR);
|
|
}
|
|
|
|
|
|
return( E_NOINTERFACE );
|
|
}
|
|
|
|
// IServiceProvider::QueryService
|
|
STDMETHODIMP
|
|
CBaseBSCB::QueryService
|
|
(
|
|
REFGUID rsid,
|
|
REFIID iid,
|
|
void **ppvObj
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::QueryService, ("") );
|
|
|
|
HRESULT hr = E_NOINTERFACE;
|
|
|
|
if (iid==IID_IBindStatusCallback)
|
|
{
|
|
*ppvObj =(IBindStatusCallbackMsg*)this;
|
|
AddRef();
|
|
return(NOERROR);
|
|
}
|
|
|
|
|
|
if( m_callback )
|
|
hr = m_callback->QueryInterface( iid, ppvObj );
|
|
|
|
if( FAILED(hr) && !m_callbackServiceProvider && !m_bCheckedForServiceProvider )
|
|
{
|
|
m_bCheckedForServiceProvider = 1;
|
|
|
|
if( m_callback )
|
|
{
|
|
hr = m_callback->QueryInterface
|
|
(
|
|
IID_IServiceProvider,
|
|
(void**)&m_callbackServiceProvider
|
|
);
|
|
}
|
|
|
|
if( SUCCEEDED(hr) && m_callbackServiceProvider )
|
|
hr = m_callbackServiceProvider->QueryService(rsid,iid,ppvObj);
|
|
else
|
|
hr = E_NOINTERFACE; // BUGBUG: what's that error code again?
|
|
}
|
|
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::OnStartBinding
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnStartBinding
|
|
(
|
|
DWORD grfBSCOption,
|
|
IBinding *pib
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::OnStartBinding, ("flags: %#08x, IBinding: %#08x",grfBSCOption,pib) );
|
|
|
|
CHECK_INTERFACE(pib);
|
|
|
|
HRESULT hr = SignalOnStartBinding(grfBSCOption,pib);
|
|
|
|
// smooth over user's e_not_implemented for when we
|
|
// return to urlmon
|
|
|
|
if( hr == E_NOTIMPL )
|
|
hr = NOERROR;
|
|
else
|
|
HANDLE_ABORT(hr);
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
pib->AddRef();
|
|
m_binding = pib;
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::GetPriority
|
|
STDMETHODIMP
|
|
CBaseBSCB::GetPriority
|
|
(
|
|
LONG *pnPriority
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::GetPriority, ("pnPriority: %#08x", pnPriority) );
|
|
CHECK_POINTER(pnPriority);
|
|
|
|
if (!pnPriority)
|
|
return E_POINTER;
|
|
|
|
HRESULT hr = SignalOnGetPriority(pnPriority);
|
|
|
|
if( hr == E_NOTIMPL )
|
|
{
|
|
// only override if caller doesn't implement.
|
|
*pnPriority = NORMAL_PRIORITY_CLASS;
|
|
hr = NOERROR;
|
|
}
|
|
else
|
|
{
|
|
HANDLE_ABORT(hr);
|
|
}
|
|
|
|
return( hr );
|
|
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::OnLowResource
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnLowResource( DWORD rsv)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::OnLowResource, ("resv: %#08x",rsv) );
|
|
|
|
HRESULT hr = SignalOnLowResource(rsv);
|
|
|
|
// Keep downloading...
|
|
|
|
if( hr == E_NOTIMPL )
|
|
hr = NOERROR;
|
|
else
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::OnStopBinding
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnStopBinding
|
|
(
|
|
HRESULT hresult,
|
|
LPCWSTR szError
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::OnStopBinding, ("%#08X %ws", hresult, szError ? szError : L"[no error]" ) );
|
|
|
|
// Store the hresult so we can return it to caller in the
|
|
// blocking/sync case.
|
|
|
|
HRESULT hr = SignalOnStopBinding( m_finalResult = hresult, szError );
|
|
|
|
if( m_binding )
|
|
{
|
|
m_binding->Release();
|
|
m_binding = 0;
|
|
}
|
|
|
|
if( hr == E_NOTIMPL )
|
|
hr = NOERROR;
|
|
else
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::GetBindInfo
|
|
|
|
STDMETHODIMP
|
|
CBaseBSCB::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO* pbindinfo
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::GetBindInfo, ("grfBINDF: %#08x, pbinfinfo ",grfBINDF) );
|
|
|
|
CHECK_POINTER(grfBINDF);
|
|
CHECK_POINTER(pbindinfo);
|
|
|
|
*grfBINDF = 0;
|
|
|
|
HRESULT hr = SignalGetBindInfo(grfBINDF,pbindinfo);
|
|
|
|
if( SUCCEEDED(hr) || (hr == E_NOTIMPL) )
|
|
{
|
|
// Let the derived class choose the bind flags
|
|
|
|
if(m_encoding)
|
|
{
|
|
*grfBINDF |= m_encoding;
|
|
pbindinfo->grfBindInfoF |= m_encoding;
|
|
}
|
|
|
|
hr = NOERROR;
|
|
}
|
|
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// IBindStatusCallback::OnObjectAvailable
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnObjectAvailable
|
|
(
|
|
REFIID riid,
|
|
IUnknown *punk
|
|
)
|
|
{
|
|
// This should never be called
|
|
CHECK_METHOD(CBaseBSCB::OnObjectAvailable, ("!") );
|
|
UOSASSERT(0 && "This should never be called");
|
|
return(NOERROR);
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnProgress
|
|
(
|
|
ULONG ulProgress,
|
|
ULONG ulProgressMax,
|
|
ULONG ulStatusCode,
|
|
LPCWSTR szStatusText
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::OnProgress, ("!") );
|
|
|
|
// URL moniker has a habit of passing ZERO
|
|
// into ulProgressMax. So.. let's at least
|
|
// pass in the amount we have so far...
|
|
|
|
m_maxSize = ulProgressMax ? ulProgressMax : ulProgress;
|
|
|
|
// This is useful information for the IStream implementation
|
|
|
|
|
|
if( ulStatusCode == BINDSTATUS_USINGCACHEDCOPY )
|
|
m_bInCache = TRUE;
|
|
|
|
HRESULT hr;
|
|
|
|
hr = SignalOnProgress( ulStatusCode, ulProgress, ulProgressMax, szStatusText );
|
|
|
|
if( hr == E_NOTIMPL )
|
|
hr = NOERROR;
|
|
else
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
// IBindStatusCallback::OnDataAvailable.
|
|
|
|
STDMETHODIMP
|
|
CBaseBSCB::OnDataAvailable
|
|
(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed
|
|
)
|
|
{
|
|
CHECK_METHOD(CBaseBSCB::OnDataAvailable,
|
|
("Flags: %x, dwSize: %d", grfBSCF, dwSize) );
|
|
|
|
HRESULT hr = NOERROR;
|
|
|
|
// N.B Assumption here is that the pstgmed->pstm will always be the same
|
|
|
|
if( !m_UserStream )
|
|
{
|
|
// We need to bump the refcount every time we
|
|
// copy and store the pointer.
|
|
|
|
m_UserStream = pstgmed->pstm;
|
|
m_UserStream->AddRef();
|
|
}
|
|
|
|
if (*m_szCacheFileName == NULL)
|
|
{
|
|
STATSTG statstg;
|
|
DWORD dwVal = 0;
|
|
|
|
if (m_UserStream->Stat(&statstg,dwVal) == S_OK)
|
|
{
|
|
if (0==WideCharToMultiByte( CP_ACP, 0, statstg.pwcsName, lstrlenW(statstg.pwcsName)+1, m_szCacheFileName,
|
|
MAX_PATH, NULL, NULL))
|
|
{
|
|
m_szCacheFileName[0] = NULL;
|
|
}
|
|
if (statstg.pwcsName)
|
|
{
|
|
CoTaskMemFree(statstg.pwcsName);
|
|
statstg.pwcsName = NULL;
|
|
}
|
|
}
|
|
else
|
|
m_szCacheFileName[0] = NULL;
|
|
}
|
|
|
|
hr = SignalOnData( grfBSCF, dwSize, pformatetc );
|
|
|
|
// Tell the blocking state machine we are have data.
|
|
// ClearState( WAITING_FOR_DATA );
|
|
|
|
if( hr == E_NOTIMPL )
|
|
hr = NOERROR;
|
|
else
|
|
HANDLE_ABORT(hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
// IBindStatusCallback::MessagePending.
|
|
|
|
STDMETHODIMP CBaseBSCB::MessagePending(DWORD dwPendingType, DWORD dwPendingRecursion, DWORD dwReserved)
|
|
{
|
|
MSG msg;
|
|
HRESULT hr = NOERROR;
|
|
|
|
if (m_callback && !_pBSCBMsg)
|
|
{
|
|
hr = m_callback->QueryInterface(IID_IBindStatusCallbackMsg, (void **) &_pBSCBMsg);
|
|
}
|
|
|
|
if (_pBSCBMsg && hr == NOERROR )
|
|
{
|
|
hr = _pBSCBMsg->MessagePending(dwPendingType, dwPendingRecursion, dwReserved );
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
CBaseBSCB::KickOffDownload( LPCWSTR szURL )
|
|
{
|
|
HRESULT hr;
|
|
IOleObject * pOleObject = 0;
|
|
IServiceProvider * pServiceProvider = 0;
|
|
BOOL bUseCaller = (Caller() != 0);
|
|
IMoniker * pmkr = 0;
|
|
IBindCtx * pBndCtx = 0;
|
|
|
|
CHECK_POINTER(szURL);
|
|
UOSASSERT(*szURL);
|
|
|
|
|
|
IStream * pstrm = 0;
|
|
|
|
// Don't bother if we don't have a caller...
|
|
|
|
if( bUseCaller )
|
|
{
|
|
// By convention the we give the caller first crack at service
|
|
// provider. The assumption here is that if they implement it
|
|
// they have the decency to forward QS's to their container.
|
|
|
|
hr = Caller()->QueryInterface( IID_IServiceProvider,
|
|
(void**)&pServiceProvider );
|
|
|
|
if( FAILED(hr) )
|
|
{
|
|
// Ok, now try the 'slow way' : maybe the object is an 'OLE' object
|
|
// that knows about it's client site:
|
|
|
|
hr = Caller()->QueryInterface( IID_IOleObject, (void**)&pOleObject );
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
IOleClientSite * pClientSite = 0;
|
|
|
|
hr = pOleObject->GetClientSite(&pClientSite);
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
// Now see if we have a service provider at that site
|
|
hr = pClientSite->QueryInterface
|
|
( IID_IServiceProvider,
|
|
(void**)&pServiceProvider );
|
|
}
|
|
|
|
if( pClientSite )
|
|
pClientSite->Release();
|
|
}
|
|
else
|
|
{
|
|
// Ok, it's not an OLE object, maybe it's one of these
|
|
// new fangled 'ObjectWithSites':
|
|
|
|
IObjectWithSite * pObjWithSite = 0;
|
|
|
|
hr = Caller()->QueryInterface( IID_IObjectWithSite,
|
|
(void**)&pObjWithSite );
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
// Now see if we have a service provider at that site
|
|
|
|
hr = pObjWithSite->GetSite(IID_IServiceProvider,
|
|
(void**)&pServiceProvider);
|
|
}
|
|
|
|
if( pObjWithSite )
|
|
pObjWithSite->Release();
|
|
|
|
}
|
|
if( pOleObject )
|
|
pOleObject->Release();
|
|
|
|
}
|
|
|
|
// BUGBUG: In the code above we stop looking at one level up --
|
|
// this may be too harsh and we should loop on client sites
|
|
// until we get to the top...
|
|
|
|
if( !pServiceProvider )
|
|
hr = E_UNEXPECTED;
|
|
|
|
IBindHost * pBindHost = 0;
|
|
|
|
// Ok, we have a service provider, let's see if BindHost is
|
|
// available. (Here there is some upward delegation going on
|
|
// via service provider).
|
|
|
|
if( SUCCEEDED(hr) )
|
|
hr = pServiceProvider->QueryService( SID_SBindHost, IID_IBindHost,
|
|
(void**)&pBindHost );
|
|
|
|
if( pServiceProvider )
|
|
pServiceProvider->Release();
|
|
|
|
pmkr = 0;
|
|
|
|
if( pBindHost )
|
|
{
|
|
// This allows the container to actually drive the download
|
|
// by creating it's own moniker.
|
|
|
|
hr = pBindHost->CreateMoniker( LPOLESTR(szURL),NULL, &pmkr,0 );
|
|
|
|
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
// This allows containers to hook the download for
|
|
// doing progress and aborting
|
|
|
|
hr = pBindHost->MonikerBindToStorage(pmkr, NULL, this, IID_IStream,(void**)&pstrm);
|
|
}
|
|
|
|
pBindHost->Release();
|
|
}
|
|
else
|
|
{
|
|
bUseCaller = 0;
|
|
}
|
|
}
|
|
|
|
if( !bUseCaller )
|
|
{
|
|
// If you are here, then either the caller didn't pass
|
|
// a 'caller' pointer or the caller is not in a BindHost
|
|
// friendly environment.
|
|
|
|
hr = ::CreateURLMoniker( 0, szURL, &pmkr );
|
|
|
|
if( SUCCEEDED(hr) )
|
|
hr = ::CreateBindCtx( 0, &pBndCtx );
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
// Register US (not the caller) as the callback. This allows
|
|
// us to hook all notfiications from URL moniker and filter
|
|
// and manipulate to our satifisfaction.
|
|
hr = ::RegisterBindStatusCallback( pBndCtx, this, 0, 0L );
|
|
}
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
hr = pmkr->BindToStorage( pBndCtx, NULL, IID_IStream, (void**)&pstrm );
|
|
|
|
// Smooth out the error code
|
|
if( IS_E_PENDING(hr) )
|
|
hr = S_OK;
|
|
}
|
|
|
|
}
|
|
|
|
if( pstrm )
|
|
pstrm->Release();
|
|
|
|
if( pmkr )
|
|
pmkr->Release();
|
|
|
|
if( pBndCtx )
|
|
pBndCtx->Release();
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// CPullDownload
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
// placeholder for covering the URL moniker anomolies
|
|
|
|
class CPullDownload : public CBaseBSCB
|
|
{
|
|
public:
|
|
CPullDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback );
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
};
|
|
|
|
inline
|
|
CPullDownload::CPullDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback )
|
|
: CBaseBSCB(caller,bscof,callback)
|
|
{
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CPullDownload::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO * pbindinfo
|
|
)
|
|
{
|
|
// pointers are validated in base class
|
|
|
|
HRESULT hr = CBaseBSCB::GetBindInfo( grfBINDF, pbindinfo );
|
|
|
|
if( SUCCEEDED(hr))
|
|
{
|
|
if (*grfBINDF & BINDF_ENFORCERESTRICTED)
|
|
*grfBINDF = BINDF_ASYNCSTORAGE | BINDF_PULLDATA | BINDF_ASYNCHRONOUS | BINDF_ENFORCERESTRICTED;
|
|
else
|
|
*grfBINDF = BINDF_ASYNCSTORAGE | BINDF_PULLDATA | BINDF_ASYNCHRONOUS;
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
//----------------------------------------------------------
|
|
//
|
|
// Push Stream API
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
//
|
|
// Class used for implementing push model downloading when used
|
|
// in combination with the CStream object.
|
|
//
|
|
// The general design for is this class pumps a
|
|
// CBitBucket object with bits and the CStream object makes
|
|
// those bits available to the caller for reading.
|
|
//
|
|
|
|
class CPushDownload : public CBaseBSCB
|
|
{
|
|
public:
|
|
CPushDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback );
|
|
~CPushDownload();
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
protected:
|
|
|
|
// CBaseBSCB
|
|
|
|
virtual void Neutralize();
|
|
|
|
|
|
// IBindStatusCallback
|
|
|
|
STDMETHODIMP OnDataAvailable
|
|
(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC * pFmtetc,
|
|
STGMEDIUM * pstgmed
|
|
) ;
|
|
|
|
private:
|
|
HRESULT CleanupPush();
|
|
};
|
|
|
|
|
|
CPushDownload::CPushDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback )
|
|
: CBaseBSCB(caller,bscof, callback)
|
|
{
|
|
}
|
|
|
|
CPushDownload::~CPushDownload()
|
|
{
|
|
CleanupPush();
|
|
}
|
|
|
|
void
|
|
CPushDownload::Neutralize()
|
|
{
|
|
// We have to do special cleanup.
|
|
|
|
CleanupPush();
|
|
|
|
CBaseBSCB::Neutralize();
|
|
}
|
|
|
|
HRESULT
|
|
CPushDownload::CleanupPush()
|
|
{
|
|
return(NOERROR);
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CPushDownload::OnDataAvailable
|
|
(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC * pFmtetc,
|
|
STGMEDIUM * pstgmed
|
|
)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
|
|
if( SUCCEEDED(hr) && pstgmed->pstm )
|
|
{
|
|
|
|
m_UserStream = pstgmed->pstm;
|
|
|
|
// Add ref again because we are copying and storing the ptr
|
|
|
|
m_UserStream->AddRef();
|
|
|
|
}
|
|
|
|
if( SUCCEEDED(hr) || IS_E_PENDING(hr) )
|
|
hr = CBaseBSCB::OnDataAvailable(grfBSCF,dwSize,pFmtetc,pstgmed);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CPushDownload::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO * pbindinfo
|
|
)
|
|
{
|
|
|
|
HRESULT hr = CBaseBSCB::GetBindInfo( grfBINDF, pbindinfo );
|
|
|
|
// PushDownload can not be ASYNC
|
|
if (*grfBINDF & BINDF_ENFORCERESTRICTED)
|
|
*grfBINDF = BINDF_ENFORCERESTRICTED;
|
|
else
|
|
*grfBINDF = 0;
|
|
|
|
return(hr);
|
|
}
|
|
//----------------------------------------------------------
|
|
//
|
|
// Block Stream API
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
class CBlockDownload : public CPushDownload
|
|
{
|
|
public:
|
|
CBlockDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback );
|
|
~CBlockDownload();
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
HRESULT GetStream( IStream ** ppStream );
|
|
};
|
|
|
|
|
|
inline
|
|
CBlockDownload::CBlockDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback )
|
|
: CPushDownload(caller,bscof, callback)
|
|
{
|
|
|
|
}
|
|
|
|
template <class T> inline
|
|
HRESULT CheckThis( T * AThisPtr )
|
|
{
|
|
CHECK_INTERFACE(AThisPtr);
|
|
return(NOERROR);
|
|
}
|
|
|
|
CBlockDownload::~CBlockDownload()
|
|
{
|
|
CheckThis(this);
|
|
}
|
|
|
|
HRESULT
|
|
CBlockDownload::GetStream( IStream ** ppStream )
|
|
{
|
|
// REMEMBER: If you get this pointer and return it
|
|
// to caller YOU MUST add ref it before handing
|
|
// it back via an API
|
|
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if( m_UserStream )
|
|
{
|
|
*ppStream = m_UserStream;
|
|
hr = S_OK;
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CBlockDownload::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO * pbindinfo
|
|
)
|
|
{
|
|
HRESULT hr = CBaseBSCB::GetBindInfo( grfBINDF, pbindinfo );
|
|
|
|
return(hr);
|
|
}
|
|
//----------------------------------------------------------
|
|
//
|
|
// Download to file
|
|
//
|
|
//----------------------------------------------------------
|
|
|
|
//
|
|
// This class implements the File downloading code. It reads from the
|
|
// stream from urlmon and writes every buffer directly to disk.
|
|
//
|
|
|
|
class CFileDownload : public CBaseBSCB
|
|
{
|
|
public:
|
|
CFileDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback, LPCWSTR szFileName=0);
|
|
~CFileDownload();
|
|
void SetFileName(LPCWSTR);
|
|
|
|
STDMETHODIMP OnDataAvailable(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed);
|
|
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
|
|
STDMETHOD(KickOffDownload)( LPCWSTR szURL );
|
|
|
|
virtual void Neutralize();
|
|
|
|
private:
|
|
HRESULT Cleanup();
|
|
|
|
unsigned char * m_buffer;
|
|
unsigned long m_bufsize;
|
|
HANDLE m_file;
|
|
LPCWSTR m_filename;
|
|
ULONG m_okFromCache;
|
|
|
|
};
|
|
|
|
inline void
|
|
CFileDownload::SetFileName(LPCWSTR newFileName)
|
|
{
|
|
// ASSUMES Calls to this class are synchronous
|
|
|
|
m_filename = newFileName;
|
|
}
|
|
|
|
|
|
CFileDownload::CFileDownload
|
|
(
|
|
IUnknown * caller,
|
|
DWORD bscof,
|
|
LPUOSCALLBACK callback,
|
|
LPCWSTR szFileName
|
|
)
|
|
: CBaseBSCB(caller, bscof, callback)
|
|
{
|
|
m_buffer = 0;
|
|
m_bufsize = 0;
|
|
m_file = INVALID_HANDLE_VALUE;
|
|
m_filename = szFileName;
|
|
|
|
m_okFromCache = 0;
|
|
}
|
|
|
|
CFileDownload::~CFileDownload()
|
|
{
|
|
Cleanup();
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CFileDownload::KickOffDownload( LPCWSTR szURL )
|
|
{
|
|
// MAGIC: registry flag determines whether we
|
|
// nuke this guy from the cache or not
|
|
|
|
GetDLMRegDWord( _TEXT("CacheOk"), &m_okFromCache );
|
|
return( CBaseBSCB::KickOffDownload(szURL) );
|
|
}
|
|
|
|
HRESULT CFileDownload::Cleanup()
|
|
{
|
|
if( m_buffer )
|
|
{
|
|
delete m_buffer;
|
|
m_buffer = 0;
|
|
}
|
|
|
|
if( m_file != INVALID_HANDLE_VALUE )
|
|
{
|
|
CloseHandle(m_file);
|
|
m_file = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
return(NOERROR);
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CFileDownload::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO * pbindinfo
|
|
)
|
|
{
|
|
|
|
HRESULT hr = CBaseBSCB::GetBindInfo( grfBINDF, pbindinfo );
|
|
|
|
if( SUCCEEDED(hr) && !m_okFromCache )
|
|
{
|
|
if (*grfBINDF & BINDF_ENFORCERESTRICTED)
|
|
*grfBINDF = BINDF_ENFORCERESTRICTED | BINDF_PULLDATA;
|
|
else
|
|
*grfBINDF = BINDF_PULLDATA;
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CFileDownload::OnDataAvailable
|
|
(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed
|
|
)
|
|
{
|
|
// Pointers are validated in base class
|
|
|
|
HRESULT hr = CBaseBSCB::OnDataAvailable(grfBSCF,dwSize,pformatetc,pstgmed);
|
|
|
|
if( FAILED(hr) || (dwSize == m_readSoFar) )
|
|
return(hr);
|
|
|
|
if( (m_file == INVALID_HANDLE_VALUE) && dwSize )
|
|
{
|
|
CHECK_POINTER(m_filename);
|
|
|
|
hr = MyCreateFile(m_filename,m_file);
|
|
|
|
if( FAILED(hr) )
|
|
hr = E_ABORT;
|
|
}
|
|
|
|
HANDLE_ABORT(hr);
|
|
|
|
UOSASSERT( (m_file != INVALID_HANDLE_VALUE) );
|
|
|
|
// Only allocate a read buffer if the one we have is not
|
|
// big enough.
|
|
|
|
if( m_buffer && (m_bufsize < (dwSize- m_readSoFar+1)) )
|
|
{
|
|
delete m_buffer;
|
|
m_buffer = 0;
|
|
m_bufsize=0;
|
|
}
|
|
|
|
if( !m_buffer )
|
|
{
|
|
m_bufsize=dwSize- m_readSoFar+1;
|
|
DPRINTF( ("Allocating read buffer %d\n",m_bufsize) );
|
|
m_buffer = new unsigned char [(dwSize- m_readSoFar+1)];
|
|
}
|
|
|
|
CHECK_MEMORY(m_buffer);
|
|
|
|
DWORD dwReadThisMuch = dwSize - m_readSoFar;
|
|
DWORD dwActual = 0;
|
|
DWORD dwCurrentRead = 0;
|
|
|
|
unsigned char * temp = m_buffer;
|
|
do
|
|
{
|
|
hr = m_UserStream->Read(temp,dwReadThisMuch,&dwActual);
|
|
dwCurrentRead += dwActual;
|
|
dwReadThisMuch -= dwActual;
|
|
temp += dwActual;
|
|
|
|
}
|
|
while (!(hr == S_FALSE || hr == E_PENDING) && SUCCEEDED(hr));
|
|
|
|
if( dwCurrentRead )
|
|
{
|
|
m_readSoFar += (dwReadThisMuch = dwCurrentRead);
|
|
|
|
BOOL bWriteOk = ::WriteFile(
|
|
m_file, // HANDLE hFile, // handle to file to write to
|
|
m_buffer, // LPCVOID lpBuffer, // pointer to data to write to file
|
|
dwReadThisMuch, // DWORD nNumberOfBytesToWrite, // number of bytes to write
|
|
&dwActual, // pointer to number of bytes written
|
|
0 ); // LPOVERLAPPED lpOverlapped // addr. of structure needed for overlapped
|
|
|
|
if( !bWriteOk )
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
// PUMPREAD(pstgmed->pstm);
|
|
|
|
if (grfBSCF & BSCF_LASTDATANOTIFICATION)
|
|
{
|
|
CloseHandle(m_file);
|
|
m_file = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
void
|
|
CFileDownload::Neutralize()
|
|
{
|
|
// We have to do special cleanup.
|
|
|
|
Cleanup();
|
|
|
|
CBaseBSCB::Neutralize();
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------
|
|
//
|
|
// Download to Cache file
|
|
// Implementation of the CCacheFileDownload
|
|
//
|
|
//----------------------------------------------------------
|
|
// This class downloads the file to the cache and returns the cache file name
|
|
class CCacheFileDownload : public CBaseBSCB
|
|
{
|
|
public:
|
|
CCacheFileDownload( IUnknown * caller, DWORD bscof, LPUOSCALLBACK callback, LPCWSTR szFileName=0);
|
|
~CCacheFileDownload();
|
|
|
|
STDMETHODIMP OnDataAvailable(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed);
|
|
|
|
STDMETHODIMP GetBindInfo(
|
|
DWORD *grfBINDF,
|
|
BINDINFO *pbindinfo);
|
|
|
|
STDMETHOD(KickOffDownload)( LPCWSTR szURL );
|
|
private:
|
|
DWORD m_readSoFar;
|
|
};
|
|
|
|
CCacheFileDownload::CCacheFileDownload
|
|
(
|
|
IUnknown * caller,
|
|
DWORD bscof,
|
|
LPUOSCALLBACK callback,
|
|
LPCWSTR szFileName
|
|
)
|
|
: CBaseBSCB(caller, bscof, callback)
|
|
{
|
|
m_readSoFar=0;
|
|
}
|
|
|
|
CCacheFileDownload::~CCacheFileDownload()
|
|
{
|
|
// Cleanup();
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CCacheFileDownload::KickOffDownload( LPCWSTR szURL )
|
|
{
|
|
return( CBaseBSCB::KickOffDownload(szURL) );
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CCacheFileDownload::GetBindInfo
|
|
(
|
|
DWORD * grfBINDF,
|
|
BINDINFO * pbindinfo
|
|
)
|
|
{
|
|
HRESULT hr = CBaseBSCB::GetBindInfo( grfBINDF, pbindinfo );
|
|
*grfBINDF &= ~BINDF_ASYNCHRONOUS; //to fix MSN 5.0 bug 103719.
|
|
return(hr);
|
|
}
|
|
|
|
|
|
#define READBLOCKSIZE 8192
|
|
|
|
STDMETHODIMP
|
|
CCacheFileDownload::OnDataAvailable
|
|
(
|
|
DWORD grfBSCF,
|
|
DWORD dwSize,
|
|
FORMATETC *pformatetc,
|
|
STGMEDIUM *pstgmed
|
|
)
|
|
{
|
|
// Pointers are validated in base class
|
|
HRESULT hr = CBaseBSCB::OnDataAvailable(grfBSCF,dwSize,pformatetc,pstgmed);
|
|
|
|
if( FAILED(hr) || (dwSize == m_readSoFar) )
|
|
return(hr);
|
|
|
|
IStream * pstm = pstgmed->pstm;
|
|
if (pstm && dwSize > m_readSoFar)
|
|
{
|
|
DWORD dwToRead = dwSize - m_readSoFar;
|
|
DWORD dwActuallyRead = 1; //initialize to force it into loop
|
|
char* lp = NULL;
|
|
|
|
lp = new char[READBLOCKSIZE+1];
|
|
|
|
CHECK_MEMORY(lp);
|
|
|
|
while (dwActuallyRead)
|
|
{
|
|
dwActuallyRead=0;
|
|
hr = pstm->Read(lp, READBLOCKSIZE, &dwActuallyRead);
|
|
|
|
if(hr!=S_OK && hr!=E_PENDING) // If Read Fails then return Error
|
|
break;
|
|
|
|
m_readSoFar += dwActuallyRead;
|
|
}
|
|
delete lp;
|
|
}
|
|
return (hr);
|
|
}
|
|
|
|
|
|
|
|
STDAPI
|
|
URLOpenPullStreamW
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCWSTR szURL,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
CHECK_POINTER(szURL);
|
|
|
|
HRESULT hr;
|
|
|
|
CPullDownload * download = new CPullDownload(caller,URLOSTRM_NOTIFY_ONDATA,callback);
|
|
|
|
CHECK_POINTER(download);
|
|
|
|
if( !download )
|
|
hr = E_OUTOFMEMORY;
|
|
else
|
|
{
|
|
download->SetEncodingFlags(dwReserved);
|
|
hr = download->KickOffDownload(szURL);
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
STDAPI
|
|
URLDownloadToFileW
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCWSTR szURL,
|
|
LPCWSTR szFileName,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
CHECK_POINTER(szURL);
|
|
HRESULT hr;
|
|
|
|
|
|
CFileDownload * strm = new CFileDownload(caller,URLOSTRM_DONOT_NOTIFY_ONDATA,callback,szFileName);
|
|
|
|
CHECK_POINTER(strm);
|
|
|
|
|
|
if( !strm )
|
|
hr = E_OUTOFMEMORY;
|
|
else
|
|
{
|
|
strm->AddRef(); // So that we have valid handle even after OnStopBinding()
|
|
strm->SetEncodingFlags(dwReserved);
|
|
hr = strm->KickOffDownload(szURL);
|
|
}
|
|
|
|
if (strm)
|
|
strm->Release();
|
|
return(hr);
|
|
}
|
|
|
|
|
|
extern void DoThreadCleanup(BOOL bInThreadDetach);
|
|
extern BOOL g_bNT5OrGreater;
|
|
|
|
STDAPI
|
|
URLDownloadToCacheFileW
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCWSTR szURL,
|
|
LPWSTR szFileName,
|
|
DWORD dwBufLength,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
CHECK_POINTER(szURL);
|
|
|
|
HRESULT hr=S_OK;
|
|
|
|
BOOL fFileURL = FALSE;
|
|
|
|
if (dwBufLength <= 0)
|
|
{
|
|
hr = E_OUTOFMEMORY; // Buffer length invalid
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// 1,2, and 3 are reserved for the constants
|
|
// URLOSTRM_USECACHE, CACHEONLY and GETNEWESTVERSION constants
|
|
// For Compatibility with previous versions
|
|
//
|
|
dwBufLength = (dwBufLength < 4) ? MAX_PATH : dwBufLength;
|
|
szFileName[0] = NULL;
|
|
// For Cache calls
|
|
MAKE_ANSI(szURL);
|
|
|
|
|
|
CCacheFileDownload * strm = new CCacheFileDownload(caller,dwBufLength,callback,szFileName);
|
|
|
|
CHECK_POINTER(strm);
|
|
|
|
if( !strm )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
strm->AddRef(); // So that we have valid handle even after OnStopBinding()
|
|
strm->SetEncodingFlags(dwReserved);
|
|
hr = strm->KickOffDownload(szURL);
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
if (*strm->m_szCacheFileName)
|
|
{
|
|
// If it is a file URL we have to convert it to WIN32 file
|
|
CHAR szPath[MAX_PATH];
|
|
DWORD cchPath = MAX_PATH;
|
|
|
|
if(SUCCEEDED(PathCreateFromUrl(strm->m_szCacheFileName, szPath, &cchPath, 0)))
|
|
{
|
|
fFileURL = TRUE;
|
|
// url should now look like a DOS path
|
|
if (0==MultiByteToWideChar(CP_ACP, 0,
|
|
szPath,-1,
|
|
szFileName, dwBufLength))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
szFileName[0] = NULL;
|
|
}
|
|
else
|
|
hr=S_OK;
|
|
}
|
|
if (!fFileURL)
|
|
{
|
|
if (0==MultiByteToWideChar(CP_ACP, 0, strm->m_szCacheFileName,
|
|
lstrlen(strm->m_szCacheFileName)+1,
|
|
szFileName, dwBufLength))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
szFileName[0] = NULL;
|
|
}
|
|
else
|
|
hr=S_OK;
|
|
}
|
|
}
|
|
else
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
if (strm)
|
|
strm->Release();
|
|
|
|
// WinSE QFE #3411
|
|
// At a minimum, we may need to cleanup the notification hwnd created
|
|
// because this thread can become unresponsive if a broadcast message
|
|
// is sent and the client app isn't pumping messages on this thread.
|
|
// We'll go ahead an do a full tls cleanup.
|
|
|
|
// If this is NOT NT5 or greater, then our notification window is not a message window.
|
|
// Clean up our thread data if no other activity on this thread.
|
|
//
|
|
if (!g_bNT5OrGreater)
|
|
{
|
|
DoThreadCleanup(FALSE);
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
STDAPI
|
|
URLOpenBlockingStreamW
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCWSTR szURL,
|
|
LPSTREAM* ppStream,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
CHECK_POINTER(szURL);
|
|
CHECK_POINTER(ppStream);
|
|
HRESULT hr;
|
|
|
|
if (!ppStream)
|
|
return E_INVALIDARG;
|
|
|
|
CBlockDownload * strm = new CBlockDownload(caller,URLOSTRM_DONOT_NOTIFY_ONDATA,callback);
|
|
|
|
CHECK_POINTER(strm);
|
|
if( !strm )
|
|
hr = E_OUTOFMEMORY;
|
|
else
|
|
{
|
|
strm->AddRef();
|
|
strm->SetEncodingFlags(dwReserved);
|
|
hr = strm->KickOffDownload(szURL);
|
|
}
|
|
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
hr = strm->GetStream(ppStream);
|
|
|
|
// We add ref this pointer because we are handing
|
|
// it back to the user
|
|
|
|
if( SUCCEEDED(hr) )
|
|
(*ppStream)->AddRef();
|
|
}
|
|
|
|
if (strm)
|
|
strm->Release();
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
STDAPI
|
|
URLOpenStreamW
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCWSTR szURL,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
CHECK_POINTER(szURL);
|
|
HRESULT hr;
|
|
|
|
|
|
CPushDownload * strm = new CPushDownload(caller,URLOSTRM_NOTIFY_ONDATA,callback);
|
|
|
|
CHECK_POINTER(strm);
|
|
|
|
if( !strm )
|
|
hr = E_OUTOFMEMORY;
|
|
else
|
|
{
|
|
strm->SetEncodingFlags(dwReserved);
|
|
hr = strm->KickOffDownload(szURL);
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
//
|
|
// ANSI VERSION OF PUBLIC API
|
|
//
|
|
|
|
STDAPI
|
|
URLOpenPullStreamA
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCSTR szURL,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
MAKE_WIDE(szURL);
|
|
|
|
return( URLOpenPullStreamW(caller, WIDE_NAME(szURL), dwReserved, callback) );
|
|
}
|
|
|
|
STDAPI
|
|
URLDownloadToFileA
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCSTR szURL,
|
|
LPCSTR szFileName,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
MAKE_WIDE(szURL);
|
|
MAKE_WIDE(szFileName);
|
|
|
|
return( URLDownloadToFileW( caller, WIDE_NAME(szURL), WIDE_NAME(szFileName),dwReserved, callback ) );
|
|
}
|
|
|
|
STDAPI
|
|
URLDownloadToCacheFileA
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCSTR szURL,
|
|
LPSTR szFileName,
|
|
DWORD dwBufLength,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
HRESULT hr=E_OUTOFMEMORY;
|
|
|
|
if (dwBufLength <= 0)
|
|
return hr;
|
|
|
|
MAKE_WIDE(szURL);
|
|
LPWSTR lpwszfilename= new WCHAR[MAX_PATH];
|
|
|
|
if (lpwszfilename!=NULL)
|
|
{
|
|
hr=URLDownloadToCacheFileW( caller, WIDE_NAME(szURL), lpwszfilename, MAX_PATH, dwReserved, callback );
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// Convert to ANSI.
|
|
dwBufLength = (dwBufLength < 4) ? MAX_PATH : dwBufLength;
|
|
if (0==WideCharToMultiByte( CP_ACP, 0, lpwszfilename, lstrlenW(lpwszfilename)+1,szFileName,
|
|
dwBufLength, NULL, NULL))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
szFileName[0] = NULL;
|
|
}
|
|
}
|
|
else
|
|
szFileName[0] = NULL;
|
|
delete[] lpwszfilename;
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
STDAPI
|
|
URLOpenBlockingStreamA
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCSTR szURL,
|
|
LPSTREAM* ppStream,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
MAKE_WIDE(szURL);
|
|
|
|
return( URLOpenBlockingStreamW(caller,WIDE_NAME(szURL),ppStream,dwReserved,callback) );
|
|
}
|
|
|
|
|
|
STDAPI
|
|
URLOpenStreamA
|
|
(
|
|
LPUNKNOWN caller,
|
|
LPCSTR szURL,
|
|
DWORD dwReserved,
|
|
LPUOSCALLBACK callback
|
|
)
|
|
{
|
|
MAKE_WIDE(szURL);
|
|
|
|
return( URLOpenStreamW(caller,WIDE_NAME(szURL),dwReserved,callback) );
|
|
}
|
|
|
|
|
|
|