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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 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
);
39 BOOL WINAPI
FBadRglpszW(LPWSTR
*,ULONG
);
41 /* Internal: Check if a property value array is invalid */
42 static inline ULONG
PROP_BadArray(LPSPropValue lpProp
, size_t elemSize
)
44 return IsBadReadPtr(lpProp
->Value
.MVi
.lpi
, lpProp
->Value
.MVi
.cValues
* elemSize
);
47 /*************************************************************************
48 * PropCopyMore@16 (MAPI32.76)
50 * Copy a property value.
53 * lpDest [O] Destination for the copied value
54 * lpSrc [I] Property value to copy to lpDest
55 * lpMore [I] Linked memory allocation function (pass MAPIAllocateMore())
56 * lpOrig [I] Original allocation to which memory will be linked
59 * Success: S_OK. lpDest contains a deep copy of lpSrc.
60 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
61 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
64 * Any elements within the property returned should not be individually
65 * freed, as they will be freed when lpOrig is.
67 SCODE WINAPI
PropCopyMore(LPSPropValue lpDest
, LPSPropValue lpSrc
,
68 ALLOCATEMORE
*lpMore
, LPVOID lpOrig
)
73 TRACE("(%p,%p,%p,%p)\n", lpDest
, lpSrc
, lpMore
, lpOrig
);
75 if (!lpDest
|| IsBadWritePtr(lpDest
, sizeof(SPropValue
)) ||
76 FBadProp(lpSrc
) || !lpMore
)
77 return MAPI_E_INVALID_PARAMETER
;
79 /* Shallow copy first, this is sufficient for properties without pointers */
82 switch (PROP_TYPE(lpSrc
->ulPropTag
))
85 scode
= lpMore(sizeof(GUID
), lpOrig
, (LPVOID
*)&lpDest
->Value
.lpguid
);
87 memcpy(lpDest
->Value
.lpguid
, lpSrc
->Value
.lpguid
, sizeof(GUID
));
90 ulLen
= lstrlenA(lpSrc
->Value
.lpszA
) + 1u;
91 scode
= lpMore(ulLen
, lpOrig
, (LPVOID
*)&lpDest
->Value
.lpszA
);
93 memcpy(lpDest
->Value
.lpszA
, lpSrc
->Value
.lpszA
, ulLen
);
96 ulLen
= (strlenW(lpSrc
->Value
.lpszW
) + 1u) * sizeof(WCHAR
);
97 scode
= lpMore(ulLen
, lpOrig
, (LPVOID
*)&lpDest
->Value
.lpszW
);
99 memcpy(lpDest
->Value
.lpszW
, lpSrc
->Value
.lpszW
, ulLen
);
102 scode
= lpMore(lpSrc
->Value
.bin
.cb
, lpOrig
, (LPVOID
*)&lpDest
->Value
.bin
.lpb
);
103 if (SUCCEEDED(scode
))
104 memcpy(lpDest
->Value
.bin
.lpb
, lpSrc
->Value
.bin
.lpb
, lpSrc
->Value
.bin
.cb
);
107 if (lpSrc
->ulPropTag
& MV_FLAG
)
109 ulLen
= UlPropSize(lpSrc
);
111 if (PROP_TYPE(lpSrc
->ulPropTag
) == PT_MV_STRING8
||
112 PROP_TYPE(lpSrc
->ulPropTag
) == PT_MV_UNICODE
)
114 /* UlPropSize doesn't account for the string pointers */
115 ulLen
+= lpSrc
->Value
.MVszA
.cValues
* sizeof(char*);
117 else if (PROP_TYPE(lpSrc
->ulPropTag
) == PT_MV_BINARY
)
119 /* UlPropSize doesn't account for the SBinary structs */
120 ulLen
+= lpSrc
->Value
.MVbin
.cValues
* sizeof(SBinary
);
123 lpDest
->Value
.MVi
.cValues
= lpSrc
->Value
.MVi
.cValues
;
124 scode
= lpMore(ulLen
, lpOrig
, (LPVOID
*)&lpDest
->Value
.MVi
.lpi
);
128 /* Note that we could allocate the memory for each value in a
129 * multi-value property seperately, however if an allocation failed
130 * we would be left with a bunch of allocated memory, which (while
131 * not really leaked) is unusable until lpOrig is freed. So for
132 * strings and binary arrays we make a single allocation for all
133 * of the data. This is consistent since individual elements can't
137 switch (PROP_TYPE(lpSrc
->ulPropTag
))
141 char *lpNextStr
= (char*)(lpDest
->Value
.MVszA
.lppszA
+
142 lpDest
->Value
.MVszA
.cValues
);
144 for (i
= 0; i
< lpSrc
->Value
.MVszA
.cValues
; i
++)
146 ULONG ulStrLen
= lstrlenA(lpSrc
->Value
.MVszA
.lppszA
[i
]) + 1u;
148 lpDest
->Value
.MVszA
.lppszA
[i
] = lpNextStr
;
149 memcpy(lpNextStr
, lpSrc
->Value
.MVszA
.lppszA
[i
], ulStrLen
);
150 lpNextStr
+= ulStrLen
;
156 WCHAR
*lpNextStr
= (WCHAR
*)(lpDest
->Value
.MVszW
.lppszW
+
157 lpDest
->Value
.MVszW
.cValues
);
159 for (i
= 0; i
< lpSrc
->Value
.MVszW
.cValues
; i
++)
161 ULONG ulStrLen
= strlenW(lpSrc
->Value
.MVszW
.lppszW
[i
]) + 1u;
163 lpDest
->Value
.MVszW
.lppszW
[i
] = lpNextStr
;
164 memcpy(lpNextStr
, lpSrc
->Value
.MVszW
.lppszW
[i
], ulStrLen
* sizeof(WCHAR
));
165 lpNextStr
+= ulStrLen
;
171 LPBYTE lpNext
= (LPBYTE
)(lpDest
->Value
.MVbin
.lpbin
+
172 lpDest
->Value
.MVbin
.cValues
);
174 for (i
= 0; i
< lpSrc
->Value
.MVszW
.cValues
; i
++)
176 lpDest
->Value
.MVbin
.lpbin
[i
].cb
= lpSrc
->Value
.MVbin
.lpbin
[i
].cb
;
177 lpDest
->Value
.MVbin
.lpbin
[i
].lpb
= lpNext
;
178 memcpy(lpNext
, lpSrc
->Value
.MVbin
.lpbin
[i
].lpb
, lpDest
->Value
.MVbin
.lpbin
[i
].cb
);
179 lpNext
+= lpDest
->Value
.MVbin
.lpbin
[i
].cb
;
184 /* No embedded pointers, just copy the data over */
185 memcpy(lpDest
->Value
.MVi
.lpi
, lpSrc
->Value
.MVi
.lpi
, ulLen
);
194 /*************************************************************************
195 * UlPropSize@4 (MAPI32.77)
197 * Determine the size of a property in bytes.
200 * lpProp [I] Property to determine the size of
203 * Success: The size of the value in lpProp.
204 * Failure: 0, if a multi-value (array) property is invalid or the type of lpProp
208 * - The size returned does not include the size of the SPropValue struct
209 * or the size of the array of pointers for multi-valued properties that
210 * contain pointers (such as PT_MV_STRING8 or PT-MV_UNICODE).
211 * - MSDN incorrectly states that this function returns MAPI_E_CALL_FAILED if
212 * lpProp is invalid. In reality no checking is performed and this function
213 * will crash if passed an invalid property, or return 0 if the property
214 * type is PT_OBJECT or is unknown.
216 ULONG WINAPI
UlPropSize(LPSPropValue lpProp
)
220 TRACE("(%p)\n", lpProp
);
222 switch (PROP_TYPE(lpProp
->ulPropTag
))
224 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
;
230 case PT_I4
: ulRet
*= sizeof(LONG
);
232 case PT_MV_I8
: ulRet
= lpProp
->Value
.MVli
.cValues
;
233 case PT_I8
: ulRet
*= sizeof(LONG64
);
235 case PT_MV_R4
: ulRet
= lpProp
->Value
.MVflt
.cValues
;
236 case PT_R4
: ulRet
*= sizeof(float);
239 case PT_MV_R8
: ulRet
= lpProp
->Value
.MVdbl
.cValues
;
241 case PT_R8
: ulRet
*= sizeof(double);
243 case PT_MV_CURRENCY
: ulRet
= lpProp
->Value
.MVcur
.cValues
;
244 case PT_CURRENCY
: ulRet
*= sizeof(CY
);
246 case PT_MV_SYSTIME
: ulRet
= lpProp
->Value
.MVft
.cValues
;
247 case PT_SYSTIME
: ulRet
*= sizeof(FILETIME
);
249 case PT_MV_CLSID
: ulRet
= lpProp
->Value
.MVguid
.cValues
;
250 case PT_CLSID
: ulRet
*= sizeof(GUID
);
252 case PT_MV_STRING8
: ulRet
= 0u;
253 for (i
= 0; i
< lpProp
->Value
.MVszA
.cValues
; i
++)
254 ulRet
+= (lstrlenA(lpProp
->Value
.MVszA
.lppszA
[i
]) + 1u);
256 case PT_STRING8
: ulRet
= lstrlenA(lpProp
->Value
.lpszA
) + 1u;
258 case PT_MV_UNICODE
: ulRet
= 0u;
259 for (i
= 0; i
< lpProp
->Value
.MVszW
.cValues
; i
++)
260 ulRet
+= (strlenW(lpProp
->Value
.MVszW
.lppszW
[i
]) + 1u);
261 ulRet
*= sizeof(WCHAR
);
263 case PT_UNICODE
: ulRet
= (lstrlenW(lpProp
->Value
.lpszW
) + 1u) * sizeof(WCHAR
);
265 case PT_MV_BINARY
: ulRet
= 0u;
266 for (i
= 0; i
< lpProp
->Value
.MVbin
.cValues
; i
++)
267 ulRet
+= lpProp
->Value
.MVbin
.lpbin
[i
].cb
;
269 case PT_BINARY
: ulRet
= lpProp
->Value
.bin
.cb
;
279 /*************************************************************************
280 * FPropContainsProp@12 (MAPI32.78)
282 * Find a property with a given property tag in a property array.
285 * lpHaystack [I] Property to match to
286 * lpNeedle [I] Property to find in lpHaystack
287 * ulFuzzy [I] Flags controlling match type and strictness (FL_* flags from "mapidefs.h")
290 * TRUE, if lpNeedle matches lpHaystack according to the criteria of ulFuzzy.
293 * Only property types of PT_STRING8 and PT_BINARY are handled by this function.
295 BOOL WINAPI
FPropContainsProp(LPSPropValue lpHaystack
, LPSPropValue lpNeedle
, ULONG ulFuzzy
)
297 TRACE("(%p,%p,0x%08lx)\n", lpHaystack
, lpNeedle
, ulFuzzy
);
299 if (FBadProp(lpHaystack
) || FBadProp(lpNeedle
) ||
300 PROP_TYPE(lpHaystack
->ulPropTag
) != PROP_TYPE(lpNeedle
->ulPropTag
))
303 /* FIXME: Do later versions support Unicode as well? */
305 if (PROP_TYPE(lpHaystack
->ulPropTag
) == PT_STRING8
)
307 DWORD dwFlags
= 0, dwNeedleLen
, dwHaystackLen
;
309 if (ulFuzzy
& FL_IGNORECASE
)
310 dwFlags
|= NORM_IGNORECASE
;
311 if (ulFuzzy
& FL_IGNORENONSPACE
)
312 dwFlags
|= NORM_IGNORENONSPACE
;
313 if (ulFuzzy
& FL_LOOSE
)
314 dwFlags
|= (NORM_IGNORECASE
|NORM_IGNORENONSPACE
|NORM_IGNORESYMBOLS
);
316 dwNeedleLen
= lstrlenA(lpNeedle
->Value
.lpszA
);
317 dwHaystackLen
= lstrlenA(lpHaystack
->Value
.lpszA
);
319 if ((ulFuzzy
& (FL_SUBSTRING
|FL_PREFIX
)) == FL_PREFIX
)
321 if (dwNeedleLen
<= dwHaystackLen
&&
322 CompareStringA(LOCALE_USER_DEFAULT
, dwFlags
,
323 lpHaystack
->Value
.lpszA
, dwNeedleLen
,
324 lpNeedle
->Value
.lpszA
, dwNeedleLen
) == CSTR_EQUAL
)
325 return TRUE
; /* needle is a prefix of haystack */
327 else if ((ulFuzzy
& (FL_SUBSTRING
|FL_PREFIX
)) == FL_SUBSTRING
)
329 LPSTR (WINAPI
*pStrChrFn
)(LPCSTR
,WORD
) = StrChrA
;
330 LPSTR lpStr
= lpHaystack
->Value
.lpszA
;
332 if (dwFlags
& NORM_IGNORECASE
)
333 pStrChrFn
= StrChrIA
;
335 while ((lpStr
= pStrChrFn(lpStr
, *lpNeedle
->Value
.lpszA
)) != NULL
)
337 dwHaystackLen
-= (lpStr
- lpHaystack
->Value
.lpszA
);
338 if (dwNeedleLen
<= dwHaystackLen
&&
339 CompareStringA(LOCALE_USER_DEFAULT
, dwFlags
,
341 lpNeedle
->Value
.lpszA
, dwNeedleLen
) == CSTR_EQUAL
)
342 return TRUE
; /* needle is a substring of haystack */
346 else if (CompareStringA(LOCALE_USER_DEFAULT
, dwFlags
,
347 lpHaystack
->Value
.lpszA
, dwHaystackLen
,
348 lpNeedle
->Value
.lpszA
, dwNeedleLen
) == CSTR_EQUAL
)
349 return TRUE
; /* full string match */
351 else if (PROP_TYPE(lpHaystack
->ulPropTag
) == PT_BINARY
)
353 if ((ulFuzzy
& (FL_SUBSTRING
|FL_PREFIX
)) == FL_PREFIX
)
355 if (lpNeedle
->Value
.bin
.cb
<= lpHaystack
->Value
.bin
.cb
&&
356 !memcmp(lpNeedle
->Value
.bin
.lpb
, lpHaystack
->Value
.bin
.lpb
,
357 lpNeedle
->Value
.bin
.cb
))
358 return TRUE
; /* needle is a prefix of haystack */
360 else if ((ulFuzzy
& (FL_SUBSTRING
|FL_PREFIX
)) == FL_SUBSTRING
)
362 ULONG ulLen
= lpHaystack
->Value
.bin
.cb
;
363 LPBYTE lpb
= lpHaystack
->Value
.bin
.lpb
;
365 while ((lpb
= memchr(lpb
, *lpNeedle
->Value
.bin
.lpb
, ulLen
)) != NULL
)
367 ulLen
= lpHaystack
->Value
.bin
.cb
- (lpb
- lpHaystack
->Value
.bin
.lpb
);
368 if (lpNeedle
->Value
.bin
.cb
<= ulLen
&&
369 !memcmp(lpNeedle
->Value
.bin
.lpb
, lpb
, lpNeedle
->Value
.bin
.cb
))
370 return TRUE
; /* needle is a substring of haystack */
374 else if (!LPropCompareProp(lpHaystack
, lpNeedle
))
375 return TRUE
; /* needle is an exact match with haystack */
381 /*************************************************************************
382 * FPropCompareProp@12 (MAPI32.79)
384 * Compare two properties.
387 * lpPropLeft [I] Left hand property to compare to lpPropRight
388 * ulOp [I] Comparison operator (RELOP_* enum from "mapidefs.h")
389 * lpPropRight [I] Right hand property to compare to lpPropLeft
392 * TRUE, if the comparison is true, FALSE otherwise.
394 BOOL WINAPI
FPropCompareProp(LPSPropValue lpPropLeft
, ULONG ulOp
, LPSPropValue lpPropRight
)
398 TRACE("(%p,%ld,%p)\n", lpPropLeft
, ulOp
, lpPropRight
);
400 if (ulOp
> RELOP_RE
|| FBadProp(lpPropLeft
) || FBadProp(lpPropRight
))
403 if (ulOp
== RELOP_RE
)
405 FIXME("Comparison operator RELOP_RE not yet implemented!\n");
409 iCmp
= LPropCompareProp(lpPropLeft
, lpPropRight
);
413 case RELOP_LT
: return iCmp
< 0 ? TRUE
: FALSE
;
414 case RELOP_LE
: return iCmp
<= 0 ? TRUE
: FALSE
;
415 case RELOP_GT
: return iCmp
> 0 ? TRUE
: FALSE
;
416 case RELOP_GE
: return iCmp
>= 0 ? TRUE
: FALSE
;
417 case RELOP_EQ
: return iCmp
== 0 ? TRUE
: FALSE
;
418 case RELOP_NE
: return iCmp
!= 0 ? TRUE
: FALSE
;
423 /*************************************************************************
424 * LPropCompareProp@8 (MAPI32.80)
426 * Compare two properties.
429 * lpPropLeft [I] Left hand property to compare to lpPropRight
430 * lpPropRight [I] Right hand property to compare to lpPropLeft
433 * An integer less than, equal to or greater than 0, indicating that
434 * lpszStr is less than, the same, or greater than lpszComp.
436 LONG WINAPI
LPropCompareProp(LPSPropValue lpPropLeft
, LPSPropValue lpPropRight
)
440 TRACE("(%p->0x%08lx,%p->0x%08lx)\n", lpPropLeft
, lpPropLeft
->ulPropTag
,
441 lpPropRight
, lpPropRight
->ulPropTag
);
443 /* If the properties are not the same, sort by property type */
444 if (PROP_TYPE(lpPropLeft
->ulPropTag
) != PROP_TYPE(lpPropRight
->ulPropTag
))
445 return (LONG
)PROP_TYPE(lpPropLeft
->ulPropTag
) - (LONG
)PROP_TYPE(lpPropRight
->ulPropTag
);
447 switch (PROP_TYPE(lpPropLeft
->ulPropTag
))
451 return 0; /* NULLs are equal */
453 return lpPropLeft
->Value
.i
- lpPropRight
->Value
.i
;
455 return lpPropLeft
->Value
.l
- lpPropRight
->Value
.l
;
457 if (lpPropLeft
->Value
.li
.QuadPart
> lpPropRight
->Value
.li
.QuadPart
)
459 if (lpPropLeft
->Value
.li
.QuadPart
== lpPropRight
->Value
.li
.QuadPart
)
463 if (lpPropLeft
->Value
.flt
> lpPropRight
->Value
.flt
)
465 if (lpPropLeft
->Value
.flt
== lpPropRight
->Value
.flt
)
470 if (lpPropLeft
->Value
.dbl
> lpPropRight
->Value
.dbl
)
472 if (lpPropLeft
->Value
.dbl
== lpPropRight
->Value
.dbl
)
476 if (lpPropLeft
->Value
.cur
.int64
> lpPropRight
->Value
.cur
.int64
)
478 if (lpPropLeft
->Value
.cur
.int64
== lpPropRight
->Value
.cur
.int64
)
482 return CompareFileTime(&lpPropLeft
->Value
.ft
, &lpPropRight
->Value
.ft
);
484 return (lpPropLeft
->Value
.b
? 1 : 0) - (lpPropRight
->Value
.b
? 1 : 0);
486 if (lpPropLeft
->Value
.bin
.cb
== lpPropRight
->Value
.bin
.cb
)
487 iRet
= memcmp(lpPropLeft
->Value
.bin
.lpb
, lpPropRight
->Value
.bin
.lpb
,
488 lpPropLeft
->Value
.bin
.cb
);
491 iRet
= memcmp(lpPropLeft
->Value
.bin
.lpb
, lpPropRight
->Value
.bin
.lpb
,
492 min(lpPropLeft
->Value
.bin
.cb
, lpPropRight
->Value
.bin
.cb
));
495 iRet
= lpPropLeft
->Value
.bin
.cb
- lpPropRight
->Value
.bin
.cb
;
499 return lstrcmpA(lpPropLeft
->Value
.lpszA
, lpPropRight
->Value
.lpszA
);
501 return strcmpW(lpPropLeft
->Value
.lpszW
, lpPropRight
->Value
.lpszW
);
503 if (lpPropLeft
->Value
.err
> lpPropRight
->Value
.err
)
505 if (lpPropLeft
->Value
.err
== lpPropRight
->Value
.err
)
509 return memcmp(lpPropLeft
->Value
.lpguid
, lpPropRight
->Value
.lpguid
,
512 FIXME("Unhandled property type %ld", PROP_TYPE(lpPropLeft
->ulPropTag
));
516 /*************************************************************************
517 * HrGetOneProp@8 (MAPI32.135)
519 * Get a property value from an IMAPIProp object.
522 * lpIProp [I] IMAPIProp object to get the property value in
523 * ulPropTag [I] Property tag of the property to get
524 * lppProp [O] Destination for the returned property
527 * Success: S_OK. *lppProp contains the property value requested.
528 * Failure: MAPI_E_NOT_FOUND, if no property value has the tag given by ulPropTag.
530 HRESULT WINAPI
HrGetOneProp(LPMAPIPROP lpIProp
, ULONG ulPropTag
, LPSPropValue
*lppProp
)
536 TRACE("(%p,%ld,%p)\n", lpIProp
, ulPropTag
, lppProp
);
539 pta
.aulPropTag
[0] = ulPropTag
;
540 hRet
= IMAPIProp_GetProps(lpIProp
, &pta
, 0u, &ulCount
, lppProp
);
541 if (hRet
== MAPI_W_ERRORS_RETURNED
)
543 MAPIFreeBuffer(*lppProp
);
545 hRet
= MAPI_E_NOT_FOUND
;
550 /*************************************************************************
551 * HrSetOneProp@8 (MAPI32.136)
553 * Set a property value in an IMAPIProp object.
556 * lpIProp [I] IMAPIProp object to set the property value in
557 * lpProp [I] Property value to set
560 * Success: S_OK. The value in lpProp is set in lpIProp.
561 * Failure: An error result from IMAPIProp_SetProps().
563 HRESULT WINAPI
HrSetOneProp(LPMAPIPROP lpIProp
, LPSPropValue lpProp
)
565 TRACE("(%p,%p)\n", lpIProp
, lpProp
);
567 return IMAPIProp_SetProps(lpIProp
, 1u, lpProp
, NULL
);
570 /*************************************************************************
571 * FPropExists@8 (MAPI32.137)
573 * Find a property with a given property tag in an IMAPIProp object.
576 * lpIProp [I] IMAPIProp object to find the property tag in
577 * ulPropTag [I] Property tag to find
580 * TRUE, if ulPropTag matches a property held in lpIProp,
584 * if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
585 * Ids need to match for a successful match to occur.
587 BOOL WINAPI
FPropExists(LPMAPIPROP lpIProp
, ULONG ulPropTag
)
591 TRACE("(%p,%ld)\n", lpIProp
, ulPropTag
);
595 LPSPropTagArray lpTags
;
598 if (FAILED(IMAPIProp_GetPropList(lpIProp
, 0u, &lpTags
)))
601 for (i
= 0; i
< lpTags
->cValues
; i
++)
603 if (!FBadPropTag(lpTags
->aulPropTag
[i
]) &&
604 (lpTags
->aulPropTag
[i
] == ulPropTag
||
605 (PROP_TYPE(ulPropTag
) == PT_UNSPECIFIED
&&
606 PROP_ID(lpTags
->aulPropTag
[i
]) == lpTags
->aulPropTag
[i
])))
612 MAPIFreeBuffer(lpTags
);
617 /*************************************************************************
618 * PpropFindProp@12 (MAPI32.138)
620 * Find a property with a given property tag in a property array.
623 * lpProps [I] Property array to search
624 * cValues [I] Number of properties in lpProps
625 * ulPropTag [I] Property tag to find
628 * A pointer to the matching property, or NULL if none was found.
631 * if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
632 * Ids need to match for a successful match to occur.
634 LPSPropValue WINAPI
PpropFindProp(LPSPropValue lpProps
, ULONG cValues
, ULONG ulPropTag
)
636 TRACE("(%p,%ld,%ld)\n", lpProps
, cValues
, ulPropTag
);
638 if (lpProps
&& cValues
)
641 for (i
= 0; i
< cValues
; i
++)
643 if (!FBadPropTag(lpProps
[i
].ulPropTag
) &&
644 (lpProps
[i
].ulPropTag
== ulPropTag
||
645 (PROP_TYPE(ulPropTag
) == PT_UNSPECIFIED
&&
646 PROP_ID(lpProps
[i
].ulPropTag
) == PROP_ID(ulPropTag
))))
653 /*************************************************************************
654 * FreePadrlist@4 (MAPI32.139)
656 * Free the memory used by an address book list.
659 * lpAddrs [I] Address book list to free
664 VOID WINAPI
FreePadrlist(LPADRLIST lpAddrs
)
666 TRACE("(%p)\n", lpAddrs
);
668 /* Structures are binary compatible; use the same implementation */
669 return FreeProws((LPSRowSet
)lpAddrs
);
672 /*************************************************************************
673 * FreeProws@4 (MAPI32.140)
675 * Free the memory used by a row set.
678 * lpRowSet [I] Row set to free
683 VOID WINAPI
FreeProws(LPSRowSet lpRowSet
)
685 TRACE("(%p)\n", lpRowSet
);
691 for (i
= 0; i
< lpRowSet
->cRows
; i
++)
692 MAPIFreeBuffer(lpRowSet
->aRow
[i
].lpProps
);
694 MAPIFreeBuffer(lpRowSet
);
698 /*************************************************************************
699 * ScCountProps@12 (MAPI32.170)
701 * Validate and determine the length of an array of properties.
704 * iCount [I] Length of the lpProps array
705 * lpProps [I] Array of properties to validate/size
706 * pcBytes [O] If non-NULL, destination for the size of the property array
709 * Success: S_OK. If pcBytes is non-NULL, it contains the size of the propery array.
710 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid or validation
711 * of the property array fails.
713 SCODE WINAPI
ScCountProps(INT iCount
, LPSPropValue lpProps
, ULONG
*pcBytes
)
715 ULONG i
, ulCount
= iCount
, ulBytes
= 0;
717 TRACE("(%d,%p,%p)\n", iCount
, lpProps
, pcBytes
);
719 if (iCount
<= 0 || !lpProps
||
720 IsBadReadPtr(lpProps
, iCount
* sizeof(SPropValue
)))
721 return MAPI_E_INVALID_PARAMETER
;
723 for (i
= 0; i
< ulCount
; i
++)
725 ULONG ulPropSize
= 0;
727 if (FBadProp(&lpProps
[i
]) || lpProps
[i
].ulPropTag
== PROP_ID_NULL
||
728 lpProps
[i
].ulPropTag
== PROP_ID_INVALID
)
729 return MAPI_E_INVALID_PARAMETER
;
731 if (PROP_TYPE(lpProps
[i
].ulPropTag
) != PT_OBJECT
)
733 ulPropSize
= UlPropSize(&lpProps
[i
]);
735 return MAPI_E_INVALID_PARAMETER
;
738 switch (PROP_TYPE(lpProps
[i
].ulPropTag
))
752 ulPropSize
+= sizeof(SPropValue
);
755 ulPropSize
+= lpProps
[i
].Value
.MVszA
.cValues
* sizeof(char*) + sizeof(SPropValue
);
759 ulPropSize
+= lpProps
[i
].Value
.MVszA
.cValues
* sizeof(char*) + sizeof(SPropValue
);
762 ulPropSize
+= lpProps
[i
].Value
.MVbin
.cValues
* sizeof(SBinary
) + sizeof(SPropValue
);
765 ulPropSize
= sizeof(SPropValue
);
768 ulBytes
+= ulPropSize
;
776 /*************************************************************************
777 * ScCopyProps@16 (MAPI32.171)
779 * Copy an array of property values into a buffer suited for serialisation.
782 * cValues [I] Number of properties in lpProps
783 * lpProps [I] Property array to copy
784 * lpDst [O] Destination for the serialised data
785 * lpCount [O] If non-NULL, destination for the number of bytes of data written to lpDst
788 * Success: S_OK. lpDst contains the serialised data from lpProps.
789 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
792 * The resulting property value array is stored in a contigous block starting at lpDst.
794 SCODE WINAPI
ScCopyProps(int cValues
, LPSPropValue lpProps
, LPVOID lpDst
, ULONG
*lpCount
)
796 LPSPropValue lpDest
= (LPSPropValue
)lpDst
;
797 char *lpDataDest
= (char *)(lpDest
+ cValues
);
798 ULONG ulLen
, i
, iter
;
800 TRACE("(%d,%p,%p,%p)\n", cValues
, lpProps
, lpDst
, lpCount
);
802 if (!lpProps
|| cValues
< 0 || !lpDest
)
803 return MAPI_E_INVALID_PARAMETER
;
805 memcpy(lpDst
, lpProps
, cValues
* sizeof(SPropValue
));
807 for (iter
= 0; iter
< cValues
; iter
++)
809 switch (PROP_TYPE(lpProps
->ulPropTag
))
812 lpDest
->Value
.lpguid
= (LPGUID
)lpDataDest
;
813 memcpy(lpDest
->Value
.lpguid
, lpProps
->Value
.lpguid
, sizeof(GUID
));
814 lpDataDest
+= sizeof(GUID
);
817 ulLen
= lstrlenA(lpProps
->Value
.lpszA
) + 1u;
818 lpDest
->Value
.lpszA
= lpDataDest
;
819 memcpy(lpDest
->Value
.lpszA
, lpProps
->Value
.lpszA
, ulLen
);
823 ulLen
= (strlenW(lpProps
->Value
.lpszW
) + 1u) * sizeof(WCHAR
);
824 lpDest
->Value
.lpszW
= (LPWSTR
)lpDataDest
;
825 memcpy(lpDest
->Value
.lpszW
, lpProps
->Value
.lpszW
, ulLen
);
829 lpDest
->Value
.bin
.lpb
= (LPBYTE
)lpDataDest
;
830 memcpy(lpDest
->Value
.bin
.lpb
, lpProps
->Value
.bin
.lpb
, lpProps
->Value
.bin
.cb
);
831 lpDataDest
+= lpProps
->Value
.bin
.cb
;
834 if (lpProps
->ulPropTag
& MV_FLAG
)
836 lpDest
->Value
.MVi
.cValues
= lpProps
->Value
.MVi
.cValues
;
837 /* Note: Assignment uses lppszA but covers all cases by union aliasing */
838 lpDest
->Value
.MVszA
.lppszA
= (char**)lpDataDest
;
840 switch (PROP_TYPE(lpProps
->ulPropTag
))
844 lpDataDest
+= lpProps
->Value
.MVszA
.cValues
* sizeof(char *);
846 for (i
= 0; i
< lpProps
->Value
.MVszA
.cValues
; i
++)
848 ULONG ulStrLen
= lstrlenA(lpProps
->Value
.MVszA
.lppszA
[i
]) + 1u;
850 lpDest
->Value
.MVszA
.lppszA
[i
] = lpDataDest
;
851 memcpy(lpDataDest
, lpProps
->Value
.MVszA
.lppszA
[i
], ulStrLen
);
852 lpDataDest
+= ulStrLen
;
858 lpDataDest
+= lpProps
->Value
.MVszW
.cValues
* sizeof(WCHAR
*);
860 for (i
= 0; i
< lpProps
->Value
.MVszW
.cValues
; i
++)
862 ULONG ulStrLen
= (strlenW(lpProps
->Value
.MVszW
.lppszW
[i
]) + 1u) * sizeof(WCHAR
);
864 lpDest
->Value
.MVszW
.lppszW
[i
] = (LPWSTR
)lpDataDest
;
865 memcpy(lpDataDest
, lpProps
->Value
.MVszW
.lppszW
[i
], ulStrLen
);
866 lpDataDest
+= ulStrLen
;
872 lpDataDest
+= lpProps
->Value
.MVszW
.cValues
* sizeof(SBinary
);
874 for (i
= 0; i
< lpProps
->Value
.MVszW
.cValues
; i
++)
876 lpDest
->Value
.MVbin
.lpbin
[i
].cb
= lpProps
->Value
.MVbin
.lpbin
[i
].cb
;
877 lpDest
->Value
.MVbin
.lpbin
[i
].lpb
= (LPBYTE
)lpDataDest
;
878 memcpy(lpDataDest
, lpProps
->Value
.MVbin
.lpbin
[i
].lpb
, lpDest
->Value
.MVbin
.lpbin
[i
].cb
);
879 lpDataDest
+= lpDest
->Value
.MVbin
.lpbin
[i
].cb
;
884 /* No embedded pointers, just copy the data over */
885 ulLen
= UlPropSize(lpProps
);
886 memcpy(lpDest
->Value
.MVi
.lpi
, lpProps
->Value
.MVi
.lpi
, ulLen
);
897 *lpCount
= lpDataDest
- (char *)lpDst
;
902 /*************************************************************************
903 * ScRelocProps@20 (MAPI32.172)
905 * Relocate the pointers in an array of property values after it has been copied.
908 * cValues [I] Number of properties in lpProps
909 * lpProps [O] Property array to relocate the pointers in.
910 * lpOld [I] Position where the data was copied from
911 * lpNew [I] Position where the data was copied to
912 * lpCount [O] If non-NULL, destination for the number of bytes of data at lpDst
915 * Success: S_OK. Any pointers in lpProps are relocated.
916 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
919 * MSDN states that this function can be used for serialisation by passing
920 * NULL as either lpOld or lpNew, thus converting any pointers in lpProps
921 * between offsets and pointers. This does not work in native (it crashes),
922 * and cannot be made to work in Wine because the original interface design
923 * is deficient. The only use left for this function is to remap pointers
924 * in a contigous property array that has been copied with memcpy() to another
927 SCODE WINAPI
ScRelocProps(int cValues
, LPSPropValue lpProps
, LPVOID lpOld
,
928 LPVOID lpNew
, ULONG
*lpCount
)
930 static const BOOL bBadPtr
= TRUE
; /* Windows bug - Assumes source is bad */
931 LPSPropValue lpDest
= (LPSPropValue
)lpProps
;
932 ULONG ulCount
= cValues
* sizeof(SPropValue
);
933 ULONG ulLen
, i
, iter
;
935 TRACE("(%d,%p,%p,%p,%p)\n", cValues
, lpProps
, lpOld
, lpNew
, lpCount
);
937 if (!lpProps
|| cValues
< 0 || !lpOld
|| !lpNew
)
938 return MAPI_E_INVALID_PARAMETER
;
940 /* The reason native doesn't work as MSDN states is that it assumes that
941 * the lpProps pointer contains valid pointers. This is obviously not
942 * true if the array is being read back from serialisation (the pointers
943 * are just offsets). Native can't actually work converting the pointers to
944 * offsets either, because it converts any array pointers to offsets then
945 * _dereferences the offset_ in order to convert the array elements!
947 * The code below would handle both cases except that the design of this
948 * function makes it impossible to know when the pointers in lpProps are
949 * valid. If both lpOld and lpNew are non-NULL, native reads the pointers
950 * after converting them, so we must do the same. Its seems this
951 * functionality was never tested by MS.
954 #define RELOC_PTR(p) (((char*)(p)) - (char*)lpOld + (char*)lpNew)
956 for (iter
= 0; iter
< cValues
; iter
++)
958 switch (PROP_TYPE(lpDest
->ulPropTag
))
961 lpDest
->Value
.lpguid
= (LPGUID
)RELOC_PTR(lpDest
->Value
.lpguid
);
962 ulCount
+= sizeof(GUID
);
965 ulLen
= bBadPtr
? 0 : lstrlenA(lpDest
->Value
.lpszA
) + 1u;
966 lpDest
->Value
.lpszA
= (LPSTR
)RELOC_PTR(lpDest
->Value
.lpszA
);
968 ulLen
= lstrlenA(lpDest
->Value
.lpszA
) + 1u;
972 ulLen
= bBadPtr
? 0 : (lstrlenW(lpDest
->Value
.lpszW
) + 1u) * sizeof(WCHAR
);
973 lpDest
->Value
.lpszW
= (LPWSTR
)RELOC_PTR(lpDest
->Value
.lpszW
);
975 ulLen
= (strlenW(lpDest
->Value
.lpszW
) + 1u) * sizeof(WCHAR
);
979 lpDest
->Value
.bin
.lpb
= (LPBYTE
)RELOC_PTR(lpDest
->Value
.bin
.lpb
);
980 ulCount
+= lpDest
->Value
.bin
.cb
;
983 if (lpDest
->ulPropTag
& MV_FLAG
)
985 /* Since we have to access the array elements, don't map the
986 * array unless it is invalid (otherwise, map it at the end)
989 lpDest
->Value
.MVszA
.lppszA
= (LPSTR
*)RELOC_PTR(lpDest
->Value
.MVszA
.lppszA
);
991 switch (PROP_TYPE(lpProps
->ulPropTag
))
995 ulCount
+= lpDest
->Value
.MVszA
.cValues
* sizeof(char *);
997 for (i
= 0; i
< lpDest
->Value
.MVszA
.cValues
; i
++)
999 ULONG ulStrLen
= bBadPtr
? 0 : lstrlenA(lpDest
->Value
.MVszA
.lppszA
[i
]) + 1u;
1001 lpDest
->Value
.MVszA
.lppszA
[i
] = (LPSTR
)RELOC_PTR(lpDest
->Value
.MVszA
.lppszA
[i
]);
1003 ulStrLen
= lstrlenA(lpDest
->Value
.MVszA
.lppszA
[i
]) + 1u;
1004 ulCount
+= ulStrLen
;
1010 ulCount
+= lpDest
->Value
.MVszW
.cValues
* sizeof(WCHAR
*);
1012 for (i
= 0; i
< lpDest
->Value
.MVszW
.cValues
; i
++)
1014 ULONG ulStrLen
= bBadPtr
? 0 : (strlenW(lpDest
->Value
.MVszW
.lppszW
[i
]) + 1u) * sizeof(WCHAR
);
1016 lpDest
->Value
.MVszW
.lppszW
[i
] = (LPWSTR
)RELOC_PTR(lpDest
->Value
.MVszW
.lppszW
[i
]);
1018 ulStrLen
= (strlenW(lpDest
->Value
.MVszW
.lppszW
[i
]) + 1u) * sizeof(WCHAR
);
1019 ulCount
+= ulStrLen
;
1025 ulCount
+= lpDest
->Value
.MVszW
.cValues
* sizeof(SBinary
);
1027 for (i
= 0; i
< lpDest
->Value
.MVszW
.cValues
; i
++)
1029 lpDest
->Value
.MVbin
.lpbin
[i
].lpb
= (LPBYTE
)RELOC_PTR(lpDest
->Value
.MVbin
.lpbin
[i
].lpb
);
1030 ulCount
+= lpDest
->Value
.MVbin
.lpbin
[i
].cb
;
1035 ulCount
+= UlPropSize(lpDest
);
1039 lpDest
->Value
.MVszA
.lppszA
= (LPSTR
*)RELOC_PTR(lpDest
->Value
.MVszA
.lppszA
);
1051 /*************************************************************************
1052 * LpValFindProp@12 (MAPI32.173)
1054 * Find a property with a given property id in a property array.
1057 * ulPropTag [I] Property tag containing property id to find
1058 * cValues [I] Number of properties in lpProps
1059 * lpProps [I] Property array to search
1062 * A pointer to the matching property, or NULL if none was found.
1065 * This function matches only on the property id and does not care if the
1066 * property types differ.
1068 LPSPropValue WINAPI
LpValFindProp(ULONG ulPropTag
, ULONG cValues
, LPSPropValue lpProps
)
1070 TRACE("(%ld,%ld,%p)\n", ulPropTag
, cValues
, lpProps
);
1072 if (lpProps
&& cValues
)
1075 for (i
= 0; i
< cValues
; i
++)
1077 if (PROP_ID(ulPropTag
) == PROP_ID(lpProps
[i
].ulPropTag
))
1084 /*************************************************************************
1085 * ScDupPropset@16 (MAPI32.174)
1087 * Duplicate a property value array into a contigous block of memory.
1090 * cValues [I] Number of properties in lpProps
1091 * lpProps [I] Property array to duplicate
1092 * lpAlloc [I] Memory allocation function, use MAPIAllocateBuffer()
1093 * lpNewProp [O] Destination for the newly duplicated property value array
1096 * Success: S_OK. *lpNewProp contains the duplicated array.
1097 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1098 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1100 SCODE WINAPI
ScDupPropset(int cValues
, LPSPropValue lpProps
,
1101 LPALLOCATEBUFFER lpAlloc
, LPSPropValue
*lpNewProp
)
1106 TRACE("(%d,%p,%p,%p)\n", cValues
, lpProps
, lpAlloc
, lpNewProp
);
1108 sc
= ScCountProps(cValues
, lpProps
, &ulCount
);
1111 sc
= lpAlloc(ulCount
, (LPVOID
*)lpNewProp
);
1113 sc
= ScCopyProps(cValues
, lpProps
, *lpNewProp
, &ulCount
);
1118 /*************************************************************************
1119 * FBadRglpszA@8 (MAPI32.175)
1121 * Determine if an array of strings is invalid
1124 * lppszStrs [I] Array of strings to check
1125 * ulCount [I] Number of strings in lppszStrs
1128 * TRUE, if lppszStrs is invalid, FALSE otherwise.
1130 BOOL WINAPI
FBadRglpszA(LPSTR
*lppszStrs
, ULONG ulCount
)
1134 TRACE("(%p,%ld)\n", lppszStrs
, ulCount
);
1139 if (!lppszStrs
|| IsBadReadPtr(lppszStrs
, ulCount
* sizeof(LPWSTR
)))
1142 for (i
= 0; i
< ulCount
; i
++)
1144 if (!lppszStrs
[i
] || IsBadStringPtrA(lppszStrs
[i
], -1))
1150 /*************************************************************************
1151 * FBadRglpszW@8 (MAPI32.176)
1155 BOOL WINAPI
FBadRglpszW(LPWSTR
*lppszStrs
, ULONG ulCount
)
1159 TRACE("(%p,%ld)\n", lppszStrs
, ulCount
);
1164 if (!lppszStrs
|| IsBadReadPtr(lppszStrs
, ulCount
* sizeof(LPWSTR
)))
1167 for (i
= 0; i
< ulCount
; i
++)
1169 if (!lppszStrs
[i
] || IsBadStringPtrW(lppszStrs
[i
], -1))
1175 /*************************************************************************
1176 * FBadRowSet@4 (MAPI32.177)
1178 * Determine if a row is invalid
1181 * lpRow [I] Row to check
1184 * TRUE, if lpRow is invalid, FALSE otherwise.
1186 BOOL WINAPI
FBadRowSet(LPSRowSet lpRowSet
)
1189 TRACE("(%p)\n", lpRowSet
);
1191 if (!lpRowSet
|| IsBadReadPtr(lpRowSet
, CbSRowSet(lpRowSet
)))
1194 for (i
= 0; i
< lpRowSet
->cRows
; i
++)
1196 if (FBadRow(&lpRowSet
->aRow
[i
]))
1202 /*************************************************************************
1203 * FBadPropTag@4 (MAPI32.179)
1205 * Determine if a property tag is invalid
1208 * ulPropTag [I] Property tag to check
1211 * TRUE, if ulPropTag is invalid, FALSE otherwise.
1213 ULONG WINAPI
FBadPropTag(ULONG ulPropTag
)
1215 TRACE("(0x%08lx)\n", ulPropTag
);
1217 switch (ulPropTag
& (~MV_FLAG
& PROP_TYPE_MASK
))
1219 case PT_UNSPECIFIED
:
1241 /*************************************************************************
1242 * FBadRow@4 (MAPI32.180)
1244 * Determine if a row is invalid
1247 * lpRow [I] Row to check
1250 * TRUE, if lpRow is invalid, FALSE otherwise.
1252 ULONG WINAPI
FBadRow(LPSRow lpRow
)
1255 TRACE("(%p)\n", lpRow
);
1257 if (!lpRow
|| IsBadReadPtr(lpRow
, sizeof(SRow
)) || !lpRow
->lpProps
||
1258 IsBadReadPtr(lpRow
->lpProps
, lpRow
->cValues
* sizeof(SPropValue
)))
1261 for (i
= 0; i
< lpRow
->cValues
; i
++)
1263 if (FBadProp(&lpRow
->lpProps
[i
]))
1269 /*************************************************************************
1270 * FBadProp@4 (MAPI32.181)
1272 * Determine if a property is invalid
1275 * lpProp [I] Property to check
1278 * TRUE, if lpProp is invalid, FALSE otherwise.
1280 ULONG WINAPI
FBadProp(LPSPropValue lpProp
)
1282 if (!lpProp
|| IsBadReadPtr(lpProp
, sizeof(SPropValue
)) ||
1283 FBadPropTag(lpProp
->ulPropTag
))
1286 switch (PROP_TYPE(lpProp
->ulPropTag
))
1288 /* Single value properties containing pointers */
1290 if (!lpProp
->Value
.lpszA
|| IsBadStringPtrA(lpProp
->Value
.lpszA
, -1))
1294 if (!lpProp
->Value
.lpszW
|| IsBadStringPtrW(lpProp
->Value
.lpszW
, -1))
1298 if (IsBadReadPtr(lpProp
->Value
.bin
.lpb
, lpProp
->Value
.bin
.cb
))
1302 if (IsBadReadPtr(lpProp
->Value
.lpguid
, sizeof(GUID
)))
1306 /* Multiple value properties (arrays) containing no pointers */
1308 return PROP_BadArray(lpProp
, sizeof(SHORT
));
1310 return PROP_BadArray(lpProp
, sizeof(LONG
));
1311 case PT_MV_LONGLONG
:
1312 return PROP_BadArray(lpProp
, sizeof(LONG64
));
1314 return PROP_BadArray(lpProp
, sizeof(float));
1316 return PROP_BadArray(lpProp
, sizeof(FILETIME
));
1319 return PROP_BadArray(lpProp
, sizeof(double));
1320 case PT_MV_CURRENCY
:
1321 return PROP_BadArray(lpProp
, sizeof(CY
));
1323 return PROP_BadArray(lpProp
, sizeof(GUID
));
1325 /* Multiple value properties containing pointers */
1327 return FBadRglpszA(lpProp
->Value
.MVszA
.lppszA
,
1328 lpProp
->Value
.MVszA
.cValues
);
1330 return FBadRglpszW(lpProp
->Value
.MVszW
.lppszW
,
1331 lpProp
->Value
.MVszW
.cValues
);
1333 return FBadEntryList((LPENTRYLIST
)&lpProp
->Value
.MVbin
);
1338 /*************************************************************************
1339 * FBadColumnSet@4 (MAPI32.182)
1341 * Determine if an array of property tags is invalid
1344 * lpCols [I] Property tag array to check
1347 * TRUE, if lpCols is invalid, FALSE otherwise.
1349 ULONG WINAPI
FBadColumnSet(LPSPropTagArray lpCols
)
1351 ULONG ulRet
= FALSE
, i
;
1353 TRACE("(%p)\n", lpCols
);
1355 if (!lpCols
|| IsBadReadPtr(lpCols
, CbSPropTagArray(lpCols
)))
1359 for (i
= 0; i
< lpCols
->cValues
; i
++)
1361 if ((lpCols
->aulPropTag
[i
] & PROP_TYPE_MASK
) == PT_ERROR
||
1362 FBadPropTag(lpCols
->aulPropTag
[i
]))
1369 TRACE("Returning %s\n", ulRet
? "TRUE" : "FALSE");
1374 /**************************************************************************
1375 * IMAPIProp {MAPI32}
1377 * The default Mapi interface for manipulating object properties.
1380 * This object provides an interface to an objects properties. It is exposed
1381 * by several types of Mapi objects in order to simplify the querying and
1382 * modification of properties.
1387 /* A single property in a property data collection */
1391 ULONG ulAccess
; /* The property value access level */
1392 LPSPropValue value
; /* The property value */
1393 } IPropDataItem
, *LPIPropDataItem
;
1395 /* The main property data collection structure */
1398 IPropDataVtbl
*lpVtbl
;
1399 LONG lRef
; /* Reference count */
1400 ALLOCATEBUFFER
*lpAlloc
; /* Memory allocation routine */
1401 ALLOCATEMORE
*lpMore
; /* Linked memory allocation routine */
1402 FREEBUFFER
*lpFree
; /* Memory free routine */
1403 ULONG ulObjAccess
; /* Object access level */
1404 ULONG ulNumValues
; /* Number of items in values list */
1405 struct list values
; /* List of property values */
1406 CRITICAL_SECTION cs
; /* Lock for thread safety */
1409 /* Internal - Get a property value, assumes lock is held */
1410 static IPropDataItem
*IMAPIPROP_GetValue(IPropDataImpl
*This
, ULONG ulPropTag
)
1412 struct list
*cursor
;
1414 LIST_FOR_EACH(cursor
, &This
->values
)
1416 LPIPropDataItem current
= LIST_ENTRY(cursor
, IPropDataItem
, entry
);
1417 /* Note that propery types don't have to match, just Id's */
1418 if (PROP_ID(current
->value
->ulPropTag
) == PROP_ID(ulPropTag
))
1424 /* Internal - Add a new property value, assumes lock is held */
1425 static IPropDataItem
*IMAPIPROP_AddValue(IPropDataImpl
*This
,
1426 LPSPropValue lpProp
)
1429 LPIPropDataItem lpNew
;
1432 hRet
= This
->lpAlloc(sizeof(IPropDataItem
), &lpMem
);
1434 if (SUCCEEDED(hRet
))
1437 lpNew
->ulAccess
= IPROP_READWRITE
;
1439 /* Allocate the value seperately so we can update it easily */
1441 hRet
= This
->lpAlloc(sizeof(SPropValue
), &lpMem
);
1442 if (SUCCEEDED(hRet
))
1444 lpNew
->value
= lpMem
;
1446 hRet
= PropCopyMore(lpNew
->value
, lpProp
, This
->lpMore
, lpMem
);
1447 if (SUCCEEDED(hRet
))
1449 list_add_tail(&This
->values
, &lpNew
->entry
);
1450 This
->ulNumValues
++;
1453 This
->lpFree(lpNew
->value
);
1455 This
->lpFree(lpNew
);
1460 /* Internal - Lock an IPropData object */
1461 static inline void IMAPIPROP_Lock(IPropDataImpl
*This
)
1463 RtlEnterCriticalSection(&This
->cs
);
1466 /* Internal - Unlock an IPropData object */
1467 static inline void IMAPIPROP_Unlock(IPropDataImpl
*This
)
1469 RtlLeaveCriticalSection(&This
->cs
);
1472 /* This one seems to be missing from mapidefs.h */
1473 #define CbNewSPropProblemArray(c) \
1474 (offsetof(SPropProblemArray,aProblem)+(c)*sizeof(SPropProblem))
1476 /**************************************************************************
1477 * IMAPIProp_QueryInterface {MAPI32}
1479 * Inherited method from the IUnknown Interface.
1480 * See IUnknown_QueryInterface.
1483 * This object exposes the following interfaces:
1484 * - IUnknown() : The default interface for all COM-Objects.
1485 * - IMAPIProp() : The default Mapi interface for manipulating object properties.
1487 static inline HRESULT WINAPI
1488 IMAPIProp_fnQueryInterface(LPMAPIPROP iface
, REFIID riid
, LPVOID
*ppvObj
)
1490 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1492 TRACE("(%p,%s,%p)\n", This
, debugstr_guid(riid
), ppvObj
);
1494 if (!ppvObj
|| !riid
)
1495 return MAPI_E_INVALID_PARAMETER
;
1499 if(IsEqualIID(riid
, &IID_IUnknown
) ||
1500 IsEqualIID(riid
, &IID_IMAPIProp
) ||
1501 IsEqualIID(riid
, &IID_IMAPIPropData
))
1504 IPropData_AddRef(iface
);
1505 TRACE("returning %p\n", *ppvObj
);
1509 TRACE("returning E_NOINTERFACE\n");
1510 return MAPI_E_INTERFACE_NOT_SUPPORTED
;
1513 /**************************************************************************
1514 * IMAPIProp_AddRef {MAPI32}
1516 * Inherited method from the IUnknown Interface.
1517 * See IUnknown_AddRef.
1519 static inline ULONG WINAPI
IMAPIProp_fnAddRef(LPMAPIPROP iface
)
1521 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1523 TRACE("(%p)->(count=%ld)\n", This
, This
->lRef
);
1525 return InterlockedIncrement(&This
->lRef
);
1528 /**************************************************************************
1529 * IMAPIProp_Release {MAPI32}
1531 * Inherited method from the IUnknown Interface.
1532 * See IUnknown_Release.
1534 static inline ULONG WINAPI
IMAPIProp_fnRelease(LPMAPIPROP iface
)
1536 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1538 TRACE("(%p)->()\n", This
);
1540 if (!InterlockedDecrement(&This
->lRef
))
1542 TRACE("Destroying IPropData (%p)\n",This
);
1544 /* Note: No need to lock, since no other thread is referencing iface */
1545 while (!list_empty(&This
->values
))
1547 struct list
*head
= list_head(&This
->values
);
1548 LPIPropDataItem current
= LIST_ENTRY(head
, IPropDataItem
, entry
);
1550 This
->lpFree(current
->value
);
1551 This
->lpFree(current
);
1553 RtlDeleteCriticalSection(&This
->cs
);
1557 return (ULONG
)This
->lRef
;
1560 /**************************************************************************
1561 * IMAPIProp_GetLastError {MAPI32}
1563 * Get information about the last error that ocurred in an IMAPIProp object.
1566 * iface [I] IMAPIProp object that experienced the error
1567 * hRes [I] Result of the call that returned an error
1568 * ulFlags [I] 0=return Ascii strings, MAPI_UNICODE=return Unicode strings
1569 * lppError [O] Destination for detailed error information
1572 * Success: S_OK. *lppError contains details about the last error.
1573 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1574 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1577 * - If this function succeeds, the returned information in *lppError must be
1578 * freed using MAPIFreeBuffer() once the caller is finished with it.
1579 * - It is possible for this function to suceed and set *lppError to NULL,
1580 * if there is no further information to report about hRes.
1582 static inline HRESULT WINAPI
1583 IMAPIProp_fnGetLastError(LPMAPIPROP iface
, HRESULT hRes
,
1584 ULONG ulFlags
, LPMAPIERROR
*lppError
)
1586 TRACE("(%p,0x%08lX,0x%08lX,%p)\n", iface
, hRes
, ulFlags
, lppError
);
1588 if (!lppError
|| SUCCEEDED(hRes
) || (ulFlags
& ~MAPI_UNICODE
))
1589 return MAPI_E_INVALID_PARAMETER
;
1595 /**************************************************************************
1596 * IMAPIProp_SaveChanges {MAPI32}
1598 * Update any changes made to a tansactional IMAPIProp object.
1601 * iface [I] IMAPIProp object to update
1602 * ulFlags [I] Flags controlling the update.
1605 * Success: S_OK. Any outstanding changes are committed to the object.
1606 * Failure: An HRESULT error code describing the error.
1608 static inline HRESULT WINAPI
1609 IMAPIProp_fnSaveChanges(LPMAPIPROP iface
, ULONG ulFlags
)
1611 TRACE("(%p,0x%08lX)\n", iface
, ulFlags
);
1613 /* Since this object is not transacted we do not need to implement this */
1614 /* FIXME: Should we set the access levels to clean? */
1618 /**************************************************************************
1619 * IMAPIProp_GetProps {MAPI32}
1621 * Get property values from an IMAPIProp object.
1624 * iface [I] IMAPIProp object to get the property values from
1625 * lpTags [I] Property tage of property values to be retrieved
1626 * ulFlags [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1628 * lpCount [O] Destination for number of properties returned
1629 * lppProps [O] Destination for returned property values
1632 * Success: S_OK. *lppProps and *lpCount are updated.
1633 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1634 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
1635 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
1638 * - If MAPI_W_ERRORS_RETURNED is returned, any properties that could not be
1639 * retrieved from iface are present in lppProps with their type
1640 * changed to PT_ERROR and Id unchanged.
1642 static inline HRESULT WINAPI
1643 IMAPIProp_fnGetProps(LPMAPIPROP iface
, LPSPropTagArray lpTags
,
1644 ULONG ulFlags
, ULONG
*lpCount
, LPSPropValue
*lppProps
)
1647 HRESULT hRet
= S_OK
;
1648 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1650 TRACE("(%p,%p,0x%08lx,%p,%p) stub\n", iface
, lpTags
, ulFlags
,
1653 if (!iface
|| ulFlags
& ~MAPI_UNICODE
|| !lpTags
|| *lpCount
|| !lppProps
)
1654 return MAPI_E_INVALID_PARAMETER
;
1656 FIXME("semi-stub, flags not supported\n");
1658 *lpCount
= lpTags
->cValues
;
1663 hRet
= MAPIAllocateBuffer(*lpCount
* sizeof(SPropValue
), (LPVOID
*)lppProps
);
1667 IMAPIPROP_Lock(This
);
1669 for (i
= 0; i
< lpTags
->cValues
; i
++)
1671 HRESULT hRetTmp
= E_INVALIDARG
;
1672 LPIPropDataItem item
;
1674 item
= IMAPIPROP_GetValue(This
, lpTags
->aulPropTag
[i
]);
1677 hRetTmp
= PropCopyMore(&(*lppProps
)[i
], item
->value
,
1678 This
->lpMore
, *lppProps
);
1679 if (FAILED(hRetTmp
))
1681 hRet
= MAPI_W_ERRORS_RETURNED
;
1682 (*lppProps
)[i
].ulPropTag
=
1683 CHANGE_PROP_TYPE(lpTags
->aulPropTag
[i
], PT_ERROR
);
1687 IMAPIPROP_Unlock(This
);
1692 /**************************************************************************
1693 * MAPIProp_GetPropList {MAPI32}
1695 * Get the list of property tags for all values in an IMAPIProp object.
1698 * iface [I] IMAPIProp object to get the property tag list from
1699 * ulFlags [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1701 * lppTags [O] Destination for the retrieved peoperty tag list
1704 * Success: S_OK. *lppTags contains the tags for all available properties.
1705 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1706 * MAPI_E_BAD_CHARWIDTH, if Ascii or Unicode strings are requested
1707 * and that type of string is not supported.
1709 static inline HRESULT WINAPI
1710 IMAPIProp_fnGetPropList(LPMAPIPROP iface
, ULONG ulFlags
,
1711 LPSPropTagArray
*lppTags
)
1713 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1717 TRACE("(%p,0x%08lx,%p) stub\n", iface
, ulFlags
, lppTags
);
1719 if (!iface
|| ulFlags
& ~MAPI_UNICODE
|| !lppTags
)
1720 return MAPI_E_INVALID_PARAMETER
;
1722 FIXME("semi-stub, flags not supported\n");
1726 IMAPIPROP_Lock(This
);
1728 hRet
= MAPIAllocateBuffer(CbNewSPropTagArray(This
->ulNumValues
),
1730 if (SUCCEEDED(hRet
))
1732 struct list
*cursor
;
1735 LIST_FOR_EACH(cursor
, &This
->values
)
1737 LPIPropDataItem current
= LIST_ENTRY(cursor
, IPropDataItem
, entry
);
1738 (*lppTags
)->aulPropTag
[i
] = current
->value
->ulPropTag
;
1741 (*lppTags
)->cValues
= This
->ulNumValues
;
1744 IMAPIPROP_Unlock(This
);
1748 /**************************************************************************
1749 * IMAPIProp_OpenProperty {MAPI32}
1751 * Not documented at this time.
1754 * An HRESULT success/failure code.
1756 static inline HRESULT WINAPI
1757 IMAPIProp_fnOpenProperty(LPMAPIPROP iface
, ULONG ulPropTag
, LPCIID iid
,
1758 ULONG ulOpts
, ULONG ulFlags
, LPUNKNOWN
*lpUnk
)
1760 FIXME("(%p,%lu,%s,%lu,0x%08lx,%p) stub\n", iface
, ulPropTag
,
1761 debugstr_guid(iid
), ulOpts
, ulFlags
, lpUnk
);
1762 return MAPI_E_NO_SUPPORT
;
1766 /**************************************************************************
1767 * IMAPIProp_SetProps {MAPI32}
1769 * Add or edit the property values in an IMAPIProp object.
1772 * iface [I] IMAPIProp object to get the property tag list from
1773 * ulValues [I] Number of properties in lpProps
1774 * lpProps [I] Property values to set
1775 * lppProbs [O] Optional destination for any problems that occurred
1778 * Success: S_OK. The properties in lpProps are added to iface if they don't
1779 * exist, or changed to the values in lpProps if they do
1780 * Failure: An HRESULT error code describing the error
1782 static inline HRESULT WINAPI
1783 IMAPIProp_fnSetProps(LPMAPIPROP iface
, ULONG ulValues
,
1784 LPSPropValue lpProps
, LPSPropProblemArray
*lppProbs
)
1786 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1787 HRESULT hRet
= S_OK
;
1790 TRACE("(%p,%lu,%p,%p)\n", iface
, ulValues
, lpProps
, lppProbs
);
1792 if (!iface
|| !lpProps
)
1793 return MAPI_E_INVALID_PARAMETER
;
1795 for (i
= 0; i
< ulValues
; i
++)
1797 if (FBadProp(&lpProps
[i
]) ||
1798 PROP_TYPE(lpProps
[i
].ulPropTag
) == PT_OBJECT
||
1799 PROP_TYPE(lpProps
[i
].ulPropTag
) == PT_NULL
)
1800 return MAPI_E_INVALID_PARAMETER
;
1803 IMAPIPROP_Lock(This
);
1805 /* FIXME: Under what circumstances is lpProbs created? */
1806 for (i
= 0; i
< ulValues
; i
++)
1808 LPIPropDataItem item
= IMAPIPROP_GetValue(This
, lpProps
[i
].ulPropTag
);
1813 LPVOID lpMem
= NULL
;
1815 /* Found, so update the existing value */
1816 if (item
->value
->ulPropTag
!= lpProps
[i
].ulPropTag
)
1817 FIXME("semi-stub, overwriting type (not coercing)\n");
1819 hRetTmp
= This
->lpAlloc(sizeof(SPropValue
), &lpMem
);
1820 if (SUCCEEDED(hRetTmp
))
1822 hRetTmp
= PropCopyMore(lpMem
, &lpProps
[i
], This
->lpMore
, lpMem
);
1823 if (SUCCEEDED(hRetTmp
))
1825 This
->lpFree(item
->value
);
1826 item
->value
= lpMem
;
1829 This
->lpFree(lpMem
);
1836 if (!(item
= IMAPIPROP_AddValue(This
, &lpProps
[i
])))
1837 hRet
= MAPI_E_NOT_ENOUGH_MEMORY
;
1841 IMAPIPROP_Unlock(This
);
1845 /**************************************************************************
1846 * IMAPIProp_DeleteProps {MAPI32}
1848 * Delete one or more property values from a IMAPIProp objects.
1851 * iface [I] IMAPIProp object to remove property values from.
1852 * lpTags [I] Collection of property Id's to remove from iface.
1853 * lppProbs [O] Destination for problems encountered, if any.
1856 * Success: S_OK. Any properties in iface matching property Id's in lpTags have
1857 * been deleted. If lppProbs is non-NULL it contains details of any
1858 * errors that occurred.
1859 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1860 * E_ACCESSDENIED, if this object was created using CreateIProp() and
1861 * a subsequent call to IPropData_SetObjAcess() was made specifying
1862 * IPROP_READONLY as the access type.
1865 * - lppProbs will not be populated for cases where a property Id is present
1866 * in lpTags but not in iface.
1867 * - lppProbs should be deleted with MAPIFreeBuffer() if returned.
1869 static inline HRESULT WINAPI
1870 IMAPIProp_fnDeleteProps(LPMAPIPROP iface
, LPSPropTagArray lpTags
,
1871 LPSPropProblemArray
*lppProbs
)
1873 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
1874 ULONG i
, numProbs
= 0;
1875 HRESULT hRet
= S_OK
;
1877 TRACE("(%p,%p,%p)\n", iface
, lpTags
, lppProbs
);
1879 if (!iface
|| !lpTags
)
1880 return MAPI_E_INVALID_PARAMETER
;
1885 for (i
= 0; i
< lpTags
->cValues
; i
++)
1887 if (FBadPropTag(lpTags
->aulPropTag
[i
]) ||
1888 PROP_TYPE(lpTags
->aulPropTag
[i
]) == PT_OBJECT
||
1889 PROP_TYPE(lpTags
->aulPropTag
[i
]) == PT_NULL
)
1890 return MAPI_E_INVALID_PARAMETER
;
1893 IMAPIPROP_Lock(This
);
1895 if (This
->ulObjAccess
!= IPROP_READWRITE
)
1897 IMAPIPROP_Unlock(This
);
1898 return E_ACCESSDENIED
;
1901 for (i
= 0; i
< lpTags
->cValues
; i
++)
1903 LPIPropDataItem item
= IMAPIPROP_GetValue(This
, lpTags
->aulPropTag
[i
]);
1907 if (item
->ulAccess
& IPROP_READWRITE
)
1909 /* Everything hunky-dory, remove the item */
1910 list_remove(&item
->entry
);
1911 This
->lpFree(item
->value
); /* Also frees value pointers */
1913 This
->ulNumValues
--;
1917 /* Can't write the value. Create/populate problems array */
1920 /* Create problems array */
1921 ULONG ulSize
= CbNewSPropProblemArray(lpTags
->cValues
- i
);
1922 HRESULT hRetTmp
= MAPIAllocateBuffer(ulSize
, (LPVOID
*)lppProbs
);
1923 if (FAILED(hRetTmp
))
1928 LPSPropProblem lpProb
= &(*lppProbs
)->aProblem
[numProbs
];
1929 lpProb
->ulIndex
= i
;
1930 lpProb
->ulPropTag
= lpTags
->aulPropTag
[i
];
1931 lpProb
->scode
= E_ACCESSDENIED
;
1937 if (lppProbs
&& *lppProbs
)
1938 (*lppProbs
)->cProblem
= numProbs
;
1940 IMAPIPROP_Unlock(This
);
1945 /**************************************************************************
1946 * IMAPIProp_CopyTo {MAPI32}
1948 * Not documented at this time.
1951 * An HRESULT success/failure code.
1953 static inline HRESULT WINAPI
1954 IMAPIProp_fnCopyTo(LPMAPIPROP iface
, ULONG niids
, LPCIID lpiidExcl
,
1955 LPSPropTagArray lpPropsExcl
, ULONG ulParam
,
1956 LPMAPIPROGRESS lpIProgress
, LPCIID lpIfaceIid
, LPVOID lpDstObj
,
1957 ULONG ulFlags
, LPSPropProblemArray
*lppProbs
)
1959 FIXME("(%p,%lu,%p,%p,%lx,%p,%s,%p,0x%08lX,%p) stub\n", iface
, niids
,
1960 lpiidExcl
, lpPropsExcl
, ulParam
, lpIProgress
,
1961 debugstr_guid(lpIfaceIid
), lpDstObj
, ulFlags
, lppProbs
);
1962 return MAPI_E_NO_SUPPORT
;
1965 /**************************************************************************
1966 * IMAPIProp_CopyProps {MAPI32}
1968 * Not documented at this time.
1971 * An HRESULT success/failure code.
1973 static inline HRESULT WINAPI
1974 IMAPIProp_fnCopyProps(LPMAPIPROP iface
, LPSPropTagArray lpInclProps
,
1975 ULONG ulParam
, LPMAPIPROGRESS lpIProgress
, LPCIID lpIface
,
1976 LPVOID lpDstObj
, ULONG ulFlags
,
1977 LPSPropProblemArray
*lppProbs
)
1979 FIXME("(%p,%p,%lx,%p,%s,%p,0x%08lX,%p) stub\n", iface
, lpInclProps
,
1980 ulParam
, lpIProgress
, debugstr_guid(lpIface
), lpDstObj
, ulFlags
,
1982 return MAPI_E_NO_SUPPORT
;
1985 /**************************************************************************
1986 * IMAPIProp_GetNamesFromIDs {MAPI32}
1988 * Get the names of properties from their identifiers.
1991 * iface [I] IMAPIProp object to operate on
1992 * lppPropTags [I/O] Property identifiers to get the names for, or NULL to
1994 * iid [I] Property set identifier, or NULL
1995 * ulFlags [I] MAPI_NO_IDS=Don't return numeric named properties,
1996 * or MAPI_NO_STRINGS=Don't return strings
1997 * lpCount [O] Destination for number of properties returned
1998 * lpppNames [O] Destination for returned names
2001 * Success: S_OK. *lppPropTags and lpppNames contain the returned
2003 * Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2004 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2005 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2008 static inline HRESULT WINAPI
2009 IMAPIProp_fnGetNamesFromIDs(LPMAPIPROP iface
, LPSPropTagArray
*lppPropTags
,
2010 LPGUID iid
, ULONG ulFlags
, ULONG
*lpCount
,
2011 LPMAPINAMEID
**lpppNames
)
2013 FIXME("(%p,%p,%s,0x%08lX,%p,%p) stub\n", iface
, lppPropTags
,
2014 debugstr_guid(iid
), ulFlags
, lpCount
, lpppNames
);
2015 return MAPI_E_NO_SUPPORT
;
2018 /**************************************************************************
2019 * IMAPIProp_GetIDsFromNames {MAPI32}
2021 * Get property identifiers associated with one or more named properties.
2024 * iface [I] IMAPIProp object to operate on
2025 * ulNames [I] Number of names in lppNames
2026 * lppNames [I] Names to query or create, or NULL to query all names
2027 * ulFlags [I] Pass MAPI_CREATE to create new named properties
2028 * lppPropTags [O] Destination for queried or created property identifiers
2031 * Success: S_OK. *lppPropTags contains the property tags created or requested.
2032 * Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2033 * MAPI_E_TOO_BIG, if the object cannot process the number of
2034 * properties involved.
2035 * MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2036 * MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2039 static inline HRESULT WINAPI
2040 IMAPIProp_fnGetIDsFromNames(LPMAPIPROP iface
, ULONG ulNames
,
2041 LPMAPINAMEID
*lppNames
, ULONG ulFlags
,
2042 LPSPropTagArray
*lppPropTags
)
2044 FIXME("(%p,%ld,%p,0x%08lX,%p) stub\n",
2045 iface
, ulNames
, lppNames
, ulFlags
, lppPropTags
);
2046 return MAPI_E_NO_SUPPORT
;
2049 /**************************************************************************
2050 * IPropData {MAPI32}
2052 * A default Mapi interface to provide manipulation of object properties.
2055 * This object provides a default interface suitable in some cases as an
2056 * implementation of the IMAPIProp interface (which has no default
2057 * implementation). In addition to the IMAPIProp() methods inherited, this
2058 * interface allows read/write control over access to the object and its
2059 * individual properties.
2061 * To obtain the default implementation of this interface from Mapi, call
2067 /**************************************************************************
2068 * IPropData_QueryInterface {MAPI32}
2070 * Inherited method from the IMAPIProp Interface.
2071 * See IMAPIProp_QueryInterface.
2073 static HRESULT WINAPI
2074 IPropData_fnQueryInterface(LPPROPDATA iface
, REFIID riid
, LPVOID
*ppvObj
)
2076 return IMAPIProp_fnQueryInterface((LPMAPIPROP
)iface
, riid
, ppvObj
);
2079 /**************************************************************************
2080 * IPropData_AddRef {MAPI32}
2082 * Inherited method from the IMAPIProp Interface.
2083 * See IMAPIProp_AddRef.
2086 IPropData_fnAddRef(LPPROPDATA iface
)
2088 return IMAPIProp_fnAddRef((LPMAPIPROP
)iface
);
2091 /**************************************************************************
2092 * IPropData_Release {MAPI32}
2094 * Inherited method from the IMAPIProp Interface.
2095 * See IMAPIProp_Release.
2098 IPropData_fnRelease(LPPROPDATA iface
)
2100 return IMAPIProp_fnRelease((LPMAPIPROP
)iface
);
2103 /**************************************************************************
2104 * IPropData_GetLastError {MAPI32}
2106 * Inherited method from the IMAPIProp Interface.
2107 * See IMAPIProp_GetLastError.
2109 static HRESULT WINAPI
2110 IPropData_fnGetLastError(LPPROPDATA iface
, HRESULT hRes
, ULONG ulFlags
,
2111 LPMAPIERROR
*lppError
)
2113 return IMAPIProp_fnGetLastError((LPMAPIPROP
)iface
, hRes
, ulFlags
, lppError
);
2116 /**************************************************************************
2117 * IPropData_SaveChanges {MAPI32}
2119 * Inherited method from the IMAPIProp Interface.
2120 * See IMAPIProp_SaveChanges.
2122 static HRESULT WINAPI
2123 IPropData_fnSaveChanges(LPPROPDATA iface
, ULONG ulFlags
)
2125 return IMAPIProp_fnSaveChanges((LPMAPIPROP
)iface
, ulFlags
);
2128 /**************************************************************************
2129 * IPropData_GetProps {MAPI32}
2131 * Inherited method from the IMAPIProp Interface.
2132 * See IMAPIProp_GetProps.
2134 static HRESULT WINAPI
2135 IPropData_fnGetProps(LPPROPDATA iface
, LPSPropTagArray lpPropTags
,
2136 ULONG ulFlags
, ULONG
*lpCount
, LPSPropValue
*lppProps
)
2138 return IMAPIProp_fnGetProps((LPMAPIPROP
)iface
, lpPropTags
, ulFlags
,
2142 /**************************************************************************
2143 * IPropData_GetPropList {MAPI32}
2145 * Inherited method from the IMAPIProp Interface.
2146 * See IMAPIProp_GetPropList.
2148 static HRESULT WINAPI
2149 IPropData_fnGetPropList(LPPROPDATA iface
, ULONG ulFlags
,
2150 LPSPropTagArray
*lppPropTags
)
2152 return IMAPIProp_fnGetPropList((LPMAPIPROP
)iface
, ulFlags
, lppPropTags
);
2155 /**************************************************************************
2156 * IPropData_OpenProperty {MAPI32}
2158 * Inherited method from the IMAPIProp Interface.
2159 * See IMAPIProp_OpenProperty.
2161 static HRESULT WINAPI
2162 IPropData_fnOpenProperty(LPPROPDATA iface
, ULONG ulPropTag
, LPCIID iid
,
2163 ULONG ulOpts
, ULONG ulFlags
, LPUNKNOWN
*lpUnk
)
2165 return IMAPIProp_fnOpenProperty((LPMAPIPROP
)iface
, ulPropTag
, iid
,
2166 ulOpts
, ulFlags
, lpUnk
);
2169 /**************************************************************************
2170 * IPropData_SetProps {MAPI32}
2172 * Inherited method from the IMAPIProp Interface.
2173 * See IMAPIProp_SetProps.
2175 static HRESULT WINAPI
2176 IPropData_fnSetProps(LPPROPDATA iface
, ULONG cValues
, LPSPropValue lpProps
,
2177 LPSPropProblemArray
*lppProbs
)
2179 return IMAPIProp_fnSetProps((LPMAPIPROP
)iface
, cValues
, lpProps
, lppProbs
);
2182 /**************************************************************************
2183 * IPropData_DeleteProps {MAPI32}
2185 * Inherited method from the IMAPIProp Interface.
2186 * See IMAPIProp_DeleteProps.
2188 static HRESULT WINAPI
2189 IPropData_fnDeleteProps(LPPROPDATA iface
, LPSPropTagArray lpPropTags
,
2190 LPSPropProblemArray
*lppProbs
)
2192 return IMAPIProp_fnDeleteProps((LPMAPIPROP
)iface
, lpPropTags
, lppProbs
);
2195 /**************************************************************************
2196 * IPropData_CopyTo {MAPI32}
2198 * Inherited method from the IMAPIProp Interface.
2199 * See IMAPIProp_CopyTo.
2201 static HRESULT WINAPI
2202 IPropData_fnCopyTo(LPPROPDATA iface
, ULONG ciidExclude
, LPCIID lpIid
,
2203 LPSPropTagArray lpProps
, ULONG ulParam
,
2204 LPMAPIPROGRESS lpProgress
, LPCIID lpIface
, LPVOID lpDst
,
2205 ULONG ulFlags
, LPSPropProblemArray
*lppProbs
)
2207 return IMAPIProp_fnCopyTo((LPMAPIPROP
)iface
, ciidExclude
, lpIid
, lpProps
,
2208 ulParam
, lpProgress
, lpIface
, lpDst
,
2212 /**************************************************************************
2213 * IPropData_CopyProps {MAPI32}
2215 * Inherited method from the IMAPIProp Interface.
2216 * See IMAPIProp_CopyProps.
2218 static HRESULT WINAPI
2219 IPropData_fnCopyProps(LPPROPDATA iface
, LPSPropTagArray lpProps
,
2220 ULONG ulParam
, LPMAPIPROGRESS lpProgress
, LPCIID lpIface
,
2221 LPVOID lpDst
, ULONG ulFlags
, LPSPropProblemArray
*lppProbs
)
2223 return IMAPIProp_fnCopyProps((LPMAPIPROP
)iface
, lpProps
, ulParam
,
2224 lpProgress
, lpIface
, lpDst
, ulFlags
, lppProbs
);
2227 /**************************************************************************
2228 * IPropData_GetNamesFromIDs {MAPI32}
2230 * Inherited method from the IMAPIProp Interface.
2231 * See IMAPIProp_GetNamesFromIDs.
2233 static HRESULT WINAPI
2234 IPropData_fnGetNamesFromIDs(LPPROPDATA iface
, LPSPropTagArray
*lppPropTags
,
2235 LPGUID iid
, ULONG ulFlags
, ULONG
*lpCount
,
2236 LPMAPINAMEID
**lpppNames
)
2238 return IMAPIProp_fnGetNamesFromIDs((LPMAPIPROP
)iface
, lppPropTags
, iid
,
2239 ulFlags
, lpCount
, lpppNames
);
2242 /**************************************************************************
2243 * IPropData_GetIDsFromNames {MAPI32}
2245 * Inherited method from the IMAPIProp Interface.
2246 * See IMAPIProp_GetIDsFromNames.
2248 static HRESULT WINAPI
2249 IPropData_fnGetIDsFromNames(LPPROPDATA iface
, ULONG ulNames
,
2250 LPMAPINAMEID
*lppNames
, ULONG ulFlags
,
2251 LPSPropTagArray
*lppPropTags
)
2253 return IMAPIProp_fnGetIDsFromNames((LPMAPIPROP
)iface
, ulNames
, lppNames
,
2254 ulFlags
, lppPropTags
);
2257 /**************************************************************************
2258 * IPropData_HrSetObjAccess {MAPI32}
2260 * Set the access level of an IPropData object.
2263 * iface [I] IPropData object to set the access on
2264 * ulAccess [I] Either IPROP_READONLY or IPROP_READWRITE for read or
2265 * read/write access respectively.
2268 * Success: S_OK. The objects access level is changed.
2269 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2271 static HRESULT WINAPI
2272 IPropData_fnHrSetObjAccess(LPPROPDATA iface
, ULONG ulAccess
)
2274 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
2276 TRACE("(%p,%lx)\n", iface
, ulAccess
);
2278 if (!iface
|| ulAccess
< IPROP_READONLY
|| ulAccess
> IPROP_READWRITE
)
2279 return MAPI_E_INVALID_PARAMETER
;
2281 IMAPIPROP_Lock(This
);
2283 This
->ulObjAccess
= ulAccess
;
2285 IMAPIPROP_Unlock(This
);
2289 /* Internal - determine if an access value is bad */
2290 static inline BOOL
PROP_IsBadAccess(ULONG ulAccess
)
2294 case IPROP_READONLY
|IPROP_CLEAN
:
2295 case IPROP_READONLY
|IPROP_DIRTY
:
2296 case IPROP_READWRITE
|IPROP_CLEAN
:
2297 case IPROP_READWRITE
|IPROP_DIRTY
:
2303 /**************************************************************************
2304 * IPropData_HrSetPropAccess {MAPI32}
2306 * Set the access levels for a group of property values in an IPropData object.
2309 * iface [I] IPropData object to set access levels in.
2310 * lpTags [I] List of property Id's to set access for.
2311 * lpAccess [O] Access level for each property in lpTags.
2314 * Success: S_OK. The access level of each property value in lpTags that is
2315 * present in iface is changed.
2316 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2319 * - Each access level in lpAccess must contain at least one of IPROP_READONLY
2320 * or IPROP_READWRITE, but not both, and also IPROP_CLEAN or IPROP_DIRTY,
2321 * but not both. No other bits should be set.
2322 * - If a property Id in lpTags is not present in iface, it is ignored.
2324 static HRESULT WINAPI
2325 IPropData_fnHrSetPropAccess(LPPROPDATA iface
, LPSPropTagArray lpTags
,
2328 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
2332 TRACE("(%p,%p,%p)\n", iface
, lpTags
, lpAccess
);
2334 if (!iface
|| !lpTags
|| !lpAccess
)
2335 return MAPI_E_INVALID_PARAMETER
;
2337 for (i
= 0; i
< lpTags
->cValues
; i
++)
2339 if (FBadPropTag(lpTags
->aulPropTag
[i
]) || PROP_IsBadAccess(lpAccess
[i
]))
2340 return MAPI_E_INVALID_PARAMETER
;
2343 IMAPIPROP_Lock(This
);
2345 for (i
= 0; i
< lpTags
->cValues
; i
++)
2347 LPIPropDataItem item
= IMAPIPROP_GetValue(This
, lpTags
->aulPropTag
[i
]);
2350 item
->ulAccess
= lpAccess
[i
];
2353 IMAPIPROP_Unlock(This
);
2357 /**************************************************************************
2358 * IPropData_HrGetPropAccess {MAPI32}
2360 * Get the access levels for a group of property values in an IPropData object.
2363 * iface [I] IPropData object to get access levels from.
2364 * lppTags [O] Destination for the list of property Id's in iface.
2365 * lppAccess [O] Destination for access level for each property in lppTags.
2368 * Success: S_OK. lppTags and lppAccess contain the property Id's and the
2369 * Access level of each property value in iface.
2370 * Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid, or
2371 * MAPI_E_NOT_ENOUGH_MEMORY if memory allocation fails.
2374 * - *lppTags and *lppAccess should be freed with MAPIFreeBuffer() by the caller.
2376 static HRESULT WINAPI
2377 IPropData_fnHrGetPropAccess(LPPROPDATA iface
, LPSPropTagArray
*lppTags
,
2380 IPropDataImpl
*This
= (IPropDataImpl
*)iface
;
2385 TRACE("(%p,%p,%p) stub\n", iface
, lppTags
, lppAccess
);
2387 if (!iface
|| !lppTags
|| !lppAccess
)
2388 return MAPI_E_INVALID_PARAMETER
;
2393 IMAPIPROP_Lock(This
);
2395 hRet
= This
->lpAlloc(CbNewSPropTagArray(This
->ulNumValues
), &lpMem
);
2396 if (SUCCEEDED(hRet
))
2400 hRet
= This
->lpAlloc(This
->ulNumValues
* sizeof(ULONG
), &lpMem
);
2401 if (SUCCEEDED(hRet
))
2403 struct list
*cursor
;
2406 (*lppTags
)->cValues
= This
->ulNumValues
;
2409 LIST_FOR_EACH(cursor
, &This
->values
)
2411 LPIPropDataItem item
= LIST_ENTRY(cursor
, IPropDataItem
, entry
);
2412 (*lppTags
)->aulPropTag
[i
] = item
->value
->ulPropTag
;
2413 (*lppAccess
)[i
] = item
->ulAccess
;
2416 IMAPIPROP_Unlock(This
);
2419 This
->lpFree(*lppTags
);
2422 IMAPIPROP_Unlock(This
);
2423 return MAPI_E_NOT_ENOUGH_MEMORY
;
2426 /**************************************************************************
2427 * IPropData_HrAddObjProps {MAPI32}
2429 * Not documented at this time.
2432 * An HRESULT success/failure code.
2434 static HRESULT WINAPI
2435 IPropData_fnHrAddObjProps(LPPROPDATA iface
, LPSPropTagArray lpTags
,
2436 LPSPropProblemArray
*lppProbs
)
2441 LPSPropValue lpValues
;
2444 FIXME("(%p,%p,%p) stub\n", iface
, lpTags
, lppProbs
);
2446 if (!iface
|| !lpTags
)
2447 return MAPI_E_INVALID_PARAMETER
;
2449 /* FIXME: Below is the obvious implementation, adding all the properties
2450 * in lpTags to the object. However, it doesn't appear that this
2451 * is what this function does.
2455 if (!lpTags
->cValues
)
2458 lpValues
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
,
2459 lpTags
->cValues
* sizeof(SPropValue
));
2461 return MAPI_E_NOT_ENOUGH_MEMORY
;
2463 for (i
= 0; i
< lpTags
->cValues
; i
++)
2464 lpValues
[i
].ulPropTag
= lpTags
->aulPropTag
[i
];
2466 hRet
= IPropData_SetProps(iface
, lpTags
->cValues
, lpValues
, lppProbs
);
2467 HeapFree(GetProcessHeap(), 0, lpValues
);
2472 static struct IPropDataVtbl IPropDataImpl_vtbl
=
2474 IPropData_fnQueryInterface
,
2476 IPropData_fnRelease
,
2477 IPropData_fnGetLastError
,
2478 IPropData_fnSaveChanges
,
2479 IPropData_fnGetProps
,
2480 IPropData_fnGetPropList
,
2481 IPropData_fnOpenProperty
,
2482 IPropData_fnSetProps
,
2483 IPropData_fnDeleteProps
,
2485 IPropData_fnCopyProps
,
2486 IPropData_fnGetNamesFromIDs
,
2487 IPropData_fnGetIDsFromNames
,
2488 IPropData_fnHrSetObjAccess
,
2489 IPropData_fnHrSetPropAccess
,
2490 IPropData_fnHrGetPropAccess
,
2491 IPropData_fnHrAddObjProps
2494 /*************************************************************************
2495 * CreateIProp@24 (MAPI32.60)
2497 * Create an IPropData object.
2500 * iid [I] GUID of the object to create. Use &IID_IMAPIPropData or NULL
2501 * lpAlloc [I] Memory allocation function. Use MAPIAllocateBuffer()
2502 * lpMore [I] Linked memory allocation function. Use MAPIAllocateMore()
2503 * lpFree [I] Memory free function. Use MAPIFreeBuffer()
2504 * lpReserved [I] Reserved, set to NULL
2505 * lppPropData [O] Destination for created IPropData object
2508 * Success: S_OK. *lppPropData contains the newly created object.
2509 * Failure: MAPI_E_INTERFACE_NOT_SUPPORTED, if iid is non-NULL and not supported,
2510 * MAPI_E_INVALID_PARAMETER, if any parameter is invalid
2512 SCODE WINAPI
CreateIProp(LPCIID iid
, ALLOCATEBUFFER
*lpAlloc
,
2513 ALLOCATEMORE
*lpMore
, FREEBUFFER
*lpFree
,
2514 LPVOID lpReserved
, LPPROPDATA
*lppPropData
)
2516 IPropDataImpl
*lpPropData
;
2519 TRACE("(%s,%p,%p,%p,%p,%p)\n", debugstr_guid(iid
), lpAlloc
, lpMore
, lpFree
,
2520 lpReserved
, lppPropData
);
2523 *lppPropData
= NULL
;
2525 if (iid
&& !IsEqualGUID(iid
, &IID_IMAPIPropData
))
2526 return MAPI_E_INTERFACE_NOT_SUPPORTED
;
2528 if (!lpAlloc
|| !lpMore
|| !lpFree
|| lpReserved
|| !lppPropData
)
2529 return MAPI_E_INVALID_PARAMETER
;
2531 scode
= lpAlloc(sizeof(IPropDataImpl
), (LPVOID
*)&lpPropData
);
2533 if (SUCCEEDED(scode
))
2535 lpPropData
->lpVtbl
= &IPropDataImpl_vtbl
;
2536 lpPropData
->lRef
= 1;
2537 lpPropData
->lpAlloc
= lpAlloc
;
2538 lpPropData
->lpMore
= lpMore
;
2539 lpPropData
->lpFree
= lpFree
;
2540 lpPropData
->ulObjAccess
= IPROP_READWRITE
;
2541 lpPropData
->ulNumValues
= 0;
2542 list_init(&lpPropData
->values
);
2543 RtlInitializeCriticalSection(&lpPropData
->cs
);
2544 *lppPropData
= (LPPROPDATA
)lpPropData
;