4 * Copyright 2004 Jon Griffiths
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #define NONAMELESSUNION
23 #define NONAMELESSSTRUCT
31 #include "wine/list.h"
32 #include "wine/debug.h"
33 #include "wine/unicode.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(mapi);
38 BOOL WINAPI FBadRglpszA(LPSTR*,ULONG);
40 /* Internal: Check if a property value array is invalid */
41 static inline ULONG PROP_BadArray(LPSPropValue lpProp, size_t elemSize)
43 return IsBadReadPtr(lpProp->Value.MVi.lpi, lpProp->Value.MVi.cValues * elemSize);
46 /*************************************************************************
47 * PropCopyMore@16 (MAPI32.76)
49 * Copy a property value.
52 * lpDest [O] Destination for the copied value
53 * lpSrc [I] Property value to copy to lpDest
54 * lpMore [I] Linked memory allocation function (pass MAPIAllocateMore())
55 * lpOrig [I] Original allocation to which memory will be linked
58 * Success: S_OK. lpDest contains a deep copy of lpSrc.
59 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
60 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
63 * Any elements within the property returned should not be individually
64 * freed, as they will be freed when lpOrig is.
66 SCODE WINAPI PropCopyMore(LPSPropValue lpDest, LPSPropValue lpSrc,
67 ALLOCATEMORE *lpMore, LPVOID lpOrig)
72 TRACE("(%p,%p,%p,%p)\n", lpDest, lpSrc, lpMore, lpOrig);
74 if (!lpDest || IsBadWritePtr(lpDest, sizeof(SPropValue)) ||
75 FBadProp(lpSrc) || !lpMore)
76 return MAPI_E_INVALID_PARAMETER;
78 /* Shallow copy first, this is sufficient for properties without pointers */
81 switch (PROP_TYPE(lpSrc->ulPropTag))
84 scode = lpMore(sizeof(GUID), lpOrig, (LPVOID*)&lpDest->Value.lpguid);
86 *lpDest->Value.lpguid = *lpSrc->Value.lpguid;
89 ulLen = lstrlenA(lpSrc->Value.lpszA) + 1u;
90 scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.lpszA);
92 memcpy(lpDest->Value.lpszA, lpSrc->Value.lpszA, ulLen);
95 ulLen = (strlenW(lpSrc->Value.lpszW) + 1u) * sizeof(WCHAR);
96 scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.lpszW);
98 memcpy(lpDest->Value.lpszW, lpSrc->Value.lpszW, ulLen);
101 scode = lpMore(lpSrc->Value.bin.cb, lpOrig, (LPVOID*)&lpDest->Value.bin.lpb);
102 if (SUCCEEDED(scode))
103 memcpy(lpDest->Value.bin.lpb, lpSrc->Value.bin.lpb, lpSrc->Value.bin.cb);
106 if (lpSrc->ulPropTag & MV_FLAG)
108 ulLen = UlPropSize(lpSrc);
110 if (PROP_TYPE(lpSrc->ulPropTag) == PT_MV_STRING8 ||
111 PROP_TYPE(lpSrc->ulPropTag) == PT_MV_UNICODE)
113 /* UlPropSize doesn't account for the string pointers */
114 ulLen += lpSrc->Value.MVszA.cValues * sizeof(char*);
116 else if (PROP_TYPE(lpSrc->ulPropTag) == PT_MV_BINARY)
118 /* UlPropSize doesn't account for the SBinary structs */
119 ulLen += lpSrc->Value.MVbin.cValues * sizeof(SBinary);
122 lpDest->Value.MVi.cValues = lpSrc->Value.MVi.cValues;
123 scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.MVi.lpi);
127 /* Note that we could allocate the memory for each value in a
128 * multi-value property separately, however if an allocation failed
129 * we would be left with a bunch of allocated memory, which (while
130 * not really leaked) is unusable until lpOrig is freed. So for
131 * strings and binary arrays we make a single allocation for all
132 * of the data. This is consistent since individual elements can't
136 switch (PROP_TYPE(lpSrc->ulPropTag))
140 char *lpNextStr = (char*)(lpDest->Value.MVszA.lppszA +
141 lpDest->Value.MVszA.cValues);
143 for (i = 0; i < lpSrc->Value.MVszA.cValues; i++)
145 ULONG ulStrLen = lstrlenA(lpSrc->Value.MVszA.lppszA[i]) + 1u;
147 lpDest->Value.MVszA.lppszA[i] = lpNextStr;
148 memcpy(lpNextStr, lpSrc->Value.MVszA.lppszA[i], ulStrLen);
149 lpNextStr += ulStrLen;
155 WCHAR *lpNextStr = (WCHAR*)(lpDest->Value.MVszW.lppszW +
156 lpDest->Value.MVszW.cValues);
158 for (i = 0; i < lpSrc->Value.MVszW.cValues; i++)
160 ULONG ulStrLen = strlenW(lpSrc->Value.MVszW.lppszW[i]) + 1u;
162 lpDest->Value.MVszW.lppszW[i] = lpNextStr;
163 memcpy(lpNextStr, lpSrc->Value.MVszW.lppszW[i], ulStrLen * sizeof(WCHAR));
164 lpNextStr += ulStrLen;
170 LPBYTE lpNext = (LPBYTE)(lpDest->Value.MVbin.lpbin +
171 lpDest->Value.MVbin.cValues);
173 for (i = 0; i < lpSrc->Value.MVszW.cValues; i++)
175 lpDest->Value.MVbin.lpbin[i].cb = lpSrc->Value.MVbin.lpbin[i].cb;
176 lpDest->Value.MVbin.lpbin[i].lpb = lpNext;
177 memcpy(lpNext, lpSrc->Value.MVbin.lpbin[i].lpb, lpDest->Value.MVbin.lpbin[i].cb);
178 lpNext += lpDest->Value.MVbin.lpbin[i].cb;
183 /* No embedded pointers, just copy the data over */
184 memcpy(lpDest->Value.MVi.lpi, lpSrc->Value.MVi.lpi, ulLen);
193 /*************************************************************************
194 * UlPropSize@4 (MAPI32.77)
196 * Determine the size of a property in bytes.
199 * lpProp [I] Property to determine the size of
202 * Success: The size of the value in lpProp.
203 * Failure: 0, if a multi-value (array) property is invalid or the type of lpProp
207 * - The size returned does not include the size of the SPropValue struct
208 * or the size of the array of pointers for multi-valued properties that
209 * contain pointers (such as PT_MV_STRING8 or PT-MV_UNICODE).
210 * - MSDN incorrectly states that this function returns MAPI_E_CALL_FAILED if
211 * lpProp is invalid. In reality no checking is performed and this function
212 * will crash if passed an invalid property, or return 0 if the property
213 * type is PT_OBJECT or is unknown.
215 ULONG WINAPI UlPropSize(LPSPropValue lpProp)
219 TRACE("(%p)\n", lpProp);
221 switch (PROP_TYPE(lpProp->ulPropTag))
223 case PT_MV_I2: ulRet = lpProp->Value.MVi.cValues;
226 case PT_I2: ulRet *= sizeof(USHORT);
228 case PT_MV_I4: ulRet = lpProp->Value.MVl.cValues;
231 case PT_I4: ulRet *= sizeof(LONG);
233 case PT_MV_I8: ulRet = lpProp->Value.MVli.cValues;
235 case PT_I8: ulRet *= sizeof(LONG64);
237 case PT_MV_R4: ulRet = lpProp->Value.MVflt.cValues;
239 case PT_R4: ulRet *= sizeof(float);
242 case PT_MV_R8: ulRet = lpProp->Value.MVdbl.cValues;
245 case PT_R8: ulRet *= sizeof(double);
247 case PT_MV_CURRENCY: ulRet = lpProp->Value.MVcur.cValues;
249 case PT_CURRENCY: ulRet *= sizeof(CY);
251 case PT_MV_SYSTIME: ulRet = lpProp->Value.MVft.cValues;
253 case PT_SYSTIME: ulRet *= sizeof(FILETIME);
255 case PT_MV_CLSID: ulRet = lpProp->Value.MVguid.cValues;
257 case PT_CLSID: ulRet *= sizeof(GUID);
259 case PT_MV_STRING8: ulRet = 0u;
260 for (i = 0; i < lpProp->Value.MVszA.cValues; i++)
261 ulRet += (lstrlenA(lpProp->Value.MVszA.lppszA[i]) + 1u);
263 case PT_STRING8: ulRet = lstrlenA(lpProp->Value.lpszA) + 1u;
265 case PT_MV_UNICODE: ulRet = 0u;
266 for (i = 0; i < lpProp->Value.MVszW.cValues; i++)
267 ulRet += (strlenW(lpProp->Value.MVszW.lppszW[i]) + 1u);
268 ulRet *= sizeof(WCHAR);
270 case PT_UNICODE: ulRet = (lstrlenW(lpProp->Value.lpszW) + 1u) * sizeof(WCHAR);
272 case PT_MV_BINARY: ulRet = 0u;
273 for (i = 0; i < lpProp->Value.MVbin.cValues; i++)
274 ulRet += lpProp->Value.MVbin.lpbin[i].cb;
276 case PT_BINARY: ulRet = lpProp->Value.bin.cb;
286 /*************************************************************************
287 * FPropContainsProp@12 (MAPI32.78)
289 * Find a property with a given property tag in a property array.
292 * lpHaystack [I] Property to match to
293 * lpNeedle [I] Property to find in lpHaystack
294 * ulFuzzy [I] Flags controlling match type and strictness (FL_* flags from "mapidefs.h")
297 * TRUE, if lpNeedle matches lpHaystack according to the criteria of ulFuzzy.
300 * Only property types of PT_STRING8 and PT_BINARY are handled by this function.
302 BOOL WINAPI FPropContainsProp(LPSPropValue lpHaystack, LPSPropValue lpNeedle, ULONG ulFuzzy)
304 TRACE("(%p,%p,0x%08x)\n", lpHaystack, lpNeedle, ulFuzzy);
306 if (FBadProp(lpHaystack) || FBadProp(lpNeedle) ||
307 PROP_TYPE(lpHaystack->ulPropTag) != PROP_TYPE(lpNeedle->ulPropTag))
310 /* FIXME: Do later versions support Unicode as well? */
312 if (PROP_TYPE(lpHaystack->ulPropTag) == PT_STRING8)
314 DWORD dwFlags = 0, dwNeedleLen, dwHaystackLen;
316 if (ulFuzzy & FL_IGNORECASE)
317 dwFlags |= NORM_IGNORECASE;
318 if (ulFuzzy & FL_IGNORENONSPACE)
319 dwFlags |= NORM_IGNORENONSPACE;
320 if (ulFuzzy & FL_LOOSE)
321 dwFlags |= (NORM_IGNORECASE|NORM_IGNORENONSPACE|NORM_IGNORESYMBOLS);
323 dwNeedleLen = lstrlenA(lpNeedle->Value.lpszA);
324 dwHaystackLen = lstrlenA(lpHaystack->Value.lpszA);
326 if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_PREFIX)
328 if (dwNeedleLen <= dwHaystackLen &&
329 CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
330 lpHaystack->Value.lpszA, dwNeedleLen,
331 lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
332 return TRUE; /* needle is a prefix of haystack */
334 else if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_SUBSTRING)
336 LPSTR (WINAPI *pStrChrFn)(LPCSTR,WORD) = StrChrA;
337 LPSTR lpStr = lpHaystack->Value.lpszA;
339 if (dwFlags & NORM_IGNORECASE)
340 pStrChrFn = StrChrIA;
342 while ((lpStr = pStrChrFn(lpStr, *lpNeedle->Value.lpszA)) != NULL)
344 dwHaystackLen -= (lpStr - lpHaystack->Value.lpszA);
345 if (dwNeedleLen <= dwHaystackLen &&
346 CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
348 lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
349 return TRUE; /* needle is a substring of haystack */
353 else if (CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
354 lpHaystack->Value.lpszA, dwHaystackLen,
355 lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
356 return TRUE; /* full string match */
358 else if (PROP_TYPE(lpHaystack->ulPropTag) == PT_BINARY)
360 if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_PREFIX)
362 if (lpNeedle->Value.bin.cb <= lpHaystack->Value.bin.cb &&
363 !memcmp(lpNeedle->Value.bin.lpb, lpHaystack->Value.bin.lpb,
364 lpNeedle->Value.bin.cb))
365 return TRUE; /* needle is a prefix of haystack */
367 else if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_SUBSTRING)
369 ULONG ulLen = lpHaystack->Value.bin.cb;
370 LPBYTE lpb = lpHaystack->Value.bin.lpb;
372 while ((lpb = memchr(lpb, *lpNeedle->Value.bin.lpb, ulLen)) != NULL)
374 ulLen = lpHaystack->Value.bin.cb - (lpb - lpHaystack->Value.bin.lpb);
375 if (lpNeedle->Value.bin.cb <= ulLen &&
376 !memcmp(lpNeedle->Value.bin.lpb, lpb, lpNeedle->Value.bin.cb))
377 return TRUE; /* needle is a substring of haystack */
381 else if (!LPropCompareProp(lpHaystack, lpNeedle))
382 return TRUE; /* needle is an exact match with haystack */
388 /*************************************************************************
389 * FPropCompareProp@12 (MAPI32.79)
391 * Compare two properties.
394 * lpPropLeft [I] Left hand property to compare to lpPropRight
395 * ulOp [I] Comparison operator (RELOP_* enum from "mapidefs.h")
396 * lpPropRight [I] Right hand property to compare to lpPropLeft
399 * TRUE, if the comparison is true, FALSE otherwise.
401 BOOL WINAPI FPropCompareProp(LPSPropValue lpPropLeft, ULONG ulOp, LPSPropValue lpPropRight)
405 TRACE("(%p,%d,%p)\n", lpPropLeft, ulOp, lpPropRight);
407 if (ulOp > RELOP_RE || FBadProp(lpPropLeft) || FBadProp(lpPropRight))
410 if (ulOp == RELOP_RE)
412 FIXME("Comparison operator RELOP_RE not yet implemented!\n");
416 iCmp = LPropCompareProp(lpPropLeft, lpPropRight);
420 case RELOP_LT: return iCmp < 0 ? TRUE : FALSE;
421 case RELOP_LE: return iCmp <= 0 ? TRUE : FALSE;
422 case RELOP_GT: return iCmp > 0 ? TRUE : FALSE;
423 case RELOP_GE: return iCmp >= 0 ? TRUE : FALSE;
424 case RELOP_EQ: return iCmp == 0 ? TRUE : FALSE;
425 case RELOP_NE: return iCmp != 0 ? TRUE : FALSE;
430 /*************************************************************************
431 * LPropCompareProp@8 (MAPI32.80)
433 * Compare two properties.
436 * lpPropLeft [I] Left hand property to compare to lpPropRight
437 * lpPropRight [I] Right hand property to compare to lpPropLeft
440 * An integer less than, equal to or greater than 0, indicating that
441 * lpszStr is less than, the same, or greater than lpszComp.
443 LONG WINAPI LPropCompareProp(LPSPropValue lpPropLeft, LPSPropValue lpPropRight)
447 TRACE("(%p->0x%08x,%p->0x%08x)\n", lpPropLeft, lpPropLeft->ulPropTag,
448 lpPropRight, lpPropRight->ulPropTag);
450 /* If the properties are not the same, sort by property type */
451 if (PROP_TYPE(lpPropLeft->ulPropTag) != PROP_TYPE(lpPropRight->ulPropTag))
452 return (LONG)PROP_TYPE(lpPropLeft->ulPropTag) - (LONG)PROP_TYPE(lpPropRight->ulPropTag);
454 switch (PROP_TYPE(lpPropLeft->ulPropTag))
458 return 0; /* NULLs are equal */
460 return lpPropLeft->Value.i - lpPropRight->Value.i;
462 return lpPropLeft->Value.l - lpPropRight->Value.l;
464 if (lpPropLeft->Value.li.QuadPart > lpPropRight->Value.li.QuadPart)
466 if (lpPropLeft->Value.li.QuadPart == lpPropRight->Value.li.QuadPart)
470 if (lpPropLeft->Value.flt > lpPropRight->Value.flt)
472 if (lpPropLeft->Value.flt == lpPropRight->Value.flt)
477 if (lpPropLeft->Value.dbl > lpPropRight->Value.dbl)
479 if (lpPropLeft->Value.dbl == lpPropRight->Value.dbl)
483 if (lpPropLeft->Value.cur.int64 > lpPropRight->Value.cur.int64)
485 if (lpPropLeft->Value.cur.int64 == lpPropRight->Value.cur.int64)
489 return CompareFileTime(&lpPropLeft->Value.ft, &lpPropRight->Value.ft);
491 return (lpPropLeft->Value.b ? 1 : 0) - (lpPropRight->Value.b ? 1 : 0);
493 if (lpPropLeft->Value.bin.cb == lpPropRight->Value.bin.cb)
494 iRet = memcmp(lpPropLeft->Value.bin.lpb, lpPropRight->Value.bin.lpb,
495 lpPropLeft->Value.bin.cb);
498 iRet = memcmp(lpPropLeft->Value.bin.lpb, lpPropRight->Value.bin.lpb,
499 min(lpPropLeft->Value.bin.cb, lpPropRight->Value.bin.cb));
502 iRet = lpPropLeft->Value.bin.cb - lpPropRight->Value.bin.cb;
506 return lstrcmpA(lpPropLeft->Value.lpszA, lpPropRight->Value.lpszA);
508 return strcmpW(lpPropLeft->Value.lpszW, lpPropRight->Value.lpszW);
510 if (lpPropLeft->Value.err > lpPropRight->Value.err)
512 if (lpPropLeft->Value.err == lpPropRight->Value.err)
516 return memcmp(lpPropLeft->Value.lpguid, lpPropRight->Value.lpguid,
519 FIXME("Unhandled property type %d\n", PROP_TYPE(lpPropLeft->ulPropTag));
523 /*************************************************************************
524 * HrGetOneProp@8 (MAPI32.135)
526 * Get a property value from an IMAPIProp object.
529 * lpIProp [I] IMAPIProp object to get the property value in
530 * ulPropTag [I] Property tag of the property to get
531 * lppProp [O] Destination for the returned property
534 * Success: S_OK. *lppProp contains the property value requested.
535 * Failure: MAPI_E_NOT_FOUND, if no property value has the tag given by ulPropTag.
537 HRESULT WINAPI HrGetOneProp(LPMAPIPROP lpIProp, ULONG ulPropTag, LPSPropValue *lppProp)
543 TRACE("(%p,%d,%p)\n", lpIProp, ulPropTag, lppProp);
546 pta.aulPropTag[0] = ulPropTag;
547 hRet = IMAPIProp_GetProps(lpIProp, &pta, 0u, &ulCount, lppProp);
548 if (hRet == MAPI_W_ERRORS_RETURNED)
550 MAPIFreeBuffer(*lppProp);
552 hRet = MAPI_E_NOT_FOUND;
557 /*************************************************************************
558 * HrSetOneProp@8 (MAPI32.136)
560 * Set a property value in an IMAPIProp object.
563 * lpIProp [I] IMAPIProp object to set the property value in
564 * lpProp [I] Property value to set
567 * Success: S_OK. The value in lpProp is set in lpIProp.
568 * Failure: An error result from IMAPIProp_SetProps().
570 HRESULT WINAPI HrSetOneProp(LPMAPIPROP lpIProp, LPSPropValue lpProp)
572 TRACE("(%p,%p)\n", lpIProp, lpProp);
574 return IMAPIProp_SetProps(lpIProp, 1u, lpProp, NULL);
577 /*************************************************************************
578 * FPropExists@8 (MAPI32.137)
580 * Find a property with a given property tag in an IMAPIProp object.
583 * lpIProp [I] IMAPIProp object to find the property tag in
584 * ulPropTag [I] Property tag to find
587 * TRUE, if ulPropTag matches a property held in lpIProp,
591 * if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
592 * Ids need to match for a successful match to occur.
594 BOOL WINAPI FPropExists(LPMAPIPROP lpIProp, ULONG ulPropTag)
598 TRACE("(%p,%d)\n", lpIProp, ulPropTag);
602 LPSPropTagArray lpTags;
605 if (FAILED(IMAPIProp_GetPropList(lpIProp, 0u, &lpTags)))
608 for (i = 0; i < lpTags->cValues; i++)
610 if (!FBadPropTag(lpTags->aulPropTag[i]) &&
611 (lpTags->aulPropTag[i] == ulPropTag ||
612 (PROP_TYPE(ulPropTag) == PT_UNSPECIFIED &&
613 PROP_ID(lpTags->aulPropTag[i]) == lpTags->aulPropTag[i])))
619 MAPIFreeBuffer(lpTags);
624 /*************************************************************************
625 * PpropFindProp@12 (MAPI32.138)
627 * Find a property with a given property tag in a property array.
630 * lpProps [I] Property array to search
631 * cValues [I] Number of properties in lpProps
632 * ulPropTag [I] Property tag to find
635 * A pointer to the matching property, or NULL if none was found.
638 * if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
639 * Ids need to match for a successful match to occur.
641 LPSPropValue WINAPI PpropFindProp(LPSPropValue lpProps, ULONG cValues, ULONG ulPropTag)
643 TRACE("(%p,%d,%d)\n", lpProps, cValues, ulPropTag);
645 if (lpProps && cValues)
648 for (i = 0; i < cValues; i++)
650 if (!FBadPropTag(lpProps[i].ulPropTag) &&
651 (lpProps[i].ulPropTag == ulPropTag ||
652 (PROP_TYPE(ulPropTag) == PT_UNSPECIFIED &&
653 PROP_ID(lpProps[i].ulPropTag) == PROP_ID(ulPropTag))))
660 /*************************************************************************
661 * FreePadrlist@4 (MAPI32.139)
663 * Free the memory used by an address book list.
666 * lpAddrs [I] Address book list to free
671 VOID WINAPI FreePadrlist(LPADRLIST lpAddrs)
673 TRACE("(%p)\n", lpAddrs);
675 /* Structures are binary compatible; use the same implementation */
676 FreeProws((LPSRowSet)lpAddrs);
679 /*************************************************************************
680 * FreeProws@4 (MAPI32.140)
682 * Free the memory used by a row set.
685 * lpRowSet [I] Row set to free
690 VOID WINAPI FreeProws(LPSRowSet lpRowSet)
692 TRACE("(%p)\n", lpRowSet);
698 for (i = 0; i < lpRowSet->cRows; i++)
699 MAPIFreeBuffer(lpRowSet->aRow[i].lpProps);
701 MAPIFreeBuffer(lpRowSet);
705 /*************************************************************************
706 * ScCountProps@12 (MAPI32.170)
708 * Validate and determine the length of an array of properties.
711 * iCount [I] Length of the lpProps array
712 * lpProps [I] Array of properties to validate/size
713 * pcBytes [O] If non-NULL, destination for the size of the property array
716 * Success: S_OK. If pcBytes is non-NULL, it contains the size of the
718 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid or validation
719 * of the property array fails.
721 SCODE WINAPI ScCountProps(INT iCount, LPSPropValue lpProps, ULONG *pcBytes)
723 ULONG i, ulCount = iCount, ulBytes = 0;
725 TRACE("(%d,%p,%p)\n", iCount, lpProps, pcBytes);
727 if (iCount <= 0 || !lpProps ||
728 IsBadReadPtr(lpProps, iCount * sizeof(SPropValue)))
729 return MAPI_E_INVALID_PARAMETER;
731 for (i = 0; i < ulCount; i++)
733 ULONG ulPropSize = 0;
735 if (FBadProp(&lpProps[i]) || lpProps[i].ulPropTag == PROP_ID_NULL ||
736 lpProps[i].ulPropTag == PROP_ID_INVALID)
737 return MAPI_E_INVALID_PARAMETER;
739 if (PROP_TYPE(lpProps[i].ulPropTag) != PT_OBJECT)
741 ulPropSize = UlPropSize(&lpProps[i]);
743 return MAPI_E_INVALID_PARAMETER;
746 switch (PROP_TYPE(lpProps[i].ulPropTag))
760 ulPropSize += sizeof(SPropValue);
763 ulPropSize += lpProps[i].Value.MVszA.cValues * sizeof(char*) + sizeof(SPropValue);
767 ulPropSize += lpProps[i].Value.MVszA.cValues * sizeof(char*) + sizeof(SPropValue);
770 ulPropSize += lpProps[i].Value.MVbin.cValues * sizeof(SBinary) + sizeof(SPropValue);
773 ulPropSize = sizeof(SPropValue);
776 ulBytes += ulPropSize;
784 /*************************************************************************
785 * ScCopyProps@16 (MAPI32.171)
787 * Copy an array of property values into a buffer suited for serialisation.
790 * cValues [I] Number of properties in lpProps
791 * lpProps [I] Property array to copy
792 * lpDst [O] Destination for the serialised data
793 * lpCount [O] If non-NULL, destination for the number of bytes of data written to lpDst
796 * Success: S_OK. lpDst contains the serialised data from lpProps.
797 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
800 * The resulting property value array is stored in a contiguous block starting at lpDst.
802 SCODE WINAPI ScCopyProps(int cValues, LPSPropValue lpProps, LPVOID lpDst, ULONG *lpCount)
804 LPSPropValue lpDest = (LPSPropValue)lpDst;
805 char *lpDataDest = (char *)(lpDest + cValues);
809 TRACE("(%d,%p,%p,%p)\n", cValues, lpProps, lpDst, lpCount);
811 if (!lpProps || cValues < 0 || !lpDest)
812 return MAPI_E_INVALID_PARAMETER;
814 memcpy(lpDst, lpProps, cValues * sizeof(SPropValue));
816 for (iter = 0; iter < cValues; iter++)
818 switch (PROP_TYPE(lpProps->ulPropTag))
821 lpDest->Value.lpguid = (LPGUID)lpDataDest;
822 *lpDest->Value.lpguid = *lpProps->Value.lpguid;
823 lpDataDest += sizeof(GUID);
826 ulLen = lstrlenA(lpProps->Value.lpszA) + 1u;
827 lpDest->Value.lpszA = lpDataDest;
828 memcpy(lpDest->Value.lpszA, lpProps->Value.lpszA, ulLen);
832 ulLen = (strlenW(lpProps->Value.lpszW) + 1u) * sizeof(WCHAR);
833 lpDest->Value.lpszW = (LPWSTR)lpDataDest;
834 memcpy(lpDest->Value.lpszW, lpProps->Value.lpszW, ulLen);
838 lpDest->Value.bin.lpb = (LPBYTE)lpDataDest;
839 memcpy(lpDest->Value.bin.lpb, lpProps->Value.bin.lpb, lpProps->Value.bin.cb);
840 lpDataDest += lpProps->Value.bin.cb;
843 if (lpProps->ulPropTag & MV_FLAG)
845 lpDest->Value.MVi.cValues = lpProps->Value.MVi.cValues;
846 /* Note: Assignment uses lppszA but covers all cases by union aliasing */
847 lpDest->Value.MVszA.lppszA = (char**)lpDataDest;
849 switch (PROP_TYPE(lpProps->ulPropTag))
853 lpDataDest += lpProps->Value.MVszA.cValues * sizeof(char *);
855 for (i = 0; i < lpProps->Value.MVszA.cValues; i++)
857 ULONG ulStrLen = lstrlenA(lpProps->Value.MVszA.lppszA[i]) + 1u;
859 lpDest->Value.MVszA.lppszA[i] = lpDataDest;
860 memcpy(lpDataDest, lpProps->Value.MVszA.lppszA[i], ulStrLen);
861 lpDataDest += ulStrLen;
867 lpDataDest += lpProps->Value.MVszW.cValues * sizeof(WCHAR *);
869 for (i = 0; i < lpProps->Value.MVszW.cValues; i++)
871 ULONG ulStrLen = (strlenW(lpProps->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
873 lpDest->Value.MVszW.lppszW[i] = (LPWSTR)lpDataDest;
874 memcpy(lpDataDest, lpProps->Value.MVszW.lppszW[i], ulStrLen);
875 lpDataDest += ulStrLen;
881 lpDataDest += lpProps->Value.MVszW.cValues * sizeof(SBinary);
883 for (i = 0; i < lpProps->Value.MVszW.cValues; i++)
885 lpDest->Value.MVbin.lpbin[i].cb = lpProps->Value.MVbin.lpbin[i].cb;
886 lpDest->Value.MVbin.lpbin[i].lpb = (LPBYTE)lpDataDest;
887 memcpy(lpDataDest, lpProps->Value.MVbin.lpbin[i].lpb, lpDest->Value.MVbin.lpbin[i].cb);
888 lpDataDest += lpDest->Value.MVbin.lpbin[i].cb;
893 /* No embedded pointers, just copy the data over */
894 ulLen = UlPropSize(lpProps);
895 memcpy(lpDest->Value.MVi.lpi, lpProps->Value.MVi.lpi, ulLen);
906 *lpCount = lpDataDest - (char *)lpDst;
911 /*************************************************************************
912 * ScRelocProps@20 (MAPI32.172)
914 * Relocate the pointers in an array of property values after it has been copied.
917 * cValues [I] Number of properties in lpProps
918 * lpProps [O] Property array to relocate the pointers in.
919 * lpOld [I] Position where the data was copied from
920 * lpNew [I] Position where the data was copied to
921 * lpCount [O] If non-NULL, destination for the number of bytes of data at lpDst
924 * Success: S_OK. Any pointers in lpProps are relocated.
925 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
928 * MSDN states that this function can be used for serialisation by passing
929 * NULL as either lpOld or lpNew, thus converting any pointers in lpProps
930 * between offsets and pointers. This does not work in native (it crashes),
931 * and cannot be made to work in Wine because the original interface design
932 * is deficient. The only use left for this function is to remap pointers
933 * in a contiguous property array that has been copied with memcpy() to
934 * another memory location.
936 SCODE WINAPI ScRelocProps(int cValues, LPSPropValue lpProps, LPVOID lpOld,
937 LPVOID lpNew, ULONG *lpCount)
939 static const BOOL bBadPtr = TRUE; /* Windows bug - Assumes source is bad */
940 LPSPropValue lpDest = lpProps;
941 ULONG ulCount = cValues * sizeof(SPropValue);
945 TRACE("(%d,%p,%p,%p,%p)\n", cValues, lpProps, lpOld, lpNew, lpCount);
947 if (!lpProps || cValues < 0 || !lpOld || !lpNew)
948 return MAPI_E_INVALID_PARAMETER;
950 /* The reason native doesn't work as MSDN states is that it assumes that
951 * the lpProps pointer contains valid pointers. This is obviously not
952 * true if the array is being read back from serialisation (the pointers
953 * are just offsets). Native can't actually work converting the pointers to
954 * offsets either, because it converts any array pointers to offsets then
955 * _dereferences the offset_ in order to convert the array elements!
957 * The code below would handle both cases except that the design of this
958 * function makes it impossible to know when the pointers in lpProps are
959 * valid. If both lpOld and lpNew are non-NULL, native reads the pointers
960 * after converting them, so we must do the same. It seems this
961 * functionality was never tested by MS.
964 #define RELOC_PTR(p) (((char*)(p)) - (char*)lpOld + (char*)lpNew)
966 for (iter = 0; iter < cValues; iter++)
968 switch (PROP_TYPE(lpDest->ulPropTag))
971 lpDest->Value.lpguid = (LPGUID)RELOC_PTR(lpDest->Value.lpguid);
972 ulCount += sizeof(GUID);
975 ulLen = bBadPtr ? 0 : lstrlenA(lpDest->Value.lpszA) + 1u;
976 lpDest->Value.lpszA = RELOC_PTR(lpDest->Value.lpszA);
978 ulLen = lstrlenA(lpDest->Value.lpszA) + 1u;
982 ulLen = bBadPtr ? 0 : (lstrlenW(lpDest->Value.lpszW) + 1u) * sizeof(WCHAR);
983 lpDest->Value.lpszW = (LPWSTR)RELOC_PTR(lpDest->Value.lpszW);
985 ulLen = (strlenW(lpDest->Value.lpszW) + 1u) * sizeof(WCHAR);
989 lpDest->Value.bin.lpb = (LPBYTE)RELOC_PTR(lpDest->Value.bin.lpb);
990 ulCount += lpDest->Value.bin.cb;
993 if (lpDest->ulPropTag & MV_FLAG)
995 /* Since we have to access the array elements, don't map the
996 * array unless it is invalid (otherwise, map it at the end)
999 lpDest->Value.MVszA.lppszA = (LPSTR*)RELOC_PTR(lpDest->Value.MVszA.lppszA);
1001 switch (PROP_TYPE(lpProps->ulPropTag))
1005 ulCount += lpDest->Value.MVszA.cValues * sizeof(char *);
1007 for (i = 0; i < lpDest->Value.MVszA.cValues; i++)
1009 ULONG ulStrLen = bBadPtr ? 0 : lstrlenA(lpDest->Value.MVszA.lppszA[i]) + 1u;
1011 lpDest->Value.MVszA.lppszA[i] = RELOC_PTR(lpDest->Value.MVszA.lppszA[i]);
1013 ulStrLen = lstrlenA(lpDest->Value.MVszA.lppszA[i]) + 1u;
1014 ulCount += ulStrLen;
1020 ulCount += lpDest->Value.MVszW.cValues * sizeof(WCHAR *);
1022 for (i = 0; i < lpDest->Value.MVszW.cValues; i++)
1024 ULONG ulStrLen = bBadPtr ? 0 : (strlenW(lpDest->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
1026 lpDest->Value.MVszW.lppszW[i] = (LPWSTR)RELOC_PTR(lpDest->Value.MVszW.lppszW[i]);
1028 ulStrLen = (strlenW(lpDest->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
1029 ulCount += ulStrLen;
1035 ulCount += lpDest->Value.MVszW.cValues * sizeof(SBinary);
1037 for (i = 0; i < lpDest->Value.MVszW.cValues; i++)
1039 lpDest->Value.MVbin.lpbin[i].lpb = (LPBYTE)RELOC_PTR(lpDest->Value.MVbin.lpbin[i].lpb);
1040 ulCount += lpDest->Value.MVbin.lpbin[i].cb;
1045 ulCount += UlPropSize(lpDest);
1049 lpDest->Value.MVszA.lppszA = (LPSTR*)RELOC_PTR(lpDest->Value.MVszA.lppszA);
1061 /*************************************************************************
1062 * LpValFindProp@12 (MAPI32.173)
1064 * Find a property with a given property id in a property array.
1067 * ulPropTag [I] Property tag containing property id to find
1068 * cValues [I] Number of properties in lpProps
1069 * lpProps [I] Property array to search
1072 * A pointer to the matching property, or NULL if none was found.
1075 * This function matches only on the property id and does not care if the
1076 * property types differ.
1078 LPSPropValue WINAPI LpValFindProp(ULONG ulPropTag, ULONG cValues, LPSPropValue lpProps)
1080 TRACE("(%d,%d,%p)\n", ulPropTag, cValues, lpProps);
1082 if (lpProps && cValues)
1085 for (i = 0; i < cValues; i++)
1087 if (PROP_ID(ulPropTag) == PROP_ID(lpProps[i].ulPropTag))
1094 /*************************************************************************
1095 * ScDupPropset@16 (MAPI32.174)
1097 * Duplicate a property value array into a contiguous block of memory.
1100 * cValues [I] Number of properties in lpProps
1101 * lpProps [I] Property array to duplicate
1102 * lpAlloc [I] Memory allocation function, use MAPIAllocateBuffer()
1103 * lpNewProp [O] Destination for the newly duplicated property value array
1106 * Success: S_OK. *lpNewProp contains the duplicated array.
1107 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1108 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1110 SCODE WINAPI ScDupPropset(int cValues, LPSPropValue lpProps,
1111 LPALLOCATEBUFFER lpAlloc, LPSPropValue *lpNewProp)
1116 TRACE("(%d,%p,%p,%p)\n", cValues, lpProps, lpAlloc, lpNewProp);
1118 sc = ScCountProps(cValues, lpProps, &ulCount);
1121 sc = lpAlloc(ulCount, (LPVOID*)lpNewProp);
1123 sc = ScCopyProps(cValues, lpProps, *lpNewProp, &ulCount);
1128 /*************************************************************************
1129 * FBadRglpszA@8 (MAPI32.175)
1131 * Determine if an array of strings is invalid
1134 * lppszStrs [I] Array of strings to check
1135 * ulCount [I] Number of strings in lppszStrs
1138 * TRUE, if lppszStrs is invalid, FALSE otherwise.
1140 BOOL WINAPI FBadRglpszA(LPSTR *lppszStrs, ULONG ulCount)
1144 TRACE("(%p,%d)\n", lppszStrs, ulCount);
1149 if (!lppszStrs || IsBadReadPtr(lppszStrs, ulCount * sizeof(LPWSTR)))
1152 for (i = 0; i < ulCount; i++)
1154 if (!lppszStrs[i] || IsBadStringPtrA(lppszStrs[i], -1))
1160 /*************************************************************************
1161 * FBadRglpszW@8 (MAPI32.176)
1165 BOOL WINAPI FBadRglpszW(LPWSTR *lppszStrs, ULONG ulCount)
1169 TRACE("(%p,%d)\n", lppszStrs, ulCount);
1174 if (!lppszStrs || IsBadReadPtr(lppszStrs, ulCount * sizeof(LPWSTR)))
1177 for (i = 0; i < ulCount; i++)
1179 if (!lppszStrs[i] || IsBadStringPtrW(lppszStrs[i], -1))
1185 /*************************************************************************
1186 * FBadRowSet@4 (MAPI32.177)
1188 * Determine if a row is invalid
1191 * lpRow [I] Row to check
1194 * TRUE, if lpRow is invalid, FALSE otherwise.
1196 BOOL WINAPI FBadRowSet(LPSRowSet lpRowSet)
1199 TRACE("(%p)\n", lpRowSet);
1201 if (!lpRowSet || IsBadReadPtr(lpRowSet, CbSRowSet(lpRowSet)))
1204 for (i = 0; i < lpRowSet->cRows; i++)
1206 if (FBadRow(&lpRowSet->aRow[i]))
1212 /*************************************************************************
1213 * FBadPropTag@4 (MAPI32.179)
1215 * Determine if a property tag is invalid
1218 * ulPropTag [I] Property tag to check
1221 * TRUE, if ulPropTag is invalid, FALSE otherwise.
1223 ULONG WINAPI FBadPropTag(ULONG ulPropTag)
1225 TRACE("(0x%08x)\n", ulPropTag);
1227 switch (ulPropTag & (~MV_FLAG & PROP_TYPE_MASK))
1229 case PT_UNSPECIFIED:
1251 /*************************************************************************
1252 * FBadRow@4 (MAPI32.180)
1254 * Determine if a row is invalid
1257 * lpRow [I] Row to check
1260 * TRUE, if lpRow is invalid, FALSE otherwise.
1262 ULONG WINAPI FBadRow(LPSRow lpRow)
1265 TRACE("(%p)\n", lpRow);
1267 if (!lpRow || IsBadReadPtr(lpRow, sizeof(SRow)) || !lpRow->lpProps ||
1268 IsBadReadPtr(lpRow->lpProps, lpRow->cValues * sizeof(SPropValue)))
1271 for (i = 0; i < lpRow->cValues; i++)
1273 if (FBadProp(&lpRow->lpProps[i]))
1279 /*************************************************************************
1280 * FBadProp@4 (MAPI32.181)
1282 * Determine if a property is invalid
1285 * lpProp [I] Property to check
1288 * TRUE, if lpProp is invalid, FALSE otherwise.
1290 ULONG WINAPI FBadProp(LPSPropValue lpProp)
1292 if (!lpProp || IsBadReadPtr(lpProp, sizeof(SPropValue)) ||
1293 FBadPropTag(lpProp->ulPropTag))
1296 switch (PROP_TYPE(lpProp->ulPropTag))
1298 /* Single value properties containing pointers */
1300 if (!lpProp->Value.lpszA || IsBadStringPtrA(lpProp->Value.lpszA, -1))
1304 if (!lpProp->Value.lpszW || IsBadStringPtrW(lpProp->Value.lpszW, -1))
1308 if (IsBadReadPtr(lpProp->Value.bin.lpb, lpProp->Value.bin.cb))
1312 if (IsBadReadPtr(lpProp->Value.lpguid, sizeof(GUID)))
1316 /* Multiple value properties (arrays) containing no pointers */
1318 return PROP_BadArray(lpProp, sizeof(SHORT));
1320 return PROP_BadArray(lpProp, sizeof(LONG));
1321 case PT_MV_LONGLONG:
1322 return PROP_BadArray(lpProp, sizeof(LONG64));
1324 return PROP_BadArray(lpProp, sizeof(float));
1326 return PROP_BadArray(lpProp, sizeof(FILETIME));
1329 return PROP_BadArray(lpProp, sizeof(double));
1330 case PT_MV_CURRENCY:
1331 return PROP_BadArray(lpProp, sizeof(CY));
1333 return PROP_BadArray(lpProp, sizeof(GUID));
1335 /* Multiple value properties containing pointers */
1337 return FBadRglpszA(lpProp->Value.MVszA.lppszA,
1338 lpProp->Value.MVszA.cValues);
1340 return FBadRglpszW(lpProp->Value.MVszW.lppszW,
1341 lpProp->Value.MVszW.cValues);
1343 return FBadEntryList(&lpProp->Value.MVbin);
1348 /*************************************************************************
1349 * FBadColumnSet@4 (MAPI32.182)
1351 * Determine if an array of property tags is invalid
1354 * lpCols [I] Property tag array to check
1357 * TRUE, if lpCols is invalid, FALSE otherwise.
1359 ULONG WINAPI FBadColumnSet(LPSPropTagArray lpCols)
1361 ULONG ulRet = FALSE, i;
1363 TRACE("(%p)\n", lpCols);
1365 if (!lpCols || IsBadReadPtr(lpCols, CbSPropTagArray(lpCols)))
1369 for (i = 0; i < lpCols->cValues; i++)
1371 if ((lpCols->aulPropTag[i] & PROP_TYPE_MASK) == PT_ERROR ||
1372 FBadPropTag(lpCols->aulPropTag[i]))
1379 TRACE("Returning %s\n", ulRet ? "TRUE" : "FALSE");
1384 /**************************************************************************
1385 * IPropData {MAPI32}
1387 * A default Mapi interface to provide manipulation of object properties.
1390 * This object provides a default interface suitable in some cases as an
1391 * implementation of the IMAPIProp interface (which has no default
1392 * implementation). In addition to the IMAPIProp() methods inherited, this
1393 * interface allows read/write control over access to the object and its
1394 * individual properties.
1396 * To obtain the default implementation of this interface from Mapi, call
1402 /* A single property in a property data collection */
1406 ULONG ulAccess; /* The property value access level */
1407 LPSPropValue value; /* The property value */
1408 } IPropDataItem, *LPIPropDataItem;
1410 /* The main property data collection structure */
1413 IPropData IPropData_iface;
1414 LONG lRef; /* Reference count */
1415 ALLOCATEBUFFER *lpAlloc; /* Memory allocation routine */
1416 ALLOCATEMORE *lpMore; /* Linked memory allocation routine */
1417 FREEBUFFER *lpFree; /* Memory free routine */
1418 ULONG ulObjAccess; /* Object access level */
1419 ULONG ulNumValues; /* Number of items in values list */
1420 struct list values; /* List of property values */
1421 CRITICAL_SECTION cs; /* Lock for thread safety */
1424 static inline IPropDataImpl *impl_from_IPropData(IPropData *iface)
1426 return CONTAINING_RECORD(iface, IPropDataImpl, IPropData_iface);
1429 /* Internal - Get a property value, assumes lock is held */
1430 static IPropDataItem *IMAPIPROP_GetValue(IPropDataImpl *This, ULONG ulPropTag)
1432 struct list *cursor;
1434 LIST_FOR_EACH(cursor, &This->values)
1436 LPIPropDataItem current = LIST_ENTRY(cursor, IPropDataItem, entry);
1437 /* Note that property types don't have to match, just Id's */
1438 if (PROP_ID(current->value->ulPropTag) == PROP_ID(ulPropTag))
1444 /* Internal - Add a new property value, assumes lock is held */
1445 static IPropDataItem *IMAPIPROP_AddValue(IPropDataImpl *This,
1446 LPSPropValue lpProp)
1449 LPIPropDataItem lpNew;
1452 hRet = This->lpAlloc(sizeof(IPropDataItem), &lpMem);
1454 if (SUCCEEDED(hRet))
1457 lpNew->ulAccess = IPROP_READWRITE;
1459 /* Allocate the value separately so we can update it easily */
1461 hRet = This->lpAlloc(sizeof(SPropValue), &lpMem);
1462 if (SUCCEEDED(hRet))
1464 lpNew->value = lpMem;
1466 hRet = PropCopyMore(lpNew->value, lpProp, This->lpMore, lpMem);
1467 if (SUCCEEDED(hRet))
1469 list_add_tail(&This->values, &lpNew->entry);
1470 This->ulNumValues++;
1473 This->lpFree(lpNew->value);
1475 This->lpFree(lpNew);
1480 /* Internal - Lock an IPropData object */
1481 static inline void IMAPIPROP_Lock(IPropDataImpl *This)
1483 EnterCriticalSection(&This->cs);
1486 /* Internal - Unlock an IPropData object */
1487 static inline void IMAPIPROP_Unlock(IPropDataImpl *This)
1489 LeaveCriticalSection(&This->cs);
1492 /* This one seems to be missing from mapidefs.h */
1493 #define CbNewSPropProblemArray(c) \
1494 (offsetof(SPropProblemArray,aProblem)+(c)*sizeof(SPropProblem))
1496 /**************************************************************************
1497 * IPropData_QueryInterface {MAPI32}
1499 * Inherited method from the IUnknown Interface.
1500 * See IUnknown_QueryInterface.
1502 static WINAPI HRESULT IPropData_fnQueryInterface(LPPROPDATA iface, REFIID riid, LPVOID *ppvObj)
1504 IPropDataImpl *This = impl_from_IPropData(iface);
1506 TRACE("(%p,%s,%p)\n", This, debugstr_guid(riid), ppvObj);
1508 if (!ppvObj || !riid)
1509 return MAPI_E_INVALID_PARAMETER;
1513 if(IsEqualIID(riid, &IID_IUnknown) ||
1514 IsEqualIID(riid, &IID_IMAPIProp) ||
1515 IsEqualIID(riid, &IID_IMAPIPropData))
1518 IPropData_AddRef(iface);
1519 TRACE("returning %p\n", *ppvObj);
1523 TRACE("returning E_NOINTERFACE\n");
1524 return MAPI_E_INTERFACE_NOT_SUPPORTED;
1527 /**************************************************************************
1528 * IPropData_AddRef {MAPI32}
1530 * Inherited method from the IUnknown Interface.
1531 * See IUnknown_AddRef.
1533 static ULONG WINAPI IPropData_fnAddRef(LPPROPDATA iface)
1535 IPropDataImpl *This = impl_from_IPropData(iface);
1537 TRACE("(%p)->(count before=%u)\n", This, This->lRef);
1539 return InterlockedIncrement(&This->lRef);
1542 /**************************************************************************
1543 * IPropData_Release {MAPI32}
1545 * Inherited method from the IUnknown Interface.
1546 * See IUnknown_Release.
1548 static ULONG WINAPI IPropData_fnRelease(LPPROPDATA iface)
1550 IPropDataImpl *This = impl_from_IPropData(iface);
1553 TRACE("(%p)->(count before=%u)\n", This, This->lRef);
1555 lRef = InterlockedDecrement(&This->lRef);
1558 TRACE("Destroying IPropData (%p)\n",This);
1560 /* Note: No need to lock, since no other thread is referencing iface */
1561 while (!list_empty(&This->values))
1563 struct list *head = list_head(&This->values);
1564 LPIPropDataItem current = LIST_ENTRY(head, IPropDataItem, entry);
1566 This->lpFree(current->value);
1567 This->lpFree(current);
1569 This->cs.DebugInfo->Spare[0] = 0;
1570 DeleteCriticalSection(&This->cs);
1576 /**************************************************************************
1577 * IPropData_GetLastError {MAPI32}
1579 * Get information about the last error that occurred in an IMAPIProp object.
1582 * iface [I] IMAPIProp object that experienced the error
1583 * hRes [I] Result of the call that returned an error
1584 * ulFlags [I] 0=return Ascii strings, MAPI_UNICODE=return Unicode strings
1585 * lppError [O] Destination for detailed error information
1588 * Success: S_OK. *lppError contains details about the last error.
1589 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1590 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1593 * - If this function succeeds, the returned information in *lppError must be
1594 * freed using MAPIFreeBuffer() once the caller is finished with it.
1595 * - It is possible for this function to succeed and set *lppError to NULL,
1596 * if there is no further information to report about hRes.
1598 static HRESULT WINAPI IPropData_fnGetLastError(LPPROPDATA iface, HRESULT hRes, ULONG ulFlags,
1599 LPMAPIERROR *lppError)
1601 TRACE("(%p,0x%08X,0x%08X,%p)\n", iface, hRes, ulFlags, lppError);
1603 if (!lppError || SUCCEEDED(hRes) || (ulFlags & ~MAPI_UNICODE))
1604 return MAPI_E_INVALID_PARAMETER;
1610 /**************************************************************************
1611 * IPropData_SaveChanges {MAPI32}
1613 * Update any changes made to a transactional IMAPIProp object.
1616 * iface [I] IMAPIProp object to update
1617 * ulFlags [I] Flags controlling the update.
1620 * Success: S_OK. Any outstanding changes are committed to the object.
1621 * Failure: An HRESULT error code describing the error.
1623 static HRESULT WINAPI IPropData_fnSaveChanges(LPPROPDATA iface, ULONG ulFlags)
1625 TRACE("(%p,0x%08X)\n", iface, ulFlags);
1627 /* Since this object is not transacted we do not need to implement this */
1628 /* FIXME: Should we set the access levels to clean? */
1632 /**************************************************************************
1633 * IPropData_GetProps {MAPI32}
1635 * Get property values from an IMAPIProp object.
1638 * iface [I] IMAPIProp object to get the property values from
1639 * lpTags [I] Property tage of property values to be retrieved
1640 * ulFlags [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1642 * lpCount [O] Destination for number of properties returned
1643 * lppProps [O] Destination for returned property values
1646 * Success: S_OK. *lppProps and *lpCount are updated.
1647 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1648 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
1649 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
1652 * - If MAPI_W_ERRORS_RETURNED is returned, any properties that could not be
1653 * retrieved from iface are present in lppProps with their type
1654 * changed to PT_ERROR and Id unchanged.
1656 static HRESULT WINAPI IPropData_fnGetProps(LPPROPDATA iface, LPSPropTagArray lpTags, ULONG ulFlags,
1657 ULONG *lpCount, LPSPropValue *lppProps)
1659 IPropDataImpl *This = impl_from_IPropData(iface);
1661 HRESULT hRet = S_OK;
1663 TRACE("(%p,%p,0x%08x,%p,%p) stub\n", iface, lpTags, ulFlags,
1666 if (!iface || ulFlags & ~MAPI_UNICODE || !lpTags || *lpCount || !lppProps)
1667 return MAPI_E_INVALID_PARAMETER;
1669 FIXME("semi-stub, flags not supported\n");
1671 *lpCount = lpTags->cValues;
1676 hRet = MAPIAllocateBuffer(*lpCount * sizeof(SPropValue), (LPVOID*)lppProps);
1680 IMAPIPROP_Lock(This);
1682 for (i = 0; i < lpTags->cValues; i++)
1684 HRESULT hRetTmp = E_INVALIDARG;
1685 LPIPropDataItem item;
1687 item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
1690 hRetTmp = PropCopyMore(&(*lppProps)[i], item->value,
1691 This->lpMore, *lppProps);
1692 if (FAILED(hRetTmp))
1694 hRet = MAPI_W_ERRORS_RETURNED;
1695 (*lppProps)[i].ulPropTag =
1696 CHANGE_PROP_TYPE(lpTags->aulPropTag[i], PT_ERROR);
1700 IMAPIPROP_Unlock(This);
1705 /**************************************************************************
1706 * MAPIProp_GetPropList {MAPI32}
1708 * Get the list of property tags for all values in an IMAPIProp object.
1711 * iface [I] IMAPIProp object to get the property tag list from
1712 * ulFlags [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1714 * lppTags [O] Destination for the retrieved property tag list
1717 * Success: S_OK. *lppTags contains the tags for all available properties.
1718 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1719 * MAPI_E_BAD_CHARWIDTH, if Ascii or Unicode strings are requested
1720 * and that type of string is not supported.
1722 static HRESULT WINAPI IPropData_fnGetPropList(LPPROPDATA iface, ULONG ulFlags,
1723 LPSPropTagArray *lppTags)
1725 IPropDataImpl *This = impl_from_IPropData(iface);
1729 TRACE("(%p,0x%08x,%p) stub\n", iface, ulFlags, lppTags);
1731 if (!iface || ulFlags & ~MAPI_UNICODE || !lppTags)
1732 return MAPI_E_INVALID_PARAMETER;
1734 FIXME("semi-stub, flags not supported\n");
1738 IMAPIPROP_Lock(This);
1740 hRet = MAPIAllocateBuffer(CbNewSPropTagArray(This->ulNumValues),
1742 if (SUCCEEDED(hRet))
1744 struct list *cursor;
1747 LIST_FOR_EACH(cursor, &This->values)
1749 LPIPropDataItem current = LIST_ENTRY(cursor, IPropDataItem, entry);
1750 (*lppTags)->aulPropTag[i] = current->value->ulPropTag;
1753 (*lppTags)->cValues = This->ulNumValues;
1756 IMAPIPROP_Unlock(This);
1760 /**************************************************************************
1761 * IPropData_OpenProperty {MAPI32}
1763 * Not documented at this time.
1766 * An HRESULT success/failure code.
1768 static HRESULT WINAPI IPropData_fnOpenProperty(LPPROPDATA iface, ULONG ulPropTag, LPCIID iid,
1769 ULONG ulOpts, ULONG ulFlags, LPUNKNOWN *lpUnk)
1771 FIXME("(%p,%u,%s,%u,0x%08x,%p) stub\n", iface, ulPropTag,
1772 debugstr_guid(iid), ulOpts, ulFlags, lpUnk);
1773 return MAPI_E_NO_SUPPORT;
1777 /**************************************************************************
1778 * IPropData_SetProps {MAPI32}
1780 * Add or edit the property values in an IMAPIProp object.
1783 * iface [I] IMAPIProp object to get the property tag list from
1784 * ulValues [I] Number of properties in lpProps
1785 * lpProps [I] Property values to set
1786 * lppProbs [O] Optional destination for any problems that occurred
1789 * Success: S_OK. The properties in lpProps are added to iface if they don't
1790 * exist, or changed to the values in lpProps if they do
1791 * Failure: An HRESULT error code describing the error
1793 static HRESULT WINAPI IPropData_fnSetProps(LPPROPDATA iface, ULONG ulValues, LPSPropValue lpProps,
1794 LPSPropProblemArray *lppProbs)
1796 IPropDataImpl *This = impl_from_IPropData(iface);
1797 HRESULT hRet = S_OK;
1800 TRACE("(%p,%u,%p,%p)\n", iface, ulValues, lpProps, lppProbs);
1802 if (!iface || !lpProps)
1803 return MAPI_E_INVALID_PARAMETER;
1805 for (i = 0; i < ulValues; i++)
1807 if (FBadProp(&lpProps[i]) ||
1808 PROP_TYPE(lpProps[i].ulPropTag) == PT_OBJECT ||
1809 PROP_TYPE(lpProps[i].ulPropTag) == PT_NULL)
1810 return MAPI_E_INVALID_PARAMETER;
1813 IMAPIPROP_Lock(This);
1815 /* FIXME: Under what circumstances is lpProbs created? */
1816 for (i = 0; i < ulValues; i++)
1818 LPIPropDataItem item = IMAPIPROP_GetValue(This, lpProps[i].ulPropTag);
1823 LPVOID lpMem = NULL;
1825 /* Found, so update the existing value */
1826 if (item->value->ulPropTag != lpProps[i].ulPropTag)
1827 FIXME("semi-stub, overwriting type (not coercing)\n");
1829 hRetTmp = This->lpAlloc(sizeof(SPropValue), &lpMem);
1830 if (SUCCEEDED(hRetTmp))
1832 hRetTmp = PropCopyMore(lpMem, &lpProps[i], This->lpMore, lpMem);
1833 if (SUCCEEDED(hRetTmp))
1835 This->lpFree(item->value);
1836 item->value = lpMem;
1839 This->lpFree(lpMem);
1846 if (!IMAPIPROP_AddValue(This, &lpProps[i]))
1847 hRet = MAPI_E_NOT_ENOUGH_MEMORY;
1851 IMAPIPROP_Unlock(This);
1855 /**************************************************************************
1856 * IPropData_DeleteProps {MAPI32}
1858 * Delete one or more property values from an IMAPIProp object.
1861 * iface [I] IMAPIProp object to remove property values from.
1862 * lpTags [I] Collection of property Id's to remove from iface.
1863 * lppProbs [O] Destination for problems encountered, if any.
1866 * Success: S_OK. Any properties in iface matching property Id's in lpTags have
1867 * been deleted. If lppProbs is non-NULL it contains details of any
1868 * errors that occurred.
1869 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1870 * E_ACCESSDENIED, if this object was created using CreateIProp() and
1871 * a subsequent call to IPropData_SetObjAccess() was made specifying
1872 * IPROP_READONLY as the access type.
1875 * - lppProbs will not be populated for cases where a property Id is present
1876 * in lpTags but not in iface.
1877 * - lppProbs should be deleted with MAPIFreeBuffer() if returned.
1879 static HRESULT WINAPI IPropData_fnDeleteProps(LPPROPDATA iface, LPSPropTagArray lpTags,
1880 LPSPropProblemArray *lppProbs)
1882 IPropDataImpl *This = impl_from_IPropData(iface);
1883 ULONG i, numProbs = 0;
1884 HRESULT hRet = S_OK;
1886 TRACE("(%p,%p,%p)\n", iface, lpTags, lppProbs);
1888 if (!iface || !lpTags)
1889 return MAPI_E_INVALID_PARAMETER;
1894 for (i = 0; i < lpTags->cValues; i++)
1896 if (FBadPropTag(lpTags->aulPropTag[i]) ||
1897 PROP_TYPE(lpTags->aulPropTag[i]) == PT_OBJECT ||
1898 PROP_TYPE(lpTags->aulPropTag[i]) == PT_NULL)
1899 return MAPI_E_INVALID_PARAMETER;
1902 IMAPIPROP_Lock(This);
1904 if (This->ulObjAccess != IPROP_READWRITE)
1906 IMAPIPROP_Unlock(This);
1907 return E_ACCESSDENIED;
1910 for (i = 0; i < lpTags->cValues; i++)
1912 LPIPropDataItem item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
1916 if (item->ulAccess & IPROP_READWRITE)
1918 /* Everything hunky-dory, remove the item */
1919 list_remove(&item->entry);
1920 This->lpFree(item->value); /* Also frees value pointers */
1922 This->ulNumValues--;
1926 /* Can't write the value. Create/populate problems array */
1929 /* Create problems array */
1930 ULONG ulSize = CbNewSPropProblemArray(lpTags->cValues - i);
1931 HRESULT hRetTmp = MAPIAllocateBuffer(ulSize, (LPVOID*)lppProbs);
1932 if (FAILED(hRetTmp))
1937 LPSPropProblem lpProb = &(*lppProbs)->aProblem[numProbs];
1938 lpProb->ulIndex = i;
1939 lpProb->ulPropTag = lpTags->aulPropTag[i];
1940 lpProb->scode = E_ACCESSDENIED;
1946 if (lppProbs && *lppProbs)
1947 (*lppProbs)->cProblem = numProbs;
1949 IMAPIPROP_Unlock(This);
1954 /**************************************************************************
1955 * IPropData_CopyTo {MAPI32}
1957 * Not documented at this time.
1960 * An HRESULT success/failure code.
1962 static HRESULT WINAPI IPropData_fnCopyTo(LPPROPDATA iface, ULONG niids, LPCIID lpiidExcl,
1963 LPSPropTagArray lpPropsExcl, ULONG ulParam,
1964 LPMAPIPROGRESS lpIProgress, LPCIID lpIfaceIid,
1965 LPVOID lpDstObj, ULONG ulFlags,
1966 LPSPropProblemArray *lppProbs)
1968 FIXME("(%p,%u,%p,%p,%x,%p,%s,%p,0x%08X,%p) stub\n", iface, niids,
1969 lpiidExcl, lpPropsExcl, ulParam, lpIProgress,
1970 debugstr_guid(lpIfaceIid), lpDstObj, ulFlags, lppProbs);
1971 return MAPI_E_NO_SUPPORT;
1974 /**************************************************************************
1975 * IPropData_CopyProps {MAPI32}
1977 * Not documented at this time.
1980 * An HRESULT success/failure code.
1982 static HRESULT WINAPI IPropData_fnCopyProps(LPPROPDATA iface, LPSPropTagArray lpInclProps,
1983 ULONG ulParam, LPMAPIPROGRESS lpIProgress,
1984 LPCIID lpIface, LPVOID lpDstObj, ULONG ulFlags,
1985 LPSPropProblemArray *lppProbs)
1987 FIXME("(%p,%p,%x,%p,%s,%p,0x%08X,%p) stub\n", iface, lpInclProps,
1988 ulParam, lpIProgress, debugstr_guid(lpIface), lpDstObj, ulFlags,
1990 return MAPI_E_NO_SUPPORT;
1993 /**************************************************************************
1994 * IPropData_GetNamesFromIDs {MAPI32}
1996 * Get the names of properties from their identifiers.
1999 * iface [I] IMAPIProp object to operate on
2000 * lppPropTags [I/O] Property identifiers to get the names for, or NULL to
2002 * iid [I] Property set identifier, or NULL
2003 * ulFlags [I] MAPI_NO_IDS=Don't return numeric named properties,
2004 * or MAPI_NO_STRINGS=Don't return strings
2005 * lpCount [O] Destination for number of properties returned
2006 * lpppNames [O] Destination for returned names
2009 * Success: S_OK. *lppPropTags and lpppNames contain the returned
2011 * Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2012 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2013 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2016 static HRESULT WINAPI IPropData_fnGetNamesFromIDs(LPPROPDATA iface, LPSPropTagArray *lppPropTags,
2017 LPGUID iid, ULONG ulFlags, ULONG *lpCount,
2018 LPMAPINAMEID **lpppNames)
2020 FIXME("(%p,%p,%s,0x%08X,%p,%p) stub\n", iface, lppPropTags,
2021 debugstr_guid(iid), ulFlags, lpCount, lpppNames);
2022 return MAPI_E_NO_SUPPORT;
2025 /**************************************************************************
2026 * IPropData_GetIDsFromNames {MAPI32}
2028 * Get property identifiers associated with one or more named properties.
2031 * iface [I] IMAPIProp object to operate on
2032 * ulNames [I] Number of names in lppNames
2033 * lppNames [I] Names to query or create, or NULL to query all names
2034 * ulFlags [I] Pass MAPI_CREATE to create new named properties
2035 * lppPropTags [O] Destination for queried or created property identifiers
2038 * Success: S_OK. *lppPropTags contains the property tags created or requested.
2039 * Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2040 * MAPI_E_TOO_BIG, if the object cannot process the number of
2041 * properties involved.
2042 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2043 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2046 static HRESULT WINAPI IPropData_fnGetIDsFromNames(LPPROPDATA iface, ULONG ulNames,
2047 LPMAPINAMEID *lppNames, ULONG ulFlags,
2048 LPSPropTagArray *lppPropTags)
2050 FIXME("(%p,%d,%p,0x%08X,%p) stub\n",
2051 iface, ulNames, lppNames, ulFlags, lppPropTags);
2052 return MAPI_E_NO_SUPPORT;
2055 /**************************************************************************
2056 * IPropData_HrSetObjAccess {MAPI32}
2058 * Set the access level of an IPropData object.
2061 * iface [I] IPropData object to set the access on
2062 * ulAccess [I] Either IPROP_READONLY or IPROP_READWRITE for read or
2063 * read/write access respectively.
2066 * Success: S_OK. The objects access level is changed.
2067 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2069 static HRESULT WINAPI
2070 IPropData_fnHrSetObjAccess(LPPROPDATA iface, ULONG ulAccess)
2072 IPropDataImpl *This = impl_from_IPropData(iface);
2074 TRACE("(%p,%x)\n", iface, ulAccess);
2076 if (!iface || ulAccess < IPROP_READONLY || ulAccess > IPROP_READWRITE)
2077 return MAPI_E_INVALID_PARAMETER;
2079 IMAPIPROP_Lock(This);
2081 This->ulObjAccess = ulAccess;
2083 IMAPIPROP_Unlock(This);
2087 /* Internal - determine if an access value is bad */
2088 static inline BOOL PROP_IsBadAccess(ULONG ulAccess)
2092 case IPROP_READONLY|IPROP_CLEAN:
2093 case IPROP_READONLY|IPROP_DIRTY:
2094 case IPROP_READWRITE|IPROP_CLEAN:
2095 case IPROP_READWRITE|IPROP_DIRTY:
2101 /**************************************************************************
2102 * IPropData_HrSetPropAccess {MAPI32}
2104 * Set the access levels for a group of property values in an IPropData object.
2107 * iface [I] IPropData object to set access levels in.
2108 * lpTags [I] List of property Id's to set access for.
2109 * lpAccess [O] Access level for each property in lpTags.
2112 * Success: S_OK. The access level of each property value in lpTags that is
2113 * present in iface is changed.
2114 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2117 * - Each access level in lpAccess must contain at least one of IPROP_READONLY
2118 * or IPROP_READWRITE, but not both, and also IPROP_CLEAN or IPROP_DIRTY,
2119 * but not both. No other bits should be set.
2120 * - If a property Id in lpTags is not present in iface, it is ignored.
2122 static HRESULT WINAPI
2123 IPropData_fnHrSetPropAccess(LPPROPDATA iface, LPSPropTagArray lpTags,
2126 IPropDataImpl *This = impl_from_IPropData(iface);
2129 TRACE("(%p,%p,%p)\n", iface, lpTags, lpAccess);
2131 if (!iface || !lpTags || !lpAccess)
2132 return MAPI_E_INVALID_PARAMETER;
2134 for (i = 0; i < lpTags->cValues; i++)
2136 if (FBadPropTag(lpTags->aulPropTag[i]) || PROP_IsBadAccess(lpAccess[i]))
2137 return MAPI_E_INVALID_PARAMETER;
2140 IMAPIPROP_Lock(This);
2142 for (i = 0; i < lpTags->cValues; i++)
2144 LPIPropDataItem item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
2147 item->ulAccess = lpAccess[i];
2150 IMAPIPROP_Unlock(This);
2154 /**************************************************************************
2155 * IPropData_HrGetPropAccess {MAPI32}
2157 * Get the access levels for a group of property values in an IPropData object.
2160 * iface [I] IPropData object to get access levels from.
2161 * lppTags [O] Destination for the list of property Id's in iface.
2162 * lppAccess [O] Destination for access level for each property in lppTags.
2165 * Success: S_OK. lppTags and lppAccess contain the property Id's and the
2166 * Access level of each property value in iface.
2167 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid, or
2168 * MAPI_E_NOT_ENOUGH_MEMORY if memory allocation fails.
2171 * - *lppTags and *lppAccess should be freed with MAPIFreeBuffer() by the caller.
2173 static HRESULT WINAPI
2174 IPropData_fnHrGetPropAccess(LPPROPDATA iface, LPSPropTagArray *lppTags,
2177 IPropDataImpl *This = impl_from_IPropData(iface);
2182 TRACE("(%p,%p,%p) stub\n", iface, lppTags, lppAccess);
2184 if (!iface || !lppTags || !lppAccess)
2185 return MAPI_E_INVALID_PARAMETER;
2190 IMAPIPROP_Lock(This);
2192 hRet = This->lpAlloc(CbNewSPropTagArray(This->ulNumValues), &lpMem);
2193 if (SUCCEEDED(hRet))
2197 hRet = This->lpAlloc(This->ulNumValues * sizeof(ULONG), &lpMem);
2198 if (SUCCEEDED(hRet))
2200 struct list *cursor;
2203 (*lppTags)->cValues = This->ulNumValues;
2206 LIST_FOR_EACH(cursor, &This->values)
2208 LPIPropDataItem item = LIST_ENTRY(cursor, IPropDataItem, entry);
2209 (*lppTags)->aulPropTag[i] = item->value->ulPropTag;
2210 (*lppAccess)[i] = item->ulAccess;
2213 IMAPIPROP_Unlock(This);
2216 This->lpFree(*lppTags);
2219 IMAPIPROP_Unlock(This);
2220 return MAPI_E_NOT_ENOUGH_MEMORY;
2223 /**************************************************************************
2224 * IPropData_HrAddObjProps {MAPI32}
2226 * Not documented at this time.
2229 * An HRESULT success/failure code.
2231 static HRESULT WINAPI
2232 IPropData_fnHrAddObjProps(LPPROPDATA iface, LPSPropTagArray lpTags,
2233 LPSPropProblemArray *lppProbs)
2238 LPSPropValue lpValues;
2241 FIXME("(%p,%p,%p) stub\n", iface, lpTags, lppProbs);
2243 if (!iface || !lpTags)
2244 return MAPI_E_INVALID_PARAMETER;
2246 /* FIXME: Below is the obvious implementation, adding all the properties
2247 * in lpTags to the object. However, it doesn't appear that this
2248 * is what this function does.
2252 if (!lpTags->cValues)
2255 lpValues = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
2256 lpTags->cValues * sizeof(SPropValue));
2258 return MAPI_E_NOT_ENOUGH_MEMORY;
2260 for (i = 0; i < lpTags->cValues; i++)
2261 lpValues[i].ulPropTag = lpTags->aulPropTag[i];
2263 hRet = IPropData_SetProps(iface, lpTags->cValues, lpValues, lppProbs);
2264 HeapFree(GetProcessHeap(), 0, lpValues);
2269 static const IPropDataVtbl IPropDataImpl_vtbl =
2271 IPropData_fnQueryInterface,
2273 IPropData_fnRelease,
2274 IPropData_fnGetLastError,
2275 IPropData_fnSaveChanges,
2276 IPropData_fnGetProps,
2277 IPropData_fnGetPropList,
2278 IPropData_fnOpenProperty,
2279 IPropData_fnSetProps,
2280 IPropData_fnDeleteProps,
2282 IPropData_fnCopyProps,
2283 IPropData_fnGetNamesFromIDs,
2284 IPropData_fnGetIDsFromNames,
2285 IPropData_fnHrSetObjAccess,
2286 IPropData_fnHrSetPropAccess,
2287 IPropData_fnHrGetPropAccess,
2288 IPropData_fnHrAddObjProps
2291 /*************************************************************************
2292 * CreateIProp@24 (MAPI32.60)
2294 * Create an IPropData object.
2297 * iid [I] GUID of the object to create. Use &IID_IMAPIPropData or NULL
2298 * lpAlloc [I] Memory allocation function. Use MAPIAllocateBuffer()
2299 * lpMore [I] Linked memory allocation function. Use MAPIAllocateMore()
2300 * lpFree [I] Memory free function. Use MAPIFreeBuffer()
2301 * lpReserved [I] Reserved, set to NULL
2302 * lppPropData [O] Destination for created IPropData object
2305 * Success: S_OK. *lppPropData contains the newly created object.
2306 * Failure: MAPI_E_INTERFACE_NOT_SUPPORTED, if iid is non-NULL and not supported,
2307 * MAPI_E_INVALID_PARAMETER, if any parameter is invalid
2309 SCODE WINAPI CreateIProp(LPCIID iid, ALLOCATEBUFFER *lpAlloc,
2310 ALLOCATEMORE *lpMore, FREEBUFFER *lpFree,
2311 LPVOID lpReserved, LPPROPDATA *lppPropData)
2313 IPropDataImpl *lpPropData;
2316 TRACE("(%s,%p,%p,%p,%p,%p)\n", debugstr_guid(iid), lpAlloc, lpMore, lpFree,
2317 lpReserved, lppPropData);
2320 *lppPropData = NULL;
2322 if (iid && !IsEqualGUID(iid, &IID_IMAPIPropData))
2323 return MAPI_E_INTERFACE_NOT_SUPPORTED;
2325 if (!lpAlloc || !lpMore || !lpFree || lpReserved || !lppPropData)
2326 return MAPI_E_INVALID_PARAMETER;
2328 scode = lpAlloc(sizeof(IPropDataImpl), (LPVOID*)&lpPropData);
2330 if (SUCCEEDED(scode))
2332 lpPropData->IPropData_iface.lpVtbl = &IPropDataImpl_vtbl;
2333 lpPropData->lRef = 1;
2334 lpPropData->lpAlloc = lpAlloc;
2335 lpPropData->lpMore = lpMore;
2336 lpPropData->lpFree = lpFree;
2337 lpPropData->ulObjAccess = IPROP_READWRITE;
2338 lpPropData->ulNumValues = 0;
2339 list_init(&lpPropData->values);
2340 InitializeCriticalSection(&lpPropData->cs);
2341 lpPropData->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": IPropDataImpl.cs");
2342 *lppPropData = &lpPropData->IPropData_iface;