|
|
/*++
Copyright (C) 1996-2001 Microsoft Corporation
Module Name:
SAFEARRY.CPP
Abstract:
CSafeArray implementation.
Notes: (1) Support only for arrays with origin at 0 or 1. Can VB deal with a SAFEARRAY of origin zero?
(2) Support only for the following OA types: VT_BSTR, VT_VARIANT, VT_UI1, VT_I2, VT_I4, VT_R8
History:
08-Apr-96 a-raymcc Created. 18-Mar-99 a-dcrews Added out-of-memory exception handling
--*/
#include "precomp.h"
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <WT_safearry.h>
#include <WT_arrtempl.h>
typedef struct { DWORD m_nMaxElementUsed; DWORD m_nFlags; DWORD m_nGrowBy; DWORD m_nStatus; DWORD m_nVarType; SAFEARRAYBOUND m_bound; } PersistHeaderBlock;
//***************************************************************************
//
// CSafeArray::CheckType
//
// Verifies that the constructor is being invoked with a supported type.
//
// PARAMETERS:
// nTest
// One of the supported OLE VT_ constants.
//
//***************************************************************************
void CSafeArray::CheckType(int nTest) { if (nTest != VT_BSTR && nTest != VT_VARIANT && nTest != VT_UI1 && nTest != VT_I2 && nTest != VT_I4 && nTest != VT_R4 && nTest != VT_R8 && nTest != VT_BOOL && nTest != VT_DISPATCH && nTest != VT_UNKNOWN ) Fatal("Caller attempted to use unsupported OLE Automation Type (VT_*)"); }
//***************************************************************************
//
// CSafeArray::CSafeArray
//
// Constructor which creates a new SAFEARRAY.
//
// PARAMETERS:
// vt
// An OLE VT_ type indicator, indicating the element type.
// nFlags
// The destruct policy, either <no_delete> or <auto_delete>. With
// <no_delete>, the underlying SAFEARRAY is not deallocated, whereas
// with <auto_delete> the destructor destroys the SAFEARRAY.
// nSize
// The initial size of the SAFEARRAY.
// nGrowBy
// The amount the SAFEARRAY should grow by when the user attempts to
// add elements to a full array.
//
//***************************************************************************
CSafeArray::CSafeArray( IN int vt, IN int nFlags, IN int nSize, IN int nGrowBy ) { CheckType(vt);
m_nMaxElementUsed = -1; m_nFlags = nFlags; m_nGrowBy = nGrowBy; m_nVarType = vt;
// Allocate the array.
// ===================
m_bound.cElements = nSize; m_bound.lLbound = 0;
m_pArray = SafeArrayCreate(vt, 1, &m_bound);
if (m_pArray == 0) m_nStatus = failed; else m_nStatus = no_error; }
//***************************************************************************
//
// CSafeArray::CSafeArray
//
// Constructor based on an existing SAFEARRAY.
//
// PARAMETERS:
// pSrc
// A pointer to an existing SAFEARRAY which is used as a source
// during object construction.
// nType
// One of the OLE VT_ type indicators.
// nFlags
// OR'ed Bit flags indicating the bind vs. copy, and the
// object destruct policy.
//
// The destruct policy is either <no_delete> or <auto_delete>. With
// <no_delete>, the underlying SAFEARRAY is not deallocated, whereas
// with <auto_delete> the destructor destroys the SAFEARRAY.
//
// Binding is indicated by <bind>, in which case the SAFEARRAY
// pointed to by <pSrc> becomes the internal SAFEARRAY of the
// object. Otherwise, this constructor makes a new copy of the
// SAFEARRAY for internal use.
// nGrowBy
// How much to grow the array by when it fills and the user attempts
// to add more elements. This allows the array to grow in chunks
// so that continuous Add() operations do not operate slowly on
// large arrays.
//
//***************************************************************************
CSafeArray::CSafeArray( IN SAFEARRAY *pSrcCopy, IN int nType, IN int nFlags, IN int nGrowBy ) { m_nStatus = no_error;
CheckType(nType);
// Verify that this is only a 1-dimensional array.
// ===============================================
if (1 != SafeArrayGetDim(pSrcCopy)) m_nStatus = failed;
// Now copy the source or 'bind' the incoming array.
// ====================================================
if (nFlags & bind) m_pArray = pSrcCopy; else if (SafeArrayCopy(pSrcCopy, &m_pArray) != S_OK) m_nStatus = failed;
// Get bound information.
// ======================
LONG uBound = 0; if (S_OK != SafeArrayGetUBound(m_pArray, 1, &uBound)) m_nStatus = failed;
// Correct the Upper Bound into a size.
// ====================================
m_bound.cElements = uBound + 1; m_bound.lLbound = 0; m_nMaxElementUsed = uBound; m_nVarType = nType; m_nGrowBy = nGrowBy; m_nFlags = nFlags & 3; // Mask out the acquire & copy bits.
}
//***************************************************************************
//
// CSafeArray::GetScalarAt
//
// For class internal use. This function returns the element at
// the specified index.
//
// PARAMETERS:
// nIndex
// The index at which to retrieve the scalar.
//
// RETURN VALUE:
// The scalar at the specified the location.
//
//***************************************************************************
SA_ArrayScalar CSafeArray::GetScalarAt(IN int nIndex) { SA_ArrayScalar retval = {0};
// Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return retval;
SafeArrayGetElement(m_pArray, (long *) &nIndex, &retval); return retval; }
//***************************************************************************
//
// CSafeArray assignment operator.
//
//***************************************************************************
CSafeArray& CSafeArray::operator =(IN CSafeArray &Src) { Empty();
m_nMaxElementUsed = Src.m_nMaxElementUsed; m_nFlags = Src.m_nFlags; m_nGrowBy = Src.m_nGrowBy; m_nStatus = Src.m_nStatus; m_nVarType = Src.m_nVarType; m_bound = Src.m_bound;
if (SafeArrayCopy(Src.m_pArray, &m_pArray) != S_OK) m_nStatus = failed;
return *this; }
//***************************************************************************
//
// Copy constructor.
//
// This is implemented primarily via the assignment operator.
//
//***************************************************************************
CSafeArray::CSafeArray(CSafeArray &Src) { m_nMaxElementUsed = 0; m_nFlags = 0; m_nGrowBy = 0; m_nStatus = no_error; m_nVarType = VT_NULL; m_pArray = 0; m_bound.cElements = 0; m_bound.lLbound = 0;
*this = Src; }
//***************************************************************************
//
// CSafeArray::Add
//
// Adds the BSTR to the array, growing the array if required.
//
// PARAMETERS:
// Src
// The source BSTR to add to the array. If NULL, then a
// blank string is added by the underlying SAFEARRAY implementation.
// (there is no way to prevent this). This can point to
// an LPWSTR as well.
//
// RETURN VALUE:
// <no_error> or <failed>.
//
//***************************************************************************
int CSafeArray::AddBSTR(IN BSTR Src) { // If there is no more room in the array, then expand it.
// ======================================================
if (m_nMaxElementUsed == (int) m_bound.cElements - 1) {
if (m_nGrowBy == 0) return range_error;
m_bound.cElements += m_nGrowBy;
if (S_OK != SafeArrayRedim(m_pArray, &m_bound)) m_nStatus = failed; }
m_nMaxElementUsed++;
BSTR Copy = SysAllocString(Src); CSysFreeMe auto1(Copy);
if (SafeArrayPutElement(m_pArray, (long *) &m_nMaxElementUsed, Copy) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::AddVariant
//
// Adds the specified VARIANT to the array.
//
// PARAMETERS:
// pSrc
// A pointer to the source VARIANT, which is copied.
//
// RETURN VALUE:
// range_error, failed, no_error
//
//***************************************************************************
int CSafeArray::AddVariant(IN VARIANT *pSrc) { // If there is no more room in the array, then expand it.
// ======================================================
if (m_nMaxElementUsed == (int) m_bound.cElements - 1) {
if (m_nGrowBy == 0) return range_error; m_bound.cElements += m_nGrowBy;
if (S_OK != SafeArrayRedim(m_pArray, &m_bound)) m_nStatus = failed; }
m_nMaxElementUsed++;
if (SafeArrayPutElement(m_pArray, (long *) &m_nMaxElementUsed, pSrc) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::AddDispatch
//
// Adds the specified IDispatch* to the array.
//
// PARAMETERS:
// pSrc
// A pointer to the source IDispatch*, which is AddRefed.
//
// RETURN VALUE:
// range_error, failed, no_error
//
//***************************************************************************
int CSafeArray::AddDispatch(IN IDispatch *pDisp) { // If there is no more room in the array, then expand it.
// ======================================================
if (m_nMaxElementUsed == (int) m_bound.cElements - 1) {
if (m_nGrowBy == 0) return range_error; m_bound.cElements += m_nGrowBy;
if (S_OK != SafeArrayRedim(m_pArray, &m_bound)) m_nStatus = failed; }
m_nMaxElementUsed++;
if (SafeArrayPutElement(m_pArray, (long *) &m_nMaxElementUsed, pDisp) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::AddUnknown
//
// Adds the specified IUnknown* to the array.
//
// PARAMETERS:
// pSrc
// A pointer to the source IUnknown*, which is AddRefed.
//
// RETURN VALUE:
// range_error, failed, no_error
//
//***************************************************************************
int CSafeArray::AddUnknown(IN IUnknown *pUnk) { // If there is no more room in the array, then expand it.
// ======================================================
if (m_nMaxElementUsed == (int) m_bound.cElements - 1) {
if (m_nGrowBy == 0) return range_error; m_bound.cElements += m_nGrowBy;
if (S_OK != SafeArrayRedim(m_pArray, &m_bound)) m_nStatus = failed; }
m_nMaxElementUsed++;
if (SafeArrayPutElement(m_pArray, (long *) &m_nMaxElementUsed, pUnk) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::GetBSTRAt
//
// If the array type is VT_BSTR, this returns the string at the specified
// index.
//
// PARAMETERS:
// nIndex
// The array index for which the string is requried.
//
// RETURN VALUE:
// A dynamically allocated BSTR which must be freed with SysFreeString.
// NULL is returned on error. If NULL was originally added at this
// location, a string with zero length will be returned, which still
// must be freed with SysFreeString.
//
//***************************************************************************
BSTR CSafeArray::GetBSTRAt(int nIndex) { BSTR StrPtr = 0;
if (nIndex >= (int) m_bound.cElements) return NULL;
if (S_OK != SafeArrayGetElement(m_pArray, (long *) &nIndex, &StrPtr)) return NULL;
return StrPtr; }
//***************************************************************************
//
// CSafeArray::GetVariantAt
//
// PARAMETERS:
// nIndex
// The array index from which to retrieve the VARIANT.
//
// RETURN VALUE:
// Returns a new VARIANT at the specified location. The receiver must
// call VariantClear() on this VARIANT when it is no longer used.
//
//***************************************************************************
VARIANT CSafeArray::GetVariantAt(int nIndex) { VARIANT Var; VariantInit(&Var);
if (nIndex >= (int) m_bound.cElements) return Var;
if (S_OK != SafeArrayGetElement(m_pArray, (long *) &nIndex, &Var)) return Var;
return Var; }
//***************************************************************************
//
// CSafeArray::GetDispatchAt
//
// PARAMETERS:
// nIndex
// The array index from which to retrieve the IDispatch*.
//
// RETURN VALUE:
// Returns the IDispatch* at the specified location. The receiver must
// call Release on this pointer (if not NULL) when it is no longer used.
//
//***************************************************************************
IDispatch* CSafeArray::GetDispatchAt(int nIndex) { IDispatch* pDisp; if (nIndex >= (int) m_bound.cElements) return NULL;
if (S_OK != SafeArrayGetElement(m_pArray, (long *) &nIndex, &pDisp)) return NULL;
return pDisp; }
//***************************************************************************
//
// CSafeArray::GetUnknownAt
//
// PARAMETERS:
// nIndex
// The array index from which to retrieve the IUnknown*.
//
// RETURN VALUE:
// Returns the IUnknown* at the specified location. The receiver must
// call Release on this pointer (if not NULL) when it is no longer used.
//
//***************************************************************************
IUnknown* CSafeArray::GetUnknownAt(int nIndex) { IUnknown* pUnk; if (nIndex >= (int) m_bound.cElements) return NULL;
if (S_OK != SafeArrayGetElement(m_pArray, (long *) &nIndex, &pUnk)) return NULL;
return pUnk; }
//***************************************************************************
//
// CSafeArray::SetAt
//
// Replaces the BSTR value at the specified array index. The original
// BSTR value is automatically deallocated and replaced by the new value.
// You can only call this to replace an existing element or to add a
// new element to the end (one position past the last element). If the
// array size is 10, you can call this with 0..10, but not 11 or higher.
//
// PARAMETERS:
// nIndex
// The position at which to replace the element.
// Str
// The new string.
// nFlags
// If <acquire> this function acquires ownership of the string and
// can delete it. Otherwise, the caller retains ownership of the
// string.
//
// RETURN VALUE:
// no_error
// range_error
// failed
//
//***************************************************************************
int CSafeArray::SetBSTRAt( IN int nIndex, IN BSTR Str ) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Check to see if we are adding a new element.
// ============================================
if (nIndex == m_nMaxElementUsed + 1) return AddBSTR(Str);
BSTR Copy = SysAllocString(Str); CSysFreeMe auto1(Copy);
// If here, we are replacing an element.
// =====================================
if (SafeArrayPutElement(m_pArray, (long *) &nIndex, Copy) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::SetVariantAt
//
// Sets the VARIANT at the specified index.
//
// PARAMETERS:
// nIndex
// The index at which to set the VARIANT. The original contents
// at this location are automatically deallocated and replaced.
// pVal
// Used as the source of the new value. This is treated as read-only.
//
// RETURN VALUE:
// no_error, failed, range_error
//
//***************************************************************************
int CSafeArray::SetVariantAt( IN int nIndex, IN VARIANT *pVal ) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Check to see if we are adding a new element.
// ============================================
if (nIndex == m_nMaxElementUsed + 1) return AddVariant(pVal);
// If here, we are replacing an element.
// =====================================
if (SafeArrayPutElement(m_pArray, (long *) &nIndex, pVal) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::SetDispatchAt
//
// Sets the IDispatch* at the specified index.
//
// PARAMETERS:
// nIndex
// The index at which to set the IDispatch*. The original contents
// at this location are automatically Released and replaced.
// pVal
// Used as the source of the new value. This is treated as read-only.
//
// RETURN VALUE:
// no_error, failed, range_error
//
//***************************************************************************
int CSafeArray::SetDispatchAt( IN int nIndex, IN IDispatch *pDisp ) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Check to see if we are adding a new element.
// ============================================
if (nIndex == m_nMaxElementUsed + 1) return AddDispatch(pDisp);
// If here, we are replacing an element.
// =====================================
if (SafeArrayPutElement(m_pArray, (long *) &nIndex, pDisp) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::SetUnknownAt
//
// Sets the IUnknown* at the specified index.
//
// PARAMETERS:
// nIndex
// The index at which to set the IUnknown*. The original contents
// at this location are automatically Released and replaced.
// pVal
// Used as the source of the new value. This is treated as read-only.
//
// RETURN VALUE:
// no_error, failed, range_error
//
//***************************************************************************
int CSafeArray::SetUnknownAt( IN int nIndex, IN IUnknown *pUnk ) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Check to see if we are adding a new element.
// ============================================
if (nIndex == m_nMaxElementUsed + 1) return AddUnknown(pUnk);
// If here, we are replacing an element.
// =====================================
if (SafeArrayPutElement(m_pArray, (long *) &nIndex, pUnk) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::RemoveAt
//
// Removes the element at the specified index. After a series of these
// operations, the caller should call the Trim() function.
//
// PARAMETERS:
// nIndex
// The target index for element removal.
//
// RETURN VALUE:
// no_error, range_error
//
//***************************************************************************
int CSafeArray::RemoveAt(IN int nIndex) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Copy element n+1 into n.
// ========================
BSTR strVal; VARIANT v; SA_ArrayScalar scalar; IDispatch* pDisp; IUnknown* pUnk;
for (long i = nIndex; i < m_nMaxElementUsed; i++) { long nNext = i + 1;
if (m_nVarType == VT_BSTR) { SafeArrayGetElement(m_pArray, &nNext, &strVal); SafeArrayPutElement(m_pArray, &i, strVal); SysFreeString(strVal); } else if (m_nVarType == VT_VARIANT) { SafeArrayGetElement(m_pArray, &nNext, &v); SafeArrayPutElement(m_pArray, &i, &v); VariantClear(&v); } else if (m_nVarType == VT_DISPATCH) { SafeArrayGetElement(m_pArray, &nNext, &pDisp); SafeArrayPutElement(m_pArray, &i, pDisp); if(pDisp) pDisp->Release(); } else if (m_nVarType == VT_UNKNOWN) { SafeArrayGetElement(m_pArray, &nNext, &pUnk); SafeArrayPutElement(m_pArray, &i, pUnk); if(pUnk) pUnk->Release(); } else { SafeArrayGetElement(m_pArray, &nNext, &scalar); SafeArrayPutElement(m_pArray, &i, &scalar); } }
m_nMaxElementUsed--; return no_error; }
//***************************************************************************
//
// CSafeArray::SetScalarAt
//
// For class internal use. Sets the scalar type at the specified index.
//
// PARAMETERS:
// nIndex
// The target index.
// val
// The new value.
//
// RETURN VALUES:
// range_error, failed, no_error
//
//***************************************************************************
int CSafeArray::SetScalarAt(IN int nIndex, IN SA_ArrayScalar val) { // Check for out-of-range condition.
// =================================
if (nIndex > m_nMaxElementUsed + 1) return range_error;
// Check to see if we are adding a new element.
// ============================================
if (nIndex == m_nMaxElementUsed + 1) return AddScalar(val);
// If here, we are replacing an element.
// =====================================
if (SafeArrayPutElement(m_pArray, (long *) &nIndex, &val) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::AddScalar
//
// For class internal use only.
//
// Adds a new scalar to the 'end' of the array, growing it if required
// and if possible.
//
// PARAMETERS:
// val
// The new value.
//
// RETURN VALUE:
// no_error, range_error, failed
//
//***************************************************************************
int CSafeArray::AddScalar(IN SA_ArrayScalar val) { // If there is no more room in the array, then expand it.
// ======================================================
if (m_nMaxElementUsed == (int) m_bound.cElements - 1) {
if (m_nGrowBy == 0) return range_error;
m_bound.cElements += m_nGrowBy;
if (S_OK != SafeArrayRedim(m_pArray, &m_bound)) m_nStatus = failed; }
m_nMaxElementUsed++;
if (SafeArrayPutElement(m_pArray, (long *) &m_nMaxElementUsed, &val) != S_OK) { m_nStatus = failed; return failed; }
return no_error; }
//***************************************************************************
//
// CSafeArray::Empty
//
// Empties the SAFEARRAY.
//
//***************************************************************************
void CSafeArray::Empty() { m_nMaxElementUsed = 0; m_nFlags = 0; m_nGrowBy = 0; m_nStatus = no_error; m_nVarType = VT_NULL; if (m_pArray) SafeArrayDestroy(m_pArray); m_pArray = 0; m_bound.cElements = 0; m_bound.lLbound = 0; }
//***************************************************************************
//
// CSafeArray::GetArrayCopy
//
// RETURN VALUE:
// A copy of the internal SAFEARRAY or NULL on error.
//
//***************************************************************************
SAFEARRAY *CSafeArray::GetArrayCopy() { SAFEARRAY *pCopy = 0; if (SafeArrayCopy(m_pArray, &pCopy) != S_OK) return 0; return pCopy; }
//***************************************************************************
//
// CSafeArray destructor.
//
// If the internal flags are set to auto_delete, then the internal
// SAFEARRAY is destroyed during destruction.
//
//***************************************************************************
CSafeArray::~CSafeArray() { if (m_nFlags == auto_delete) SafeArrayDestroy(m_pArray); }
//***************************************************************************
//
// CSafeArray::Trim
//
//***************************************************************************
int CSafeArray::Trim() { m_bound.cElements = m_nMaxElementUsed + 1;
// HACK for NT 3.51: may not redimention to size 0
// ===============================================
if(m_bound.cElements == 0) { SafeArrayDestroy(m_pArray); m_pArray = SafeArrayCreate(m_nVarType, 1, &m_bound); } else { SafeArrayRedim(m_pArray, &m_bound); }
return no_error; }
//***************************************************************************
//
//***************************************************************************
void CSafeArray::Fatal(const char *pszMsg) { // MessageBox(0, pszMsg, "CSafeArray FATAL Error",
// MB_OK | MB_SYSTEMMODAL | MB_ICONEXCLAMATION);
}
|