windows-nt/Source/XPSP1/NT/admin/pchealth/helpctr/shell/misc/scriptablestream.cpp
2020-09-26 16:20:57 +08:00

336 lines
6.9 KiB
C++

/******************************************************************************
Copyright (c) 2000 Microsoft Corporation
Module Name:
ScriptableStream.cpp
Abstract:
This file contains the implementation of the CPCHScriptableStream class,
which is a scriptable wrapper for IStream.
Revision History:
Davide Massarenti (Dmassare) 10/06/2000
created
******************************************************************************/
#include "stdafx.h"
////////////////////////////////////////////////////////////////////////////////
HRESULT CPCHScriptableStream::ReadToHGLOBAL( /*[in]*/ long lCount, /*[out]*/ HGLOBAL& hg, /*[out]*/ ULONG& lReadTotal )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::ReadToHGLOBAL" );
HRESULT hr;
lReadTotal = 0;
if(lCount < 0)
{
static const ULONG c_BUFSIZE = 4096;
ULONG lRead;
__MPC_EXIT_IF_ALLOC_FAILS(hr, hg, ::GlobalAlloc( GMEM_FIXED, c_BUFSIZE ));
while(1)
{
HGLOBAL hg2;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Read( &((BYTE*)hg)[lReadTotal], c_BUFSIZE, &lRead ));
if(hr == S_FALSE || lRead == 0) break;
lReadTotal += lRead;
//
// Increase buffer.
//
__MPC_EXIT_IF_ALLOC_FAILS(hr, hg2, ::GlobalReAlloc( hg, lReadTotal + c_BUFSIZE, 0 ));
hg = hg2;
}
}
else
{
__MPC_EXIT_IF_ALLOC_FAILS(hr, hg, ::GlobalAlloc( GMEM_FIXED, lCount ));
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Read( hg, lCount, &lReadTotal ));
}
hr = S_OK;
__HCP_FUNC_CLEANUP;
__HCP_FUNC_EXIT(hr);
}
/////////////////////////////////////////////////////////////////////////////
STDMETHODIMP CPCHScriptableStream::get_Size( /*[out, retval]*/ long *plSize )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::get_Size" );
HRESULT hr;
STATSTG stat;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_POINTER_AND_SET(plSize,0);
__MPC_PARAMCHECK_END();
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Stat( &stat, STATFLAG_NONAME ));
*plSize = (long)stat.cbSize.QuadPart;
hr = S_OK;
__HCP_FUNC_CLEANUP;
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::Read( /*[in]*/ long lCount, /*[out, retval]*/ VARIANT *pvData )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::Read" );
HRESULT hr;
CComVariant vArray;
HGLOBAL hg = NULL;
ULONG lReadTotal;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_NOTNULL(pvData);
__MPC_PARAMCHECK_END();
__MPC_EXIT_IF_METHOD_FAILS(hr, ReadToHGLOBAL( lCount, hg, lReadTotal ));
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::ConvertBufferToVariant( (BYTE*)hg, (DWORD)lReadTotal, vArray ));
__MPC_EXIT_IF_METHOD_FAILS(hr, vArray.Detach( pvData ));
hr = S_OK;
__HCP_FUNC_CLEANUP;
if(hg) ::GlobalFree( hg );
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::ReadHex( /*[in]*/ long lCount, /*[out, retval]*/ BSTR *pbstrData )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::ReadHex" );
HRESULT hr;
HGLOBAL hg = NULL;
ULONG lReadTotal;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_POINTER_AND_SET(pbstrData,0);
__MPC_PARAMCHECK_END();
__MPC_EXIT_IF_METHOD_FAILS(hr, ReadToHGLOBAL( lCount, hg, lReadTotal ));
if(lReadTotal)
{
CComBSTR bstrHex;
HGLOBAL hg2;
//
// Trim down the size of the HGLOBAL.
//
__MPC_EXIT_IF_ALLOC_FAILS(hr, hg2, ::GlobalReAlloc( hg, lReadTotal, 0 ));
hg = hg2;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::ConvertHGlobalToHex( hg, bstrHex ));
*pbstrData = bstrHex.Detach();
}
hr = S_OK;
__HCP_FUNC_CLEANUP;
if(hg) ::GlobalFree( hg );
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::Write( /*[in]*/ long lCount, /*[in]*/ VARIANT vData, /*[out, retval]*/ long *plWritten )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::get_Size" );
HRESULT hr;
BYTE* rgBuf = NULL;
DWORD dwLen = 0;
bool fAccessData = false;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_POINTER_AND_SET(plWritten,0);
__MPC_PARAMCHECK_END();
switch(vData.vt)
{
case VT_EMPTY:
case VT_NULL:
break;
case VT_ARRAY | VT_UI1:
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::ConvertVariantToBuffer( &vData, rgBuf, dwLen ));
break;
case VT_ARRAY | VT_VARIANT:
{
long lBound; ::SafeArrayGetLBound( vData.parray, 1, &lBound );
long uBound; ::SafeArrayGetUBound( vData.parray, 1, &uBound );
VARIANT* pSrc;
DWORD dwPos;
dwLen = uBound - lBound + 1;
__MPC_EXIT_IF_ALLOC_FAILS(hr, rgBuf, new BYTE[dwLen]);
__MPC_EXIT_IF_METHOD_FAILS(hr, ::SafeArrayAccessData( vData.parray, (LPVOID*)&pSrc ));
fAccessData = true;
for(dwPos=0; dwPos<dwLen; dwPos++, pSrc++)
{
CComVariant v;
__MPC_EXIT_IF_METHOD_FAILS(hr, ::VariantChangeType( &v, pSrc, 0, VT_UI1 ));
rgBuf[dwPos] = v.bVal;
}
}
break;
}
if(dwLen && rgBuf)
{
ULONG lWritten;
//
// Just write the requested number of bytes.
//
if(lCount >= 0 && dwLen > lCount) dwLen = lCount;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Write( rgBuf, dwLen, &lWritten ));
*plWritten = lWritten;
}
hr = S_OK;
__HCP_FUNC_CLEANUP;
if(fAccessData) ::SafeArrayUnaccessData( vData.parray );
delete [] rgBuf;
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::WriteHex( /*[in]*/ long lCount, /*[in]*/ BSTR bstrData, /*[out, retval]*/ long *plWritten )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::get_Size" );
HRESULT hr;
HGLOBAL hg = NULL;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_POINTER_AND_SET(plWritten,0);
__MPC_PARAMCHECK_END();
if(STRINGISPRESENT(bstrData))
{
CComBSTR bstrHex( bstrData );
ULONG lWritten = 0;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::ConvertHexToHGlobal( bstrHex, hg, true ));
if(hg)
{
DWORD dwSize = ::GlobalSize( hg );
//
// Just write the requested number of bytes.
//
if(lCount >= 0 && lCount > dwSize) lCount = dwSize;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Write( (BYTE*)hg, lCount, &lWritten ));
}
*plWritten = lWritten;
}
hr = S_OK;
__HCP_FUNC_CLEANUP;
if(hg) ::GlobalFree( hg );
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::Seek( /*[in]*/ long lOffset, /*[in]*/ BSTR bstrOrigin, /*[out, retval]*/ long *plNewPos )
{
__HCP_FUNC_ENTRY( "CPCHScriptableStream::get_Size" );
HRESULT hr;
DWORD dwOrigin;
LARGE_INTEGER liMove;
ULARGE_INTEGER liNewPos;
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_STRING_NOT_EMPTY(bstrOrigin);
__MPC_PARAMCHECK_POINTER_AND_SET(plNewPos,0);
__MPC_PARAMCHECK_END();
if (!_wcsicmp( bstrOrigin, L"SET" )) dwOrigin = STREAM_SEEK_SET;
else if(!_wcsicmp( bstrOrigin, L"CUR" )) dwOrigin = STREAM_SEEK_CUR;
else if(!_wcsicmp( bstrOrigin, L"END" )) dwOrigin = STREAM_SEEK_END;
else __MPC_SET_ERROR_AND_EXIT(hr, E_INVALIDARG);
liMove.QuadPart = lOffset;
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::FileStream::Seek( liMove, dwOrigin, &liNewPos ));
*plNewPos = liNewPos.QuadPart;
hr = S_OK;
__HCP_FUNC_CLEANUP;
__HCP_FUNC_EXIT(hr);
}
STDMETHODIMP CPCHScriptableStream::Close()
{
MPC::FileStream::Close();
return S_OK;
}