mirror of https://github.com/tongzx/nt5src
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.
1541 lines
32 KiB
1541 lines
32 KiB
/***********************************************************************
|
|
*
|
|
* ABCTBL3.C
|
|
*
|
|
* Contents Table - Part 3.
|
|
*
|
|
*
|
|
* The following routines are implemented in this file.
|
|
*
|
|
*
|
|
* IVTABC_QueryInterface
|
|
* IVTABC_Release
|
|
* IVTABC_SortTable
|
|
* IVTABC_QuerySortOrder
|
|
* IVTABC_CreateBookmark
|
|
* IVTABC_FreeBookmark
|
|
* IVTABC_ExpandRow
|
|
* IVTABC_ColapseRow
|
|
* IVTABC_WaitForCompletion
|
|
* IVTABC_Abort
|
|
* IVTABC_Advise
|
|
* IVTABC_Unadvise
|
|
* IVTABC_GetStatus
|
|
* IVTABC_SetColumns
|
|
* IVTABC_QueryColumns
|
|
* IVTABC_GetCollapseState,
|
|
* IVTABC_SetCollapseState,
|
|
*
|
|
* Copyright 1992, 1993, 1994 Microsoft Corporation. All Rights Reserved.
|
|
*
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
#include "faxab.h"
|
|
|
|
/*
|
|
* Default sort order set
|
|
*/
|
|
static const SizedSSortOrderSet(1, sosIVTABC) =
|
|
{
|
|
1,
|
|
0,
|
|
0,
|
|
{
|
|
{
|
|
PR_DISPLAY_NAME, TABLE_SORT_ASCEND
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
*
|
|
*
|
|
- AVTABC_QueryInterface
|
|
-
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_QueryInterface( LPIVTABC lpIVTAbc,
|
|
REFIID lpiid,
|
|
LPVOID FAR * lppNewObj
|
|
)
|
|
{
|
|
|
|
HRESULT hResult = hrSuccess;
|
|
|
|
/* Minimally validate the lpIVTAbc parameter */
|
|
|
|
/*
|
|
* Check to see if it's big enough to be this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not big enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/* Check other parameters */
|
|
|
|
if ( IsBadReadPtr( lpiid, (UINT) SIZEOF(IID)) ||
|
|
IsBadWritePtr( lppNewObj, (UINT) SIZEOF(LPVOID))
|
|
)
|
|
{
|
|
DebugTraceSc(IVTABC_QueryInterface, E_INVALIDARG);
|
|
return ResultFromScode(E_INVALIDARG);
|
|
}
|
|
|
|
/* See if the requested interface is one of ours */
|
|
|
|
if (memcmp(lpiid, &IID_IUnknown, SIZEOF(IID)) &&
|
|
memcmp(lpiid, &IID_IMAPITable, SIZEOF(IID)))
|
|
{
|
|
*lppNewObj = NULL; /* OLE requires zeroing the [out] parameter */
|
|
DebugTraceSc(IVTABC_QueryInterface, E_NOINTERFACE);
|
|
return ResultFromScode(E_NOINTERFACE);
|
|
}
|
|
|
|
/* We'll do this one. Bump the usage count and return a new pointer. */
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
++lpIVTAbc->lcInit;
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
*lppNewObj = lpIVTAbc;
|
|
|
|
out:
|
|
DebugTraceResult(IVTABC_QueryInterface,hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_Release
|
|
-
|
|
*
|
|
* Decrement the reference count on this object and free it if
|
|
* the reference count is zero.
|
|
* Returns the reference count.
|
|
*/
|
|
|
|
STDMETHODIMP_(ULONG)
|
|
IVTABC_Release(LPIVTABC lpIVTAbc)
|
|
{
|
|
ULONG ulBK;
|
|
long lcInit;
|
|
|
|
/*
|
|
* Check to see if it's big enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
return 1;
|
|
}
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
lcInit = --lpIVTAbc->lcInit;
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
if (lcInit == 0)
|
|
{
|
|
/*
|
|
* Free up the current column set
|
|
*/
|
|
if (lpIVTAbc->lpPTAColSet != ptagaivtabcColSet)
|
|
{
|
|
lpIVTAbc->lpFreeBuff (lpIVTAbc->lpPTAColSet);
|
|
}
|
|
|
|
/*
|
|
* Close up the file
|
|
*/
|
|
if (lpIVTAbc->hFile != INVALID_HANDLE_VALUE)
|
|
{
|
|
CloseHandle(lpIVTAbc->hFile);
|
|
lpIVTAbc->hFile = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
/*
|
|
* Free up the file name
|
|
*/
|
|
lpIVTAbc->lpFreeBuff(lpIVTAbc->lpszFileName);
|
|
|
|
/*
|
|
* Rip through the bookmarks and free up any that are there
|
|
*/
|
|
for (ulBK = 0; ulBK < MAX_BOOKMARKS; ulBK++)
|
|
if (lpIVTAbc->rglpABCBK[ulBK])
|
|
{
|
|
(*(lpIVTAbc->lpFreeBuff)) (lpIVTAbc->rglpABCBK[ulBK]);
|
|
lpIVTAbc->rglpABCBK[ulBK] = NULL;
|
|
}
|
|
|
|
/*
|
|
* Free up the ANR stuff, if used
|
|
*/
|
|
lpIVTAbc->lpFreeBuff (lpIVTAbc->lpszPartialName);
|
|
|
|
FreeANRBitmaps(lpIVTAbc);
|
|
|
|
/*
|
|
* Free up the advise list, if used
|
|
*/
|
|
if (lpIVTAbc->parglpAdvise)
|
|
lpIVTAbc->lpMalloc->lpVtbl->Free(lpIVTAbc->lpMalloc, lpIVTAbc->parglpAdvise);
|
|
|
|
/* Delete critical section for this object */
|
|
DeleteCriticalSection(&lpIVTAbc->cs);
|
|
|
|
/* Deregister the idle routine */
|
|
|
|
DeregisterIdleRoutine(lpIVTAbc->ftg);
|
|
|
|
/*
|
|
* Set the vtbl to NULL. This way the client will find out
|
|
* real fast if it's calling a method on a released object. That is,
|
|
* the client will crash. Hopefully, this will happen during the
|
|
* development stage of the client.
|
|
*/
|
|
lpIVTAbc->lpVtbl = NULL;
|
|
|
|
/*
|
|
* Need to free the object
|
|
*/
|
|
|
|
lpIVTAbc->lpFreeBuff(lpIVTAbc);
|
|
return 0;
|
|
}
|
|
|
|
return lcInit;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
- IVTABC_SortTable
|
|
-
|
|
* The Microsoft At Work Fax Address Book does not resort it's views.
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_SortTable( LPIVTABC lpIVTAbc,
|
|
LPSSortOrderSet lpSortCriteria,
|
|
ULONG ulFlags
|
|
)
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's big enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check for bad sort order set. This is from the mapi utilities DLL.
|
|
*/
|
|
if (FBadSortOrderSet(lpSortCriteria))
|
|
{
|
|
hResult = ResultFromScode(MAPI_E_UNKNOWN_FLAGS);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check flags
|
|
*/
|
|
if (ulFlags & ~(TBL_ASYNC|TBL_BATCH))
|
|
{
|
|
hResult = ResultFromScode(MAPI_E_UNKNOWN_FLAGS);
|
|
goto out;
|
|
}
|
|
|
|
|
|
/*
|
|
* We don't support sorting this table
|
|
*/
|
|
hResult = ResultFromScode(MAPI_E_NO_SUPPORT);
|
|
|
|
out:
|
|
DebugTraceResult(IVTABC_SortTable, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
/*
|
|
- IVTABC_QuerySortOrder
|
|
-
|
|
*
|
|
* For this implementation there is only one sort order
|
|
*/
|
|
|
|
STDMETHODIMP
|
|
IVTABC_QuerySortOrder( LPIVTABC lpIVTAbc,
|
|
LPSSortOrderSet * lppSortCriteria
|
|
)
|
|
{
|
|
SCODE scode;
|
|
HRESULT hResult = hrSuccess;
|
|
int cbSize;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check the out parameter for writability
|
|
*/
|
|
if (IsBadWritePtr(lppSortCriteria, sizeof(LPSSortOrderSet)))
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/* Calculate size of the structure we're gonna copy */
|
|
cbSize = CbNewSSortOrderSet((int)sosIVTABC.cSorts);
|
|
|
|
scode = lpIVTAbc->lpAllocBuff(cbSize, (LPVOID *) lppSortCriteria);
|
|
if (FAILED(scode))
|
|
{
|
|
hResult = ResultFromScode(scode);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Copy the column set in
|
|
*/
|
|
if (cbSize)
|
|
memcpy(*lppSortCriteria, &sosIVTABC, cbSize);
|
|
|
|
out:
|
|
|
|
DebugTraceResult(IVTABC_QuerySortOrder, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
- IVTABC_CreateBookmark
|
|
-
|
|
* Creates a bookmark associated with a row in a table
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_CreateBookmark( LPIVTABC lpIVTAbc,
|
|
BOOKMARK * lpbkPosition
|
|
)
|
|
{
|
|
SCODE scode;
|
|
HRESULT hResult = hrSuccess;
|
|
ULONG ulBK;
|
|
LPABCBK lpABCBK = NULL;
|
|
ULONG cbRead = 0;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_CreateBookmark, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_CreateBookmark, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check the out parameter for writability
|
|
*/
|
|
if (IsBadWritePtr(lpbkPosition, SIZEOF(BOOKMARK)))
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_CreateBookmark, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
|
|
/*
|
|
* Open the file
|
|
*/
|
|
hResult = HrOpenFile(lpIVTAbc);
|
|
if (HR_FAILED(hResult))
|
|
{
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Shortcuts first
|
|
*/
|
|
if (lpIVTAbc->ulPosition == lpIVTAbc->ulMaxPos)
|
|
{
|
|
*lpbkPosition = BOOKMARK_END;
|
|
return hrSuccess;
|
|
}
|
|
|
|
/*
|
|
* search for a blank bookmark
|
|
*/
|
|
for (ulBK = 0; lpIVTAbc->rglpABCBK[ulBK] && ulBK < MAX_BOOKMARKS; ulBK++);
|
|
|
|
/* did we find any?? */
|
|
if (ulBK == MAX_BOOKMARKS)
|
|
{
|
|
hResult = ResultFromScode(MAPI_E_UNABLE_TO_COMPLETE);
|
|
|
|
goto out;
|
|
}
|
|
|
|
|
|
scode = lpIVTAbc->lpAllocBuff (SIZEOF(ABCBK),(LPVOID *) &lpABCBK);
|
|
if (FAILED(scode))
|
|
{
|
|
hResult = ResultFromScode(scode);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Fill in new bookmark
|
|
*/
|
|
lpABCBK->filetime = lpIVTAbc->filetime;
|
|
lpABCBK->ulPosition = lpIVTAbc->ulPosition;
|
|
|
|
/* Seek to position in file */
|
|
(void) SetFilePointer(lpIVTAbc->hFile, lpABCBK->ulPosition, NULL, FILE_BEGIN);
|
|
|
|
/* Read in the record at that location */
|
|
if (!ReadFile(lpIVTAbc->hFile,
|
|
(LPVOID) &(lpABCBK->abcrec), SIZEOF(ABCREC), &cbRead, NULL))
|
|
{
|
|
goto readerror;
|
|
}
|
|
/* Second check */
|
|
if (cbRead != sizeof(ABCREC))
|
|
{
|
|
goto readerror;
|
|
}
|
|
|
|
/*
|
|
* Put this in the bookmark structure
|
|
*/
|
|
lpIVTAbc->rglpABCBK[ulBK] = lpABCBK;
|
|
|
|
/* Return the bookmark */
|
|
*lpbkPosition = ulBK + 3;
|
|
|
|
out:
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
DebugTraceResult(IVTABC_CreateBookmark, hResult);
|
|
return hResult;
|
|
|
|
readerror:
|
|
/*
|
|
* Didn't get the record.
|
|
*/
|
|
|
|
/* Restore back to original position */
|
|
(void) SetFilePointer(lpIVTAbc->hFile, lpIVTAbc->ulPosition, NULL, FILE_BEGIN);
|
|
|
|
/* Free up the new bookmark */
|
|
lpIVTAbc->lpFreeBuff(lpABCBK);
|
|
|
|
hResult = ResultFromScode(MAPI_E_UNABLE_TO_COMPLETE);
|
|
SetErrorIDS(lpIVTAbc, hResult, IDS_FAB_NO_READ);
|
|
|
|
goto out;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_FreeBookmark
|
|
-
|
|
* Frees up the given bookmark
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_FreeBookmark( LPIVTABC lpIVTAbc,
|
|
BOOKMARK bkPosition
|
|
)
|
|
{
|
|
HRESULT hResult = hrSuccess;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_FreeBookmark, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_FreeBookmark, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
|
|
/*
|
|
* Don't try and free up any of the standard bookmarks
|
|
*/
|
|
if ((bkPosition != BOOKMARK_BEGINNING) &&
|
|
(bkPosition != BOOKMARK_CURRENT) &&
|
|
(bkPosition != BOOKMARK_END))
|
|
{
|
|
ULONG ulBK = (ULONG) bkPosition - 3;
|
|
|
|
/*
|
|
* See if it's in range
|
|
*/
|
|
if (ulBK >= 0 && ulBK < MAX_BOOKMARKS)
|
|
{
|
|
LPABCBK lpABCBK = NULL;
|
|
|
|
/* If it's valid... */
|
|
if (lpABCBK = lpIVTAbc->rglpABCBK[ulBK]) /* '=' on purpose */
|
|
{
|
|
/* ...free it up. */
|
|
|
|
lpIVTAbc->lpFreeBuff(lpABCBK);
|
|
lpIVTAbc->rglpABCBK[ulBK] = NULL;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* It's an error
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
DebugTraceResult(IVTABC_FreeBookmark, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_ExpandRow
|
|
-
|
|
* Stubbed out. This table doesn't implement catagorization.
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_ExpandRow( LPIVTABC lpIVTAbc,
|
|
ULONG cbIKey,
|
|
LPBYTE pbIKey,
|
|
ULONG ulRowCount,
|
|
ULONG ulFlags,
|
|
LPSRowSet FAR * lppRows,
|
|
ULONG FAR * lpulMoreRows
|
|
)
|
|
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_ExpandRow, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_ExpandRow, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
hResult = ResultFromScode(MAPI_E_NO_SUPPORT);
|
|
|
|
DebugTraceResult(IVTABC_ExpandRow, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_CollapseRow
|
|
-
|
|
* Stubbed out. This table doesn't implement catagorization.
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_CollapseRow( LPIVTABC lpIVTAbc,
|
|
ULONG cbIKey,
|
|
LPBYTE pbIKey,
|
|
ULONG ulFlags,
|
|
ULONG FAR * lpulRowCount
|
|
)
|
|
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_CollapseRow, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_CollapseRow, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
hResult = ResultFromScode(MAPI_E_NO_SUPPORT);
|
|
|
|
DebugTraceResult(IVTABC_CollapseRow, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_WaitForCompletion
|
|
-
|
|
* Stubbed out.
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_WaitForCompletion( LPIVTABC lpIVTAbc,
|
|
ULONG ulFlags,
|
|
ULONG ulTimeout,
|
|
ULONG FAR * lpulTableStatus
|
|
)
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_WaitForCompletion, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_WaitForCompletion, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
hResult = ResultFromScode(MAPI_E_NO_SUPPORT);
|
|
|
|
DebugTraceResult(IVTABC_WaitForCompletion, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_Abort
|
|
-
|
|
* Nothing ever to abort...
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_Abort(LPIVTABC lpIVTAbc)
|
|
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Validate parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_Abort, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_Abort, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
return hrSuccess;
|
|
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
*
|
|
- IVTABC_Advise
|
|
-
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_Advise( LPIVTABC lpIVTAbc,
|
|
ULONG ulEventmask,
|
|
LPMAPIADVISESINK lpAdviseSink,
|
|
ULONG FAR * lpulConnection
|
|
)
|
|
{
|
|
HRESULT hResult = hrSuccess;
|
|
UINT iAdvise;
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
DebugTraceSc(IVTABC_Advise, E_INVALIDARG);
|
|
return ResultFromScode(E_INVALIDARG);
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_Advise, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Validate the parameters
|
|
*/
|
|
|
|
if ((ulEventmask & ~(ULONG) fnevTableModified) ||
|
|
IsBadReadPtr(lpAdviseSink, SIZEOF(LPVOID)) ||
|
|
IsBadReadPtr(lpAdviseSink->lpVtbl, 3 * SIZEOF(LPVOID)) ||
|
|
IsBadWritePtr(lpulConnection, SIZEOF(ULONG)))
|
|
{
|
|
DebugTraceSc(IVTABC_Advise, E_INVALIDARG);
|
|
return ResultFromScode(E_INVALIDARG);
|
|
}
|
|
|
|
/* Get the Critical Section */
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
for (iAdvise = 0;
|
|
lpIVTAbc->parglpAdvise && iAdvise < lpIVTAbc->cAdvise;
|
|
++iAdvise)
|
|
{
|
|
if (lpIVTAbc->parglpAdvise[iAdvise] == NULL)
|
|
break;
|
|
}
|
|
|
|
if (iAdvise >= lpIVTAbc->cAdvise)
|
|
{
|
|
/*
|
|
* Realloc the array if it exists
|
|
*/
|
|
if (lpIVTAbc->parglpAdvise)
|
|
{
|
|
lpIVTAbc->parglpAdvise = lpIVTAbc->lpMalloc->lpVtbl->Realloc(
|
|
lpIVTAbc->lpMalloc,
|
|
lpIVTAbc->parglpAdvise,
|
|
(lpIVTAbc->cAdvise + 1) * SIZEOF(LPMAPIADVISESINK));
|
|
}
|
|
else
|
|
{
|
|
lpIVTAbc->parglpAdvise = lpIVTAbc->lpMalloc->lpVtbl->Alloc(
|
|
lpIVTAbc->lpMalloc,
|
|
(lpIVTAbc->cAdvise + 1) * SIZEOF(LPMAPIADVISESINK));
|
|
}
|
|
|
|
/*
|
|
* Could we get the desired memory?
|
|
*/
|
|
if (lpIVTAbc->parglpAdvise == NULL)
|
|
{
|
|
hResult = MakeResult(E_OUTOFMEMORY);
|
|
goto ret;
|
|
}
|
|
}
|
|
|
|
lpIVTAbc->cAdvise++;
|
|
|
|
*lpulConnection = lpIVTAbc->ulConnectMic + iAdvise;
|
|
|
|
lpIVTAbc->parglpAdvise[iAdvise] = lpAdviseSink;
|
|
|
|
lpAdviseSink->lpVtbl->AddRef(lpAdviseSink);
|
|
|
|
ret:
|
|
/* leave critical section */
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
DebugTraceResult(IVTABC_Advise, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
*
|
|
- IVTABC_Unadvise
|
|
-
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_Unadvise(LPIVTABC lpIVTAbc, ULONG ulConnection)
|
|
{
|
|
LPMAPIADVISESINK padvise;
|
|
UINT iAdvise;
|
|
HRESULT hResult = hrSuccess;
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
DebugTraceSc(IVTABC_Unadvise, E_INVALIDARG);
|
|
return ResultFromScode(E_INVALIDARG);
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_Unadvise, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
if (ulConnection - lpIVTAbc->ulConnectMic > (ULONG) lpIVTAbc->cAdvise)
|
|
{
|
|
DebugTraceSc(IVTABC_Unadvise, E_INVALIDARG);
|
|
return ResultFromScode(E_INVALIDARG);
|
|
}
|
|
|
|
/* Get the Critical Section */
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
iAdvise = (UINT) (ulConnection - lpIVTAbc->ulConnectMic);
|
|
padvise = lpIVTAbc->parglpAdvise[iAdvise];
|
|
padvise->lpVtbl->Release(padvise);
|
|
lpIVTAbc->parglpAdvise[iAdvise] = NULL;
|
|
lpIVTAbc->cAdvise--;
|
|
|
|
/* leave critical section */
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
DebugTraceResult(IVTABC_Unadvise, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_GetStatus
|
|
-
|
|
* Returns the status of this table. This table really isn't
|
|
* dynamic yet, but it could be...
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_GetStatus( LPIVTABC lpIVTAbc,
|
|
ULONG * lpulTableStatus,
|
|
ULONG * lpulTableType
|
|
)
|
|
{
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Parameter checking
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check the out parameters for writability
|
|
*/
|
|
if (IsBadWritePtr(lpulTableStatus, SIZEOF(ULONG))
|
|
|| IsBadWritePtr(lpulTableType, SIZEOF(ULONG)))
|
|
{
|
|
/*
|
|
* Bad out parameters
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
*lpulTableStatus = TBLSTAT_COMPLETE;
|
|
*lpulTableType = TBLTYPE_DYNAMIC;
|
|
|
|
return hrSuccess;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_SetColumns
|
|
-
|
|
*
|
|
* SetColumns for contents table.
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_SetColumns( LPIVTABC lpIVTAbc,
|
|
LPSPropTagArray lpPTAColSet,
|
|
ULONG ulFlags
|
|
)
|
|
{
|
|
SCODE scode;
|
|
HRESULT hResult = hrSuccess;
|
|
int cbSizeOfColSet;
|
|
LPSPropTagArray lpPTAColSetT;
|
|
ULONG uliCol;
|
|
|
|
/*
|
|
* Check parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_SetColumns, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check flags
|
|
*/
|
|
if (ulFlags & ~TBL_NOWAIT)
|
|
{
|
|
hResult = ResultFromScode(MAPI_E_UNKNOWN_FLAGS);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* The first element of the structure
|
|
*/
|
|
if (IsBadReadPtr(lpPTAColSet, offsetof(SPropTagArray, aulPropTag)))
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
cbSizeOfColSet = CbSPropTagArray(lpPTAColSet);
|
|
|
|
/*
|
|
* The rest of the structure
|
|
*/
|
|
if (IsBadReadPtr(lpPTAColSet, cbSizeOfColSet))
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Verify that there are no PT_ERRORs here...
|
|
*/
|
|
for (uliCol = 0; uliCol < lpPTAColSet->cValues; uliCol++)
|
|
{
|
|
if (PROP_TYPE(lpPTAColSet->aulPropTag[uliCol]) == PT_ERROR)
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Allocate a new column set.
|
|
*/
|
|
|
|
scode = lpIVTAbc->lpAllocBuff(cbSizeOfColSet,(LPVOID *) &lpPTAColSetT);
|
|
|
|
if (FAILED(scode))
|
|
{
|
|
hResult = ResultFromScode(scode);
|
|
|
|
DebugTraceResult(IVTABC_GetStatus, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Copy the column set in
|
|
*/
|
|
if (cbSizeOfColSet)
|
|
memcpy(lpPTAColSetT, lpPTAColSet, cbSizeOfColSet);
|
|
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
|
|
if (lpIVTAbc->lpPTAColSet != ptagaivtabcColSet)
|
|
{
|
|
/*
|
|
* Free up the old column set
|
|
*/
|
|
lpIVTAbc->lpFreeBuff(lpIVTAbc->lpPTAColSet);
|
|
}
|
|
|
|
lpIVTAbc->lpPTAColSet = lpPTAColSetT;
|
|
|
|
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
|
|
return hrSuccess;
|
|
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_QueryColumns
|
|
-
|
|
*
|
|
*
|
|
* I always have all my columns available... and active.
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_QueryColumns( LPIVTABC lpIVTAbc,
|
|
ULONG ulFlags,
|
|
LPSPropTagArray FAR * lppColumns
|
|
)
|
|
{
|
|
SCODE scode;
|
|
HRESULT hResult = hrSuccess;
|
|
int cbSizeOfColSet;
|
|
|
|
/*
|
|
* Check parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_QueryColumns, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_QueryColumns, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check the out parameters for writability
|
|
*/
|
|
if (IsBadWritePtr(lppColumns, SIZEOF(LPSPropTagArray)))
|
|
{
|
|
hResult = ResultFromScode(E_INVALIDARG);
|
|
|
|
DebugTraceResult(IVTABC_QueryColumns, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*
|
|
* Check flags
|
|
*/
|
|
if (ulFlags & ~TBL_ALL_COLUMNS)
|
|
{
|
|
hResult = ResultFromScode(MAPI_E_UNKNOWN_FLAGS);
|
|
|
|
DebugTraceResult(IVTABC_QueryColumns, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
|
|
EnterCriticalSection(&lpIVTAbc->cs);
|
|
|
|
/*
|
|
* Allocate enough memory for the column set
|
|
*/
|
|
if (ulFlags & TBL_ALL_COLUMNS)
|
|
|
|
cbSizeOfColSet = sizeof(ULONG) +
|
|
(int)(ptagaivtabcColSet->cValues) * SIZEOF(ULONG);
|
|
else
|
|
cbSizeOfColSet = sizeof(ULONG) +
|
|
(int)lpIVTAbc->lpPTAColSet->cValues * SIZEOF(ULONG);
|
|
|
|
|
|
scode = lpIVTAbc->lpAllocBuff (cbSizeOfColSet,(LPVOID *) lppColumns);
|
|
|
|
if (FAILED(scode))
|
|
{
|
|
hResult = ResultFromScode(scode);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Copy the column set in
|
|
*/
|
|
if (ulFlags & TBL_ALL_COLUMNS)
|
|
memcpy(*lppColumns, ptagaivtabcColSet, cbSizeOfColSet);
|
|
else
|
|
memcpy(*lppColumns, lpIVTAbc->lpPTAColSet, cbSizeOfColSet);
|
|
|
|
|
|
out:
|
|
LeaveCriticalSection(&lpIVTAbc->cs);
|
|
|
|
DebugTraceResult(IVTABC_QueryColumns, hResult);
|
|
return hResult;
|
|
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_GetCollapseState
|
|
-
|
|
* Stubbed out. Only necessary if this table were to support categorization.
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_GetCollapseState( LPIVTABC lpIVTAbc,
|
|
ULONG ulFlags,
|
|
ULONG cbInstanceKey,
|
|
LPBYTE pbInstanceKey,
|
|
ULONG FAR * lpcbCollapseState,
|
|
LPBYTE FAR * lppbCollapseState
|
|
)
|
|
{
|
|
SCODE sc = MAPI_E_NO_SUPPORT;
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Check parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
sc = E_INVALIDARG;
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
sc = E_INVALIDARG;
|
|
goto out;
|
|
}
|
|
|
|
if ( IsBadReadPtr(pbInstanceKey, (UINT) cbInstanceKey))
|
|
{
|
|
sc = MAPI_E_INVALID_PARAMETER;
|
|
goto out;
|
|
}
|
|
|
|
if ( IsBadWritePtr(lpcbCollapseState, SIZEOF(ULONG)))
|
|
{
|
|
sc = MAPI_E_INVALID_PARAMETER;
|
|
goto out;
|
|
}
|
|
|
|
if ( IsBadWritePtr(lppbCollapseState, SIZEOF(LPBYTE)))
|
|
{
|
|
sc = MAPI_E_INVALID_PARAMETER;
|
|
goto out;
|
|
}
|
|
|
|
if ( ulFlags )
|
|
{
|
|
sc = MAPI_E_UNKNOWN_FLAGS;
|
|
}
|
|
|
|
out:
|
|
|
|
hResult = ResultFromScode(sc);
|
|
DebugTraceResult(IVTABC_GetCollapseState, hResult);
|
|
return hResult;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*
|
|
- IVTABC_SetCollapseState
|
|
-
|
|
* Stubbed out. Only necessary if this table were to support categorization.
|
|
*
|
|
*
|
|
*
|
|
*/
|
|
STDMETHODIMP
|
|
IVTABC_SetCollapseState( LPIVTABC lpIVTAbc,
|
|
ULONG ulFlags,
|
|
ULONG cbCollapseState,
|
|
LPBYTE pbCollapseState,
|
|
BOOKMARK FAR * lpbkLocation
|
|
)
|
|
{
|
|
|
|
SCODE sc = MAPI_E_NO_SUPPORT;
|
|
HRESULT hResult;
|
|
|
|
/*
|
|
* Check parameters
|
|
*/
|
|
|
|
/*
|
|
* Check to see if it's large enough to hold this object
|
|
*/
|
|
if (IsBadReadPtr(lpIVTAbc, SIZEOF(IVTABC)))
|
|
{
|
|
/*
|
|
* Not large enough
|
|
*/
|
|
sc = E_INVALIDARG;
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Check to see that it's the correct vtbl
|
|
*/
|
|
if (lpIVTAbc->lpVtbl != &vtblIVTABC)
|
|
{
|
|
/*
|
|
* Not my vtbl
|
|
*/
|
|
sc = E_INVALIDARG;
|
|
goto out;
|
|
}
|
|
|
|
if ( IsBadReadPtr(pbCollapseState, (UINT) cbCollapseState))
|
|
{
|
|
sc = MAPI_E_INVALID_PARAMETER;
|
|
goto out;
|
|
}
|
|
|
|
if (IsBadWritePtr(lpbkLocation, SIZEOF(BOOKMARK)))
|
|
{
|
|
sc = MAPI_E_INVALID_PARAMETER;
|
|
goto out;
|
|
}
|
|
|
|
if ( ulFlags )
|
|
{
|
|
sc = MAPI_E_UNKNOWN_FLAGS;
|
|
}
|
|
|
|
out:
|
|
|
|
hResult = ResultFromScode(sc);
|
|
DebugTraceResult(IVTABC_SetCollapseState, hResult);
|
|
return hResult;
|
|
}
|