xref: /reactos/dll/win32/mapi32/prop.c (revision 37b2c145)
1 /*
2  * Property functions
3  *
4  * Copyright 2004 Jon Griffiths
5  *
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.
10  *
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.
15  *
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
19  */
20 
21 #include <stdarg.h>
22 
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winreg.h"
26 #include "winerror.h"
27 #include "winternl.h"
28 #include "objbase.h"
29 #include "shlwapi.h"
30 #include "wine/list.h"
31 #include "wine/debug.h"
32 #include "mapival.h"
33 
34 WINE_DEFAULT_DEBUG_CHANNEL(mapi);
35 
36 BOOL WINAPI FBadRglpszA(LPSTR*,ULONG);
37 
38 /* Internal: Check if a property value array is invalid */
39 static inline ULONG PROP_BadArray(LPSPropValue lpProp, size_t elemSize)
40 {
41     return IsBadReadPtr(lpProp->Value.MVi.lpi, lpProp->Value.MVi.cValues * elemSize);
42 }
43 
44 /*************************************************************************
45  * PropCopyMore@16 (MAPI32.76)
46  *
47  * Copy a property value.
48  *
49  * PARAMS
50  *  lpDest [O] Destination for the copied value
51  *  lpSrc  [I] Property value to copy to lpDest
52  *  lpMore [I] Linked memory allocation function (pass MAPIAllocateMore())
53  *  lpOrig [I] Original allocation to which memory will be linked
54  *
55  * RETURNS
56  *  Success: S_OK. lpDest contains a deep copy of lpSrc.
57  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
58  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
59  *
60  * NOTES
61  *  Any elements within the property returned should not be individually
62  *  freed, as they will be freed when lpOrig is.
63  */
64 SCODE WINAPI PropCopyMore(LPSPropValue lpDest, LPSPropValue lpSrc,
65                           ALLOCATEMORE *lpMore, LPVOID lpOrig)
66 {
67     ULONG ulLen, i;
68     SCODE scode = S_OK;
69 
70     TRACE("(%p,%p,%p,%p)\n", lpDest, lpSrc, lpMore, lpOrig);
71 
72     if (!lpDest || IsBadWritePtr(lpDest, sizeof(SPropValue)) ||
73         FBadProp(lpSrc) || !lpMore)
74         return MAPI_E_INVALID_PARAMETER;
75 
76     /* Shallow copy first, this is sufficient for properties without pointers */
77     *lpDest = *lpSrc;
78 
79    switch (PROP_TYPE(lpSrc->ulPropTag))
80     {
81     case PT_CLSID:
82         scode = lpMore(sizeof(GUID), lpOrig, (LPVOID*)&lpDest->Value.lpguid);
83         if (SUCCEEDED(scode))
84             *lpDest->Value.lpguid = *lpSrc->Value.lpguid;
85         break;
86     case PT_STRING8:
87         ulLen = lstrlenA(lpSrc->Value.lpszA) + 1u;
88         scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.lpszA);
89         if (SUCCEEDED(scode))
90             memcpy(lpDest->Value.lpszA, lpSrc->Value.lpszA, ulLen);
91         break;
92     case PT_UNICODE:
93         ulLen = (lstrlenW(lpSrc->Value.lpszW) + 1u) * sizeof(WCHAR);
94         scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.lpszW);
95         if (SUCCEEDED(scode))
96             memcpy(lpDest->Value.lpszW, lpSrc->Value.lpszW, ulLen);
97         break;
98     case PT_BINARY:
99         scode = lpMore(lpSrc->Value.bin.cb, lpOrig, (LPVOID*)&lpDest->Value.bin.lpb);
100         if (SUCCEEDED(scode))
101             memcpy(lpDest->Value.bin.lpb, lpSrc->Value.bin.lpb, lpSrc->Value.bin.cb);
102         break;
103     default:
104         if (lpSrc->ulPropTag & MV_FLAG)
105         {
106             ulLen = UlPropSize(lpSrc);
107 
108             if (PROP_TYPE(lpSrc->ulPropTag) == PT_MV_STRING8 ||
109                 PROP_TYPE(lpSrc->ulPropTag) == PT_MV_UNICODE)
110             {
111                 /* UlPropSize doesn't account for the string pointers */
112                 ulLen += lpSrc->Value.MVszA.cValues * sizeof(char*);
113             }
114             else if (PROP_TYPE(lpSrc->ulPropTag) == PT_MV_BINARY)
115             {
116                /* UlPropSize doesn't account for the SBinary structs */
117                ulLen += lpSrc->Value.MVbin.cValues * sizeof(SBinary);
118             }
119 
120             lpDest->Value.MVi.cValues = lpSrc->Value.MVi.cValues;
121             scode = lpMore(ulLen, lpOrig, (LPVOID*)&lpDest->Value.MVi.lpi);
122             if (FAILED(scode))
123                 break;
124 
125             /* Note that we could allocate the memory for each value in a
126              * multi-value property separately, however if an allocation failed
127              * we would be left with a bunch of allocated memory, which (while
128              * not really leaked) is unusable until lpOrig is freed. So for
129              * strings and binary arrays we make a single allocation for all
130              * of the data. This is consistent since individual elements can't
131              * be freed anyway.
132              */
133 
134             switch (PROP_TYPE(lpSrc->ulPropTag))
135             {
136             case PT_MV_STRING8:
137             {
138                 char *lpNextStr = (char*)(lpDest->Value.MVszA.lppszA +
139                                           lpDest->Value.MVszA.cValues);
140 
141                 for (i = 0; i < lpSrc->Value.MVszA.cValues; i++)
142                 {
143                     ULONG ulStrLen = lstrlenA(lpSrc->Value.MVszA.lppszA[i]) + 1u;
144 
145                     lpDest->Value.MVszA.lppszA[i] = lpNextStr;
146                     memcpy(lpNextStr, lpSrc->Value.MVszA.lppszA[i], ulStrLen);
147                     lpNextStr += ulStrLen;
148                 }
149                 break;
150             }
151             case PT_MV_UNICODE:
152             {
153                 WCHAR *lpNextStr = (WCHAR*)(lpDest->Value.MVszW.lppszW +
154                                             lpDest->Value.MVszW.cValues);
155 
156                 for (i = 0; i < lpSrc->Value.MVszW.cValues; i++)
157                 {
158                     ULONG ulStrLen = lstrlenW(lpSrc->Value.MVszW.lppszW[i]) + 1u;
159 
160                     lpDest->Value.MVszW.lppszW[i] = lpNextStr;
161                     memcpy(lpNextStr, lpSrc->Value.MVszW.lppszW[i], ulStrLen * sizeof(WCHAR));
162                     lpNextStr += ulStrLen;
163                 }
164                 break;
165             }
166             case PT_MV_BINARY:
167             {
168                 LPBYTE lpNext = (LPBYTE)(lpDest->Value.MVbin.lpbin +
169                                          lpDest->Value.MVbin.cValues);
170 
171                 for (i = 0; i < lpSrc->Value.MVszW.cValues; i++)
172                 {
173                     lpDest->Value.MVbin.lpbin[i].cb = lpSrc->Value.MVbin.lpbin[i].cb;
174                     lpDest->Value.MVbin.lpbin[i].lpb = lpNext;
175                     memcpy(lpNext, lpSrc->Value.MVbin.lpbin[i].lpb, lpDest->Value.MVbin.lpbin[i].cb);
176                     lpNext += lpDest->Value.MVbin.lpbin[i].cb;
177                 }
178                 break;
179             }
180             default:
181                 /* No embedded pointers, just copy the data over */
182                 memcpy(lpDest->Value.MVi.lpi, lpSrc->Value.MVi.lpi, ulLen);
183                 break;
184             }
185             break;
186         }
187     }
188     return scode;
189 }
190 
191 /*************************************************************************
192  * UlPropSize@4 (MAPI32.77)
193  *
194  * Determine the size of a property in bytes.
195  *
196  * PARAMS
197  *  lpProp [I] Property to determine the size of
198  *
199  * RETURNS
200  *  Success: The size of the value in lpProp.
201  *  Failure: 0, if a multi-value (array) property is invalid or the type of lpProp
202  *           is unknown.
203  *
204  * NOTES
205  *  - The size returned does not include the size of the SPropValue struct
206  *    or the size of the array of pointers for multi-valued properties that
207  *    contain pointers (such as PT_MV_STRING8 or PT-MV_UNICODE).
208  *  - MSDN incorrectly states that this function returns MAPI_E_CALL_FAILED if
209  *    lpProp is invalid. In reality no checking is performed and this function
210  *    will crash if passed an invalid property, or return 0 if the property
211  *    type is PT_OBJECT or is unknown.
212  */
213 ULONG WINAPI UlPropSize(LPSPropValue lpProp)
214 {
215     ULONG ulRet = 1u, i;
216 
217     TRACE("(%p)\n", lpProp);
218 
219     switch (PROP_TYPE(lpProp->ulPropTag))
220     {
221     case PT_MV_I2:       ulRet = lpProp->Value.MVi.cValues;
222                          /* fall through */
223     case PT_BOOLEAN:
224     case PT_I2:          ulRet *= sizeof(USHORT);
225                          break;
226     case PT_MV_I4:       ulRet = lpProp->Value.MVl.cValues;
227                          /* fall through */
228     case PT_ERROR:
229     case PT_I4:          ulRet *= sizeof(LONG);
230                          break;
231     case PT_MV_I8:       ulRet = lpProp->Value.MVli.cValues;
232                          /* fall through */
233     case PT_I8:          ulRet *= sizeof(LONG64);
234                          break;
235     case PT_MV_R4:       ulRet = lpProp->Value.MVflt.cValues;
236                          /* fall through */
237     case PT_R4:          ulRet *= sizeof(float);
238                          break;
239     case PT_MV_APPTIME:
240     case PT_MV_R8:       ulRet = lpProp->Value.MVdbl.cValues;
241                          /* fall through */
242     case PT_APPTIME:
243     case PT_R8:          ulRet *= sizeof(double);
244                          break;
245     case PT_MV_CURRENCY: ulRet = lpProp->Value.MVcur.cValues;
246                          /* fall through */
247     case PT_CURRENCY:    ulRet *= sizeof(CY);
248                          break;
249     case PT_MV_SYSTIME:  ulRet = lpProp->Value.MVft.cValues;
250                          /* fall through */
251     case PT_SYSTIME:     ulRet *= sizeof(FILETIME);
252                          break;
253     case PT_MV_CLSID:    ulRet = lpProp->Value.MVguid.cValues;
254                          /* fall through */
255     case PT_CLSID:       ulRet *= sizeof(GUID);
256                          break;
257     case PT_MV_STRING8:  ulRet = 0u;
258                          for (i = 0; i < lpProp->Value.MVszA.cValues; i++)
259                              ulRet += (lstrlenA(lpProp->Value.MVszA.lppszA[i]) + 1u);
260                          break;
261     case PT_STRING8:     ulRet = lstrlenA(lpProp->Value.lpszA) + 1u;
262                          break;
263     case PT_MV_UNICODE:  ulRet = 0u;
264                          for (i = 0; i < lpProp->Value.MVszW.cValues; i++)
265                              ulRet += (lstrlenW(lpProp->Value.MVszW.lppszW[i]) + 1u);
266                          ulRet *= sizeof(WCHAR);
267                          break;
268     case PT_UNICODE:     ulRet = (lstrlenW(lpProp->Value.lpszW) + 1u) * sizeof(WCHAR);
269                          break;
270     case PT_MV_BINARY:   ulRet = 0u;
271                          for (i = 0; i < lpProp->Value.MVbin.cValues; i++)
272                              ulRet += lpProp->Value.MVbin.lpbin[i].cb;
273                          break;
274     case PT_BINARY:      ulRet = lpProp->Value.bin.cb;
275                          break;
276     case PT_OBJECT:
277     default:             ulRet = 0u;
278                          break;
279     }
280 
281     return ulRet;
282 }
283 
284 /*************************************************************************
285  * FPropContainsProp@12 (MAPI32.78)
286  *
287  * Find a property with a given property tag in a property array.
288  *
289  * PARAMS
290  *  lpHaystack [I] Property to match to
291  *  lpNeedle   [I] Property to find in lpHaystack
292  *  ulFuzzy    [I] Flags controlling match type and strictness (FL_* flags from "mapidefs.h")
293  *
294  * RETURNS
295  *  TRUE, if lpNeedle matches lpHaystack according to the criteria of ulFuzzy.
296  *
297  * NOTES
298  *  Only property types of PT_STRING8 and PT_BINARY are handled by this function.
299  */
300 BOOL WINAPI FPropContainsProp(LPSPropValue lpHaystack, LPSPropValue lpNeedle, ULONG ulFuzzy)
301 {
302     TRACE("(%p,%p,0x%08x)\n", lpHaystack, lpNeedle, ulFuzzy);
303 
304     if (FBadProp(lpHaystack) || FBadProp(lpNeedle) ||
305         PROP_TYPE(lpHaystack->ulPropTag) != PROP_TYPE(lpNeedle->ulPropTag))
306         return FALSE;
307 
308     /* FIXME: Do later versions support Unicode as well? */
309 
310     if (PROP_TYPE(lpHaystack->ulPropTag) == PT_STRING8)
311     {
312         DWORD dwFlags = 0, dwNeedleLen, dwHaystackLen;
313 
314         if (ulFuzzy & FL_IGNORECASE)
315             dwFlags |= NORM_IGNORECASE;
316         if (ulFuzzy & FL_IGNORENONSPACE)
317             dwFlags |= NORM_IGNORENONSPACE;
318         if (ulFuzzy & FL_LOOSE)
319             dwFlags |= (NORM_IGNORECASE|NORM_IGNORENONSPACE|NORM_IGNORESYMBOLS);
320 
321         dwNeedleLen = lstrlenA(lpNeedle->Value.lpszA);
322         dwHaystackLen = lstrlenA(lpHaystack->Value.lpszA);
323 
324         if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_PREFIX)
325         {
326             if (dwNeedleLen <= dwHaystackLen &&
327                 CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
328                                lpHaystack->Value.lpszA, dwNeedleLen,
329                                lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
330                 return TRUE; /* needle is a prefix of haystack */
331         }
332         else if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_SUBSTRING)
333         {
334             LPSTR (WINAPI *pStrChrFn)(LPCSTR,WORD) = StrChrA;
335             LPSTR lpStr = lpHaystack->Value.lpszA;
336 
337             if (dwFlags & NORM_IGNORECASE)
338                 pStrChrFn = StrChrIA;
339 
340             while ((lpStr = pStrChrFn(lpStr, *lpNeedle->Value.lpszA)) != NULL)
341             {
342                 dwHaystackLen -= (lpStr - lpHaystack->Value.lpszA);
343                 if (dwNeedleLen <= dwHaystackLen &&
344                     CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
345                                lpStr, dwNeedleLen,
346                                lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
347                     return TRUE; /* needle is a substring of haystack */
348                 lpStr++;
349             }
350         }
351         else if (CompareStringA(LOCALE_USER_DEFAULT, dwFlags,
352                                 lpHaystack->Value.lpszA, dwHaystackLen,
353                                 lpNeedle->Value.lpszA, dwNeedleLen) == CSTR_EQUAL)
354             return TRUE; /* full string match */
355     }
356     else if (PROP_TYPE(lpHaystack->ulPropTag) == PT_BINARY)
357     {
358         if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_PREFIX)
359         {
360             if (lpNeedle->Value.bin.cb <= lpHaystack->Value.bin.cb &&
361                 !memcmp(lpNeedle->Value.bin.lpb, lpHaystack->Value.bin.lpb,
362                         lpNeedle->Value.bin.cb))
363                 return TRUE; /* needle is a prefix of haystack */
364         }
365         else if ((ulFuzzy & (FL_SUBSTRING|FL_PREFIX)) == FL_SUBSTRING)
366         {
367             ULONG ulLen = lpHaystack->Value.bin.cb;
368             LPBYTE lpb = lpHaystack->Value.bin.lpb;
369 
370             while ((lpb = memchr(lpb, *lpNeedle->Value.bin.lpb, ulLen)) != NULL)
371             {
372                 ulLen = lpHaystack->Value.bin.cb - (lpb - lpHaystack->Value.bin.lpb);
373                 if (lpNeedle->Value.bin.cb <= ulLen &&
374                     !memcmp(lpNeedle->Value.bin.lpb, lpb, lpNeedle->Value.bin.cb))
375                     return TRUE; /* needle is a substring of haystack */
376                 lpb++;
377             }
378         }
379         else if (!LPropCompareProp(lpHaystack, lpNeedle))
380             return TRUE; /* needle is an exact match with haystack */
381 
382     }
383     return FALSE;
384 }
385 
386 /*************************************************************************
387  * FPropCompareProp@12 (MAPI32.79)
388  *
389  * Compare two properties.
390  *
391  * PARAMS
392  *  lpPropLeft  [I] Left hand property to compare to lpPropRight
393  *  ulOp        [I] Comparison operator (RELOP_* enum from "mapidefs.h")
394  *  lpPropRight [I] Right hand property to compare to lpPropLeft
395  *
396  * RETURNS
397  *  TRUE, if the comparison is true, FALSE otherwise.
398  */
399 BOOL WINAPI FPropCompareProp(LPSPropValue lpPropLeft, ULONG ulOp, LPSPropValue lpPropRight)
400 {
401     LONG iCmp;
402 
403     TRACE("(%p,%d,%p)\n", lpPropLeft, ulOp, lpPropRight);
404 
405     if (ulOp > RELOP_RE || FBadProp(lpPropLeft) || FBadProp(lpPropRight))
406         return FALSE;
407 
408     if (ulOp == RELOP_RE)
409     {
410         FIXME("Comparison operator RELOP_RE not yet implemented!\n");
411         return FALSE;
412     }
413 
414     iCmp = LPropCompareProp(lpPropLeft, lpPropRight);
415 
416     switch (ulOp)
417     {
418     case RELOP_LT: return iCmp <  0;
419     case RELOP_LE: return iCmp <= 0;
420     case RELOP_GT: return iCmp >  0;
421     case RELOP_GE: return iCmp >= 0;
422     case RELOP_EQ: return iCmp == 0;
423     case RELOP_NE: return iCmp != 0;
424     }
425     return FALSE;
426 }
427 
428 /*************************************************************************
429  * LPropCompareProp@8 (MAPI32.80)
430  *
431  * Compare two properties.
432  *
433  * PARAMS
434  *  lpPropLeft  [I] Left hand property to compare to lpPropRight
435  *  lpPropRight [I] Right hand property to compare to lpPropLeft
436  *
437  * RETURNS
438  *  An integer less than, equal to or greater than 0, indicating that
439  *  lpszStr is less than, the same, or greater than lpszComp.
440  */
441 LONG WINAPI LPropCompareProp(LPSPropValue lpPropLeft, LPSPropValue lpPropRight)
442 {
443     LONG iRet;
444 
445     TRACE("(%p->0x%08x,%p->0x%08x)\n", lpPropLeft, lpPropLeft->ulPropTag,
446           lpPropRight, lpPropRight->ulPropTag);
447 
448     /* If the properties are not the same, sort by property type */
449     if (PROP_TYPE(lpPropLeft->ulPropTag) != PROP_TYPE(lpPropRight->ulPropTag))
450         return (LONG)PROP_TYPE(lpPropLeft->ulPropTag) - (LONG)PROP_TYPE(lpPropRight->ulPropTag);
451 
452     switch (PROP_TYPE(lpPropLeft->ulPropTag))
453     {
454     case PT_UNSPECIFIED:
455     case PT_NULL:
456         return 0; /* NULLs are equal */
457     case PT_I2:
458         return lpPropLeft->Value.i - lpPropRight->Value.i;
459     case PT_I4:
460         return lpPropLeft->Value.l - lpPropRight->Value.l;
461     case PT_I8:
462         if (lpPropLeft->Value.li.QuadPart > lpPropRight->Value.li.QuadPart)
463             return 1;
464         if (lpPropLeft->Value.li.QuadPart == lpPropRight->Value.li.QuadPart)
465             return 0;
466         return -1;
467     case PT_R4:
468         if (lpPropLeft->Value.flt > lpPropRight->Value.flt)
469             return 1;
470         if (lpPropLeft->Value.flt == lpPropRight->Value.flt)
471             return 0;
472         return -1;
473     case PT_APPTIME:
474     case PT_R8:
475         if (lpPropLeft->Value.dbl > lpPropRight->Value.dbl)
476             return 1;
477         if (lpPropLeft->Value.dbl == lpPropRight->Value.dbl)
478             return 0;
479         return -1;
480     case PT_CURRENCY:
481         if (lpPropLeft->Value.cur.int64 > lpPropRight->Value.cur.int64)
482             return 1;
483         if (lpPropLeft->Value.cur.int64 == lpPropRight->Value.cur.int64)
484             return 0;
485         return -1;
486     case PT_SYSTIME:
487         return CompareFileTime(&lpPropLeft->Value.ft, &lpPropRight->Value.ft);
488     case PT_BOOLEAN:
489         return (lpPropLeft->Value.b ? 1 : 0) - (lpPropRight->Value.b ? 1 : 0);
490     case PT_BINARY:
491         if (lpPropLeft->Value.bin.cb == lpPropRight->Value.bin.cb)
492             iRet = memcmp(lpPropLeft->Value.bin.lpb, lpPropRight->Value.bin.lpb,
493                           lpPropLeft->Value.bin.cb);
494         else
495         {
496             iRet = memcmp(lpPropLeft->Value.bin.lpb, lpPropRight->Value.bin.lpb,
497                           min(lpPropLeft->Value.bin.cb, lpPropRight->Value.bin.cb));
498 
499             if (!iRet)
500                 iRet = lpPropLeft->Value.bin.cb - lpPropRight->Value.bin.cb;
501         }
502         return iRet;
503     case PT_STRING8:
504         return lstrcmpA(lpPropLeft->Value.lpszA, lpPropRight->Value.lpszA);
505     case PT_UNICODE:
506         return lstrcmpW(lpPropLeft->Value.lpszW, lpPropRight->Value.lpszW);
507     case PT_ERROR:
508         if (lpPropLeft->Value.err > lpPropRight->Value.err)
509             return 1;
510         if (lpPropLeft->Value.err == lpPropRight->Value.err)
511             return 0;
512         return -1;
513     case PT_CLSID:
514         return memcmp(lpPropLeft->Value.lpguid, lpPropRight->Value.lpguid,
515                       sizeof(GUID));
516     }
517     FIXME("Unhandled property type %d\n", PROP_TYPE(lpPropLeft->ulPropTag));
518     return 0;
519 }
520 
521 /*************************************************************************
522  * HrGetOneProp@8 (MAPI32.135)
523  *
524  * Get a property value from an IMAPIProp object.
525  *
526  * PARAMS
527  *  lpIProp   [I] IMAPIProp object to get the property value in
528  *  ulPropTag [I] Property tag of the property to get
529  *  lppProp   [O] Destination for the returned property
530  *
531  * RETURNS
532  *  Success: S_OK. *lppProp contains the property value requested.
533  *  Failure: MAPI_E_NOT_FOUND, if no property value has the tag given by ulPropTag.
534  */
535 HRESULT WINAPI HrGetOneProp(LPMAPIPROP lpIProp, ULONG ulPropTag, LPSPropValue *lppProp)
536 {
537     SPropTagArray pta;
538     ULONG ulCount;
539     HRESULT hRet;
540 
541     TRACE("(%p,%d,%p)\n", lpIProp, ulPropTag, lppProp);
542 
543     pta.cValues = 1u;
544     pta.aulPropTag[0] = ulPropTag;
545     hRet = IMAPIProp_GetProps(lpIProp, &pta, 0u, &ulCount, lppProp);
546     if (hRet == MAPI_W_ERRORS_RETURNED)
547     {
548         MAPIFreeBuffer(*lppProp);
549         *lppProp = NULL;
550         hRet = MAPI_E_NOT_FOUND;
551     }
552     return hRet;
553 }
554 
555 /*************************************************************************
556  * HrSetOneProp@8 (MAPI32.136)
557  *
558  * Set a property value in an IMAPIProp object.
559  *
560  * PARAMS
561  *  lpIProp [I] IMAPIProp object to set the property value in
562  *  lpProp  [I] Property value to set
563  *
564  * RETURNS
565  *  Success: S_OK. The value in lpProp is set in lpIProp.
566  *  Failure: An error result from IMAPIProp_SetProps().
567  */
568 HRESULT WINAPI HrSetOneProp(LPMAPIPROP lpIProp, LPSPropValue lpProp)
569 {
570     TRACE("(%p,%p)\n", lpIProp, lpProp);
571 
572     return IMAPIProp_SetProps(lpIProp, 1u, lpProp, NULL);
573 }
574 
575 /*************************************************************************
576  * FPropExists@8 (MAPI32.137)
577  *
578  * Find a property with a given property tag in an IMAPIProp object.
579  *
580  * PARAMS
581  *  lpIProp   [I] IMAPIProp object to find the property tag in
582  *  ulPropTag [I] Property tag to find
583  *
584  * RETURNS
585  *  TRUE, if ulPropTag matches a property held in lpIProp,
586  *  FALSE, otherwise.
587  *
588  * NOTES
589  *  if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
590  *  Ids need to match for a successful match to occur.
591  */
592  BOOL WINAPI FPropExists(LPMAPIPROP lpIProp, ULONG ulPropTag)
593  {
594     BOOL bRet = FALSE;
595 
596     TRACE("(%p,%d)\n", lpIProp, ulPropTag);
597 
598     if (lpIProp)
599     {
600         LPSPropTagArray lpTags;
601         ULONG i;
602 
603         if (FAILED(IMAPIProp_GetPropList(lpIProp, 0u, &lpTags)))
604             return FALSE;
605 
606         for (i = 0; i < lpTags->cValues; i++)
607         {
608             if (!FBadPropTag(lpTags->aulPropTag[i]) &&
609                 (lpTags->aulPropTag[i] == ulPropTag ||
610                  (PROP_TYPE(ulPropTag) == PT_UNSPECIFIED &&
611                   PROP_ID(lpTags->aulPropTag[i]) == lpTags->aulPropTag[i])))
612             {
613                 bRet = TRUE;
614                 break;
615             }
616         }
617         MAPIFreeBuffer(lpTags);
618     }
619     return bRet;
620 }
621 
622 /*************************************************************************
623  * PpropFindProp@12 (MAPI32.138)
624  *
625  * Find a property with a given property tag in a property array.
626  *
627  * PARAMS
628  *  lpProps   [I] Property array to search
629  *  cValues   [I] Number of properties in lpProps
630  *  ulPropTag [I] Property tag to find
631  *
632  * RETURNS
633  *  A pointer to the matching property, or NULL if none was found.
634  *
635  * NOTES
636  *  if ulPropTag has a property type of PT_UNSPECIFIED, then only the property
637  *  Ids need to match for a successful match to occur.
638  */
639 LPSPropValue WINAPI PpropFindProp(LPSPropValue lpProps, ULONG cValues, ULONG ulPropTag)
640 {
641     TRACE("(%p,%d,%d)\n", lpProps, cValues, ulPropTag);
642 
643     if (lpProps && cValues)
644     {
645         ULONG i;
646         for (i = 0; i < cValues; i++)
647         {
648             if (!FBadPropTag(lpProps[i].ulPropTag) &&
649                 (lpProps[i].ulPropTag == ulPropTag ||
650                  (PROP_TYPE(ulPropTag) == PT_UNSPECIFIED &&
651                   PROP_ID(lpProps[i].ulPropTag) == PROP_ID(ulPropTag))))
652                 return &lpProps[i];
653         }
654     }
655     return NULL;
656 }
657 
658 /*************************************************************************
659  * FreePadrlist@4 (MAPI32.139)
660  *
661  * Free the memory used by an address book list.
662  *
663  * PARAMS
664  *  lpAddrs [I] Address book list to free
665  *
666  * RETURNS
667  *  Nothing.
668  */
669 VOID WINAPI FreePadrlist(LPADRLIST lpAddrs)
670 {
671     TRACE("(%p)\n", lpAddrs);
672 
673     /* Structures are binary compatible; use the same implementation */
674     FreeProws((LPSRowSet)lpAddrs);
675 }
676 
677 /*************************************************************************
678  * FreeProws@4 (MAPI32.140)
679  *
680  * Free the memory used by a row set.
681  *
682  * PARAMS
683  *  lpRowSet [I] Row set to free
684  *
685  * RETURNS
686  *  Nothing.
687  */
688 VOID WINAPI FreeProws(LPSRowSet lpRowSet)
689 {
690     TRACE("(%p)\n", lpRowSet);
691 
692     if (lpRowSet)
693     {
694         ULONG i;
695 
696         for (i = 0; i < lpRowSet->cRows; i++)
697             MAPIFreeBuffer(lpRowSet->aRow[i].lpProps);
698 
699         MAPIFreeBuffer(lpRowSet);
700     }
701 }
702 
703 /*************************************************************************
704  * ScCountProps@12 (MAPI32.170)
705  *
706  * Validate and determine the length of an array of properties.
707  *
708  * PARAMS
709  *  iCount  [I] Length of the lpProps array
710  *  lpProps [I] Array of properties to validate/size
711  *  pcBytes [O] If non-NULL, destination for the size of the property array
712  *
713  * RETURNS
714  *  Success: S_OK. If pcBytes is non-NULL, it contains the size of the
715  *           properties array.
716  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid or validation
717  *           of the property array fails.
718  */
719 SCODE WINAPI ScCountProps(INT iCount, LPSPropValue lpProps, ULONG *pcBytes)
720 {
721     ULONG i, ulCount = iCount, ulBytes = 0;
722 
723     TRACE("(%d,%p,%p)\n", iCount, lpProps, pcBytes);
724 
725     if (iCount <= 0 || !lpProps ||
726         IsBadReadPtr(lpProps, iCount * sizeof(SPropValue)))
727         return MAPI_E_INVALID_PARAMETER;
728 
729     for (i = 0; i < ulCount; i++)
730     {
731         ULONG ulPropSize = 0;
732 
733         if (FBadProp(&lpProps[i]) || lpProps[i].ulPropTag == PROP_ID_NULL ||
734             lpProps[i].ulPropTag == PROP_ID_INVALID)
735             return MAPI_E_INVALID_PARAMETER;
736 
737         if (PROP_TYPE(lpProps[i].ulPropTag) != PT_OBJECT)
738         {
739             ulPropSize = UlPropSize(&lpProps[i]);
740             if (!ulPropSize)
741                 return MAPI_E_INVALID_PARAMETER;
742         }
743 
744         switch (PROP_TYPE(lpProps[i].ulPropTag))
745         {
746         case PT_STRING8:
747         case PT_UNICODE:
748         case PT_CLSID:
749         case PT_BINARY:
750         case PT_MV_I2:
751         case PT_MV_I4:
752         case PT_MV_I8:
753         case PT_MV_R4:
754         case PT_MV_R8:
755         case PT_MV_CURRENCY:
756         case PT_MV_SYSTIME:
757         case PT_MV_APPTIME:
758             ulPropSize += sizeof(SPropValue);
759             break;
760         case PT_MV_CLSID:
761             ulPropSize += lpProps[i].Value.MVszA.cValues * sizeof(char*) + sizeof(SPropValue);
762             break;
763         case PT_MV_STRING8:
764         case PT_MV_UNICODE:
765             ulPropSize += lpProps[i].Value.MVszA.cValues * sizeof(char*) + sizeof(SPropValue);
766             break;
767         case PT_MV_BINARY:
768             ulPropSize += lpProps[i].Value.MVbin.cValues * sizeof(SBinary) + sizeof(SPropValue);
769             break;
770         default:
771             ulPropSize = sizeof(SPropValue);
772             break;
773         }
774         ulBytes += ulPropSize;
775     }
776     if (pcBytes)
777         *pcBytes = ulBytes;
778 
779     return S_OK;
780 }
781 
782 /*************************************************************************
783  * ScCopyProps@16 (MAPI32.171)
784  *
785  * Copy an array of property values into a buffer suited for serialisation.
786  *
787  * PARAMS
788  *  cValues   [I] Number of properties in lpProps
789  *  lpProps   [I] Property array to copy
790  *  lpDst     [O] Destination for the serialised data
791  *  lpCount   [O] If non-NULL, destination for the number of bytes of data written to lpDst
792  *
793  * RETURNS
794  *  Success: S_OK. lpDst contains the serialised data from lpProps.
795  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
796  *
797  * NOTES
798  *  The resulting property value array is stored in a contiguous block starting at lpDst.
799  */
800 SCODE WINAPI ScCopyProps(int cValues, LPSPropValue lpProps, LPVOID lpDst, ULONG *lpCount)
801 {
802     LPSPropValue lpDest = (LPSPropValue)lpDst;
803     char *lpDataDest = (char *)(lpDest + cValues);
804     ULONG ulLen, i;
805     int iter;
806 
807     TRACE("(%d,%p,%p,%p)\n", cValues, lpProps, lpDst, lpCount);
808 
809     if (!lpProps || cValues < 0 || !lpDest)
810         return MAPI_E_INVALID_PARAMETER;
811 
812     memcpy(lpDst, lpProps, cValues * sizeof(SPropValue));
813 
814     for (iter = 0; iter < cValues; iter++)
815     {
816         switch (PROP_TYPE(lpProps->ulPropTag))
817         {
818         case PT_CLSID:
819             lpDest->Value.lpguid = (LPGUID)lpDataDest;
820             *lpDest->Value.lpguid = *lpProps->Value.lpguid;
821             lpDataDest += sizeof(GUID);
822             break;
823         case PT_STRING8:
824             ulLen = lstrlenA(lpProps->Value.lpszA) + 1u;
825             lpDest->Value.lpszA = lpDataDest;
826             memcpy(lpDest->Value.lpszA, lpProps->Value.lpszA, ulLen);
827             lpDataDest += ulLen;
828             break;
829         case PT_UNICODE:
830             ulLen = (lstrlenW(lpProps->Value.lpszW) + 1u) * sizeof(WCHAR);
831             lpDest->Value.lpszW = (LPWSTR)lpDataDest;
832             memcpy(lpDest->Value.lpszW, lpProps->Value.lpszW, ulLen);
833             lpDataDest += ulLen;
834             break;
835         case PT_BINARY:
836             lpDest->Value.bin.lpb = (LPBYTE)lpDataDest;
837             memcpy(lpDest->Value.bin.lpb, lpProps->Value.bin.lpb, lpProps->Value.bin.cb);
838             lpDataDest += lpProps->Value.bin.cb;
839             break;
840         default:
841             if (lpProps->ulPropTag & MV_FLAG)
842             {
843                 lpDest->Value.MVi.cValues = lpProps->Value.MVi.cValues;
844                 /* Note: Assignment uses lppszA but covers all cases by union aliasing */
845                 lpDest->Value.MVszA.lppszA = (char**)lpDataDest;
846 
847                 switch (PROP_TYPE(lpProps->ulPropTag))
848                 {
849                 case PT_MV_STRING8:
850                 {
851                     lpDataDest += lpProps->Value.MVszA.cValues * sizeof(char *);
852 
853                     for (i = 0; i < lpProps->Value.MVszA.cValues; i++)
854                     {
855                         ULONG ulStrLen = lstrlenA(lpProps->Value.MVszA.lppszA[i]) + 1u;
856 
857                         lpDest->Value.MVszA.lppszA[i] = lpDataDest;
858                         memcpy(lpDataDest, lpProps->Value.MVszA.lppszA[i], ulStrLen);
859                         lpDataDest += ulStrLen;
860                     }
861                     break;
862                 }
863                 case PT_MV_UNICODE:
864                 {
865                     lpDataDest += lpProps->Value.MVszW.cValues * sizeof(WCHAR *);
866 
867                     for (i = 0; i < lpProps->Value.MVszW.cValues; i++)
868                     {
869                         ULONG ulStrLen = (lstrlenW(lpProps->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
870 
871                         lpDest->Value.MVszW.lppszW[i] = (LPWSTR)lpDataDest;
872                         memcpy(lpDataDest, lpProps->Value.MVszW.lppszW[i], ulStrLen);
873                         lpDataDest += ulStrLen;
874                     }
875                     break;
876                 }
877                 case PT_MV_BINARY:
878                 {
879                     lpDataDest += lpProps->Value.MVszW.cValues * sizeof(SBinary);
880 
881                     for (i = 0; i < lpProps->Value.MVszW.cValues; i++)
882                     {
883                         lpDest->Value.MVbin.lpbin[i].cb = lpProps->Value.MVbin.lpbin[i].cb;
884                         lpDest->Value.MVbin.lpbin[i].lpb = (LPBYTE)lpDataDest;
885                         memcpy(lpDataDest, lpProps->Value.MVbin.lpbin[i].lpb, lpDest->Value.MVbin.lpbin[i].cb);
886                         lpDataDest += lpDest->Value.MVbin.lpbin[i].cb;
887                     }
888                     break;
889                 }
890                 default:
891                     /* No embedded pointers, just copy the data over */
892                     ulLen = UlPropSize(lpProps);
893                     memcpy(lpDest->Value.MVi.lpi, lpProps->Value.MVi.lpi, ulLen);
894                     lpDataDest += ulLen;
895                     break;
896                 }
897                 break;
898             }
899         }
900         lpDest++;
901         lpProps++;
902     }
903     if (lpCount)
904         *lpCount = lpDataDest - (char *)lpDst;
905 
906     return S_OK;
907 }
908 
909 /*************************************************************************
910  * ScRelocProps@20 (MAPI32.172)
911  *
912  * Relocate the pointers in an array of property values after it has been copied.
913  *
914  * PARAMS
915  *  cValues   [I] Number of properties in lpProps
916  *  lpProps   [O] Property array to relocate the pointers in.
917  *  lpOld     [I] Position where the data was copied from
918  *  lpNew     [I] Position where the data was copied to
919  *  lpCount   [O] If non-NULL, destination for the number of bytes of data at lpDst
920  *
921  * RETURNS
922  *  Success: S_OK. Any pointers in lpProps are relocated.
923  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
924  *
925  * NOTES
926  *  MSDN states that this function can be used for serialisation by passing
927  *  NULL as either lpOld or lpNew, thus converting any pointers in lpProps
928  *  between offsets and pointers. This does not work in native (it crashes),
929  *  and cannot be made to work in Wine because the original interface design
930  *  is deficient. The only use left for this function is to remap pointers
931  *  in a contiguous property array that has been copied with memcpy() to
932  *  another memory location.
933  */
934 SCODE WINAPI ScRelocProps(int cValues, LPSPropValue lpProps, LPVOID lpOld,
935                           LPVOID lpNew, ULONG *lpCount)
936 {
937     static const BOOL bBadPtr = TRUE; /* Windows bug - Assumes source is bad */
938     LPSPropValue lpDest = lpProps;
939     ULONG ulCount = cValues * sizeof(SPropValue);
940     ULONG ulLen, i;
941     int iter;
942 
943     TRACE("(%d,%p,%p,%p,%p)\n", cValues, lpProps, lpOld, lpNew, lpCount);
944 
945     if (!lpProps || cValues < 0 || !lpOld || !lpNew)
946         return MAPI_E_INVALID_PARAMETER;
947 
948     /* The reason native doesn't work as MSDN states is that it assumes that
949      * the lpProps pointer contains valid pointers. This is obviously not
950      * true if the array is being read back from serialisation (the pointers
951      * are just offsets). Native can't actually work converting the pointers to
952      * offsets either, because it converts any array pointers to offsets then
953      * _dereferences the offset_ in order to convert the array elements!
954      *
955      * The code below would handle both cases except that the design of this
956      * function makes it impossible to know when the pointers in lpProps are
957      * valid. If both lpOld and lpNew are non-NULL, native reads the pointers
958      * after converting them, so we must do the same. It seems this
959      * functionality was never tested by MS.
960      */
961 
962 #define RELOC_PTR(p) (((char*)(p)) - (char*)lpOld + (char*)lpNew)
963 
964     for (iter = 0; iter < cValues; iter++)
965     {
966         switch (PROP_TYPE(lpDest->ulPropTag))
967         {
968         case PT_CLSID:
969             lpDest->Value.lpguid = (LPGUID)RELOC_PTR(lpDest->Value.lpguid);
970             ulCount += sizeof(GUID);
971             break;
972         case PT_STRING8:
973             ulLen = bBadPtr ? 0 : lstrlenA(lpDest->Value.lpszA) + 1u;
974             lpDest->Value.lpszA = RELOC_PTR(lpDest->Value.lpszA);
975             if (bBadPtr)
976                 ulLen = lstrlenA(lpDest->Value.lpszA) + 1u;
977             ulCount += ulLen;
978             break;
979         case PT_UNICODE:
980             ulLen = bBadPtr ? 0 : (lstrlenW(lpDest->Value.lpszW) + 1u) * sizeof(WCHAR);
981             lpDest->Value.lpszW = (LPWSTR)RELOC_PTR(lpDest->Value.lpszW);
982             if (bBadPtr)
983                 ulLen = (lstrlenW(lpDest->Value.lpszW) + 1u) * sizeof(WCHAR);
984             ulCount += ulLen;
985             break;
986         case PT_BINARY:
987             lpDest->Value.bin.lpb = (LPBYTE)RELOC_PTR(lpDest->Value.bin.lpb);
988             ulCount += lpDest->Value.bin.cb;
989             break;
990         default:
991             if (lpDest->ulPropTag & MV_FLAG)
992             {
993                 /* Since we have to access the array elements, don't map the
994                  * array unless it is invalid (otherwise, map it at the end)
995                  */
996                 if (bBadPtr)
997                     lpDest->Value.MVszA.lppszA = (LPSTR*)RELOC_PTR(lpDest->Value.MVszA.lppszA);
998 
999                 switch (PROP_TYPE(lpProps->ulPropTag))
1000                 {
1001                 case PT_MV_STRING8:
1002                 {
1003                     ulCount += lpDest->Value.MVszA.cValues * sizeof(char *);
1004 
1005                     for (i = 0; i < lpDest->Value.MVszA.cValues; i++)
1006                     {
1007                         ULONG ulStrLen = bBadPtr ? 0 : lstrlenA(lpDest->Value.MVszA.lppszA[i]) + 1u;
1008 
1009                         lpDest->Value.MVszA.lppszA[i] = RELOC_PTR(lpDest->Value.MVszA.lppszA[i]);
1010                         if (bBadPtr)
1011                             ulStrLen = lstrlenA(lpDest->Value.MVszA.lppszA[i]) + 1u;
1012                         ulCount += ulStrLen;
1013                     }
1014                     break;
1015                 }
1016                 case PT_MV_UNICODE:
1017                 {
1018                     ulCount += lpDest->Value.MVszW.cValues * sizeof(WCHAR *);
1019 
1020                     for (i = 0; i < lpDest->Value.MVszW.cValues; i++)
1021                     {
1022                         ULONG ulStrLen = bBadPtr ? 0 : (lstrlenW(lpDest->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
1023 
1024                         lpDest->Value.MVszW.lppszW[i] = (LPWSTR)RELOC_PTR(lpDest->Value.MVszW.lppszW[i]);
1025                         if (bBadPtr)
1026                             ulStrLen = (lstrlenW(lpDest->Value.MVszW.lppszW[i]) + 1u) * sizeof(WCHAR);
1027                         ulCount += ulStrLen;
1028                     }
1029                     break;
1030                 }
1031                 case PT_MV_BINARY:
1032                 {
1033                     ulCount += lpDest->Value.MVszW.cValues * sizeof(SBinary);
1034 
1035                     for (i = 0; i < lpDest->Value.MVszW.cValues; i++)
1036                     {
1037                         lpDest->Value.MVbin.lpbin[i].lpb = (LPBYTE)RELOC_PTR(lpDest->Value.MVbin.lpbin[i].lpb);
1038                         ulCount += lpDest->Value.MVbin.lpbin[i].cb;
1039                     }
1040                     break;
1041                 }
1042                 default:
1043                     ulCount += UlPropSize(lpDest);
1044                     break;
1045                 }
1046                 if (!bBadPtr)
1047                     lpDest->Value.MVszA.lppszA = (LPSTR*)RELOC_PTR(lpDest->Value.MVszA.lppszA);
1048                 break;
1049             }
1050         }
1051         lpDest++;
1052     }
1053     if (lpCount)
1054         *lpCount = ulCount;
1055 
1056     return S_OK;
1057 }
1058 
1059 /*************************************************************************
1060  * LpValFindProp@12 (MAPI32.173)
1061  *
1062  * Find a property with a given property id in a property array.
1063  *
1064  * PARAMS
1065  *  ulPropTag [I] Property tag containing property id to find
1066  *  cValues   [I] Number of properties in lpProps
1067  *  lpProps   [I] Property array to search
1068  *
1069  * RETURNS
1070  *  A pointer to the matching property, or NULL if none was found.
1071  *
1072  * NOTES
1073  *  This function matches only on the property id and does not care if the
1074  *  property types differ.
1075  */
1076 LPSPropValue WINAPI LpValFindProp(ULONG ulPropTag, ULONG cValues, LPSPropValue lpProps)
1077 {
1078     TRACE("(%d,%d,%p)\n", ulPropTag, cValues, lpProps);
1079 
1080     if (lpProps && cValues)
1081     {
1082         ULONG i;
1083         for (i = 0; i < cValues; i++)
1084         {
1085             if (PROP_ID(ulPropTag) == PROP_ID(lpProps[i].ulPropTag))
1086                 return &lpProps[i];
1087         }
1088     }
1089     return NULL;
1090 }
1091 
1092 /*************************************************************************
1093  * ScDupPropset@16 (MAPI32.174)
1094  *
1095  * Duplicate a property value array into a contiguous block of memory.
1096  *
1097  * PARAMS
1098  *  cValues   [I] Number of properties in lpProps
1099  *  lpProps   [I] Property array to duplicate
1100  *  lpAlloc   [I] Memory allocation function, use MAPIAllocateBuffer()
1101  *  lpNewProp [O] Destination for the newly duplicated property value array
1102  *
1103  * RETURNS
1104  *  Success: S_OK. *lpNewProp contains the duplicated array.
1105  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1106  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1107  */
1108 SCODE WINAPI ScDupPropset(int cValues, LPSPropValue lpProps,
1109                           LPALLOCATEBUFFER lpAlloc, LPSPropValue *lpNewProp)
1110 {
1111     ULONG ulCount;
1112     SCODE sc;
1113 
1114     TRACE("(%d,%p,%p,%p)\n", cValues, lpProps, lpAlloc, lpNewProp);
1115 
1116     sc = ScCountProps(cValues, lpProps, &ulCount);
1117     if (SUCCEEDED(sc))
1118     {
1119         sc = lpAlloc(ulCount, (LPVOID*)lpNewProp);
1120         if (SUCCEEDED(sc))
1121             sc = ScCopyProps(cValues, lpProps, *lpNewProp, &ulCount);
1122     }
1123     return sc;
1124 }
1125 
1126 /*************************************************************************
1127  * FBadRglpszA@8 (MAPI32.175)
1128  *
1129  * Determine if an array of strings is invalid
1130  *
1131  * PARAMS
1132  *  lppszStrs [I] Array of strings to check
1133  *  ulCount   [I] Number of strings in lppszStrs
1134  *
1135  * RETURNS
1136  *  TRUE, if lppszStrs is invalid, FALSE otherwise.
1137  */
1138 BOOL WINAPI FBadRglpszA(LPSTR *lppszStrs, ULONG ulCount)
1139 {
1140     ULONG i;
1141 
1142     TRACE("(%p,%d)\n", lppszStrs, ulCount);
1143 
1144     if (!ulCount)
1145         return FALSE;
1146 
1147     if (!lppszStrs || IsBadReadPtr(lppszStrs, ulCount * sizeof(LPWSTR)))
1148         return TRUE;
1149 
1150     for (i = 0; i < ulCount; i++)
1151     {
1152         if (!lppszStrs[i] || IsBadStringPtrA(lppszStrs[i], -1))
1153             return TRUE;
1154     }
1155     return FALSE;
1156 }
1157 
1158 /*************************************************************************
1159  * FBadRglpszW@8 (MAPI32.176)
1160  *
1161  * See FBadRglpszA.
1162  */
1163 BOOL WINAPI FBadRglpszW(LPWSTR *lppszStrs, ULONG ulCount)
1164 {
1165     ULONG i;
1166 
1167     TRACE("(%p,%d)\n", lppszStrs, ulCount);
1168 
1169     if (!ulCount)
1170         return FALSE;
1171 
1172     if (!lppszStrs || IsBadReadPtr(lppszStrs, ulCount * sizeof(LPWSTR)))
1173         return TRUE;
1174 
1175     for (i = 0; i < ulCount; i++)
1176     {
1177         if (!lppszStrs[i] || IsBadStringPtrW(lppszStrs[i], -1))
1178             return TRUE;
1179     }
1180     return FALSE;
1181 }
1182 
1183 /*************************************************************************
1184  * FBadRowSet@4 (MAPI32.177)
1185  *
1186  * Determine if a row is invalid
1187  *
1188  * PARAMS
1189  *  lpRow [I] Row to check
1190  *
1191  * RETURNS
1192  *  TRUE, if lpRow is invalid, FALSE otherwise.
1193  */
1194 BOOL WINAPI FBadRowSet(LPSRowSet lpRowSet)
1195 {
1196     ULONG i;
1197     TRACE("(%p)\n", lpRowSet);
1198 
1199     if (!lpRowSet || IsBadReadPtr(lpRowSet, CbSRowSet(lpRowSet)))
1200         return TRUE;
1201 
1202     for (i = 0; i < lpRowSet->cRows; i++)
1203     {
1204         if (FBadRow(&lpRowSet->aRow[i]))
1205             return TRUE;
1206     }
1207     return FALSE;
1208 }
1209 
1210 /*************************************************************************
1211  * FBadPropTag@4 (MAPI32.179)
1212  *
1213  * Determine if a property tag is invalid
1214  *
1215  * PARAMS
1216  *  ulPropTag [I] Property tag to check
1217  *
1218  * RETURNS
1219  *  TRUE, if ulPropTag is invalid, FALSE otherwise.
1220  */
1221 ULONG WINAPI FBadPropTag(ULONG ulPropTag)
1222 {
1223     TRACE("(0x%08x)\n", ulPropTag);
1224 
1225     switch (ulPropTag & (~MV_FLAG & PROP_TYPE_MASK))
1226     {
1227     case PT_UNSPECIFIED:
1228     case PT_NULL:
1229     case PT_I2:
1230     case PT_LONG:
1231     case PT_R4:
1232     case PT_DOUBLE:
1233     case PT_CURRENCY:
1234     case PT_APPTIME:
1235     case PT_ERROR:
1236     case PT_BOOLEAN:
1237     case PT_OBJECT:
1238     case PT_I8:
1239     case PT_STRING8:
1240     case PT_UNICODE:
1241     case PT_SYSTIME:
1242     case PT_CLSID:
1243     case PT_BINARY:
1244         return FALSE;
1245     }
1246     return TRUE;
1247 }
1248 
1249 /*************************************************************************
1250  * FBadRow@4 (MAPI32.180)
1251  *
1252  * Determine if a row is invalid
1253  *
1254  * PARAMS
1255  *  lpRow [I] Row to check
1256  *
1257  * RETURNS
1258  *  TRUE, if lpRow is invalid, FALSE otherwise.
1259  */
1260 ULONG WINAPI FBadRow(LPSRow lpRow)
1261 {
1262     ULONG i;
1263     TRACE("(%p)\n", lpRow);
1264 
1265     if (!lpRow || IsBadReadPtr(lpRow, sizeof(SRow)) || !lpRow->lpProps ||
1266         IsBadReadPtr(lpRow->lpProps, lpRow->cValues * sizeof(SPropValue)))
1267         return TRUE;
1268 
1269     for (i = 0; i < lpRow->cValues; i++)
1270     {
1271         if (FBadProp(&lpRow->lpProps[i]))
1272             return TRUE;
1273     }
1274     return FALSE;
1275 }
1276 
1277 /*************************************************************************
1278  * FBadProp@4 (MAPI32.181)
1279  *
1280  * Determine if a property is invalid
1281  *
1282  * PARAMS
1283  *  lpProp [I] Property to check
1284  *
1285  * RETURNS
1286  *  TRUE, if lpProp is invalid, FALSE otherwise.
1287  */
1288 ULONG WINAPI FBadProp(LPSPropValue lpProp)
1289 {
1290     if (!lpProp || IsBadReadPtr(lpProp, sizeof(SPropValue)) ||
1291         FBadPropTag(lpProp->ulPropTag))
1292         return TRUE;
1293 
1294     switch (PROP_TYPE(lpProp->ulPropTag))
1295     {
1296     /* Single value properties containing pointers */
1297     case PT_STRING8:
1298         if (!lpProp->Value.lpszA || IsBadStringPtrA(lpProp->Value.lpszA, -1))
1299             return TRUE;
1300         break;
1301     case PT_UNICODE:
1302         if (!lpProp->Value.lpszW || IsBadStringPtrW(lpProp->Value.lpszW, -1))
1303             return TRUE;
1304         break;
1305     case PT_BINARY:
1306         if (IsBadReadPtr(lpProp->Value.bin.lpb, lpProp->Value.bin.cb))
1307             return TRUE;
1308         break;
1309     case PT_CLSID:
1310         if (IsBadReadPtr(lpProp->Value.lpguid, sizeof(GUID)))
1311             return TRUE;
1312         break;
1313 
1314     /* Multiple value properties (arrays) containing no pointers */
1315     case PT_MV_I2:
1316         return PROP_BadArray(lpProp, sizeof(SHORT));
1317     case PT_MV_LONG:
1318         return PROP_BadArray(lpProp, sizeof(LONG));
1319     case PT_MV_LONGLONG:
1320         return PROP_BadArray(lpProp, sizeof(LONG64));
1321     case PT_MV_FLOAT:
1322         return PROP_BadArray(lpProp, sizeof(float));
1323     case PT_MV_SYSTIME:
1324         return PROP_BadArray(lpProp, sizeof(FILETIME));
1325     case PT_MV_APPTIME:
1326     case PT_MV_DOUBLE:
1327         return PROP_BadArray(lpProp, sizeof(double));
1328     case PT_MV_CURRENCY:
1329         return PROP_BadArray(lpProp, sizeof(CY));
1330     case PT_MV_CLSID:
1331         return PROP_BadArray(lpProp, sizeof(GUID));
1332 
1333     /* Multiple value properties containing pointers */
1334     case PT_MV_STRING8:
1335         return FBadRglpszA(lpProp->Value.MVszA.lppszA,
1336                            lpProp->Value.MVszA.cValues);
1337     case PT_MV_UNICODE:
1338         return FBadRglpszW(lpProp->Value.MVszW.lppszW,
1339                            lpProp->Value.MVszW.cValues);
1340     case PT_MV_BINARY:
1341         return FBadEntryList(&lpProp->Value.MVbin);
1342     }
1343     return FALSE;
1344 }
1345 
1346 /*************************************************************************
1347  * FBadColumnSet@4 (MAPI32.182)
1348  *
1349  * Determine if an array of property tags is invalid
1350  *
1351  * PARAMS
1352  *  lpCols [I] Property tag array to check
1353  *
1354  * RETURNS
1355  *  TRUE, if lpCols is invalid, FALSE otherwise.
1356  */
1357 ULONG WINAPI FBadColumnSet(LPSPropTagArray lpCols)
1358 {
1359     ULONG ulRet = FALSE, i;
1360 
1361     TRACE("(%p)\n", lpCols);
1362 
1363     if (!lpCols || IsBadReadPtr(lpCols, CbSPropTagArray(lpCols)))
1364         ulRet = TRUE;
1365     else
1366     {
1367         for (i = 0; i < lpCols->cValues; i++)
1368         {
1369             if ((lpCols->aulPropTag[i] & PROP_TYPE_MASK) == PT_ERROR ||
1370                 FBadPropTag(lpCols->aulPropTag[i]))
1371             {
1372                 ulRet = TRUE;
1373                 break;
1374             }
1375         }
1376     }
1377     TRACE("Returning %s\n", ulRet ? "TRUE" : "FALSE");
1378     return ulRet;
1379 }
1380 
1381 
1382 /**************************************************************************
1383  *  IPropData {MAPI32}
1384  *
1385  * A default Mapi interface to provide manipulation of object properties.
1386  *
1387  * DESCRIPTION
1388  *  This object provides a default interface suitable in some cases as an
1389  *  implementation of the IMAPIProp interface (which has no default
1390  *  implementation). In addition to the IMAPIProp() methods inherited, this
1391  *  interface allows read/write control over access to the object and its
1392  *  individual properties.
1393  *
1394  *  To obtain the default implementation of this interface from Mapi, call
1395  *  CreateIProp().
1396  *
1397  * METHODS
1398  */
1399 
1400 /* A single property in a property data collection */
1401 typedef struct
1402 {
1403   struct list  entry;
1404   ULONG        ulAccess; /* The property value access level */
1405   LPSPropValue value;    /* The property value */
1406 } IPropDataItem, *LPIPropDataItem;
1407 
1408  /* The main property data collection structure */
1409 typedef struct
1410 {
1411     IPropData        IPropData_iface;
1412     LONG             lRef;        /* Reference count */
1413     ALLOCATEBUFFER  *lpAlloc;     /* Memory allocation routine */
1414     ALLOCATEMORE    *lpMore;      /* Linked memory allocation routine */
1415     FREEBUFFER      *lpFree;      /* Memory free routine */
1416     ULONG            ulObjAccess; /* Object access level */
1417     ULONG            ulNumValues; /* Number of items in values list */
1418     struct list      values;      /* List of property values */
1419     CRITICAL_SECTION cs;          /* Lock for thread safety */
1420 } IPropDataImpl;
1421 
1422 static inline IPropDataImpl *impl_from_IPropData(IPropData *iface)
1423 {
1424     return CONTAINING_RECORD(iface, IPropDataImpl, IPropData_iface);
1425 }
1426 
1427 /* Internal - Get a property value, assumes lock is held */
1428 static IPropDataItem *IMAPIPROP_GetValue(IPropDataImpl *This, ULONG ulPropTag)
1429 {
1430     struct list *cursor;
1431 
1432     LIST_FOR_EACH(cursor, &This->values)
1433     {
1434         LPIPropDataItem current = LIST_ENTRY(cursor, IPropDataItem, entry);
1435         /* Note that property types don't have to match, just Id's */
1436         if (PROP_ID(current->value->ulPropTag) == PROP_ID(ulPropTag))
1437             return current;
1438     }
1439     return NULL;
1440 }
1441 
1442 /* Internal - Add a new property value, assumes lock is held */
1443 static IPropDataItem *IMAPIPROP_AddValue(IPropDataImpl *This,
1444                                          LPSPropValue lpProp)
1445 {
1446     LPVOID lpMem;
1447     LPIPropDataItem lpNew;
1448     HRESULT hRet;
1449 
1450     hRet = This->lpAlloc(sizeof(IPropDataItem), &lpMem);
1451 
1452     if (SUCCEEDED(hRet))
1453     {
1454         lpNew = lpMem;
1455         lpNew->ulAccess = IPROP_READWRITE;
1456 
1457         /* Allocate the value separately so we can update it easily */
1458         lpMem = NULL;
1459         hRet = This->lpAlloc(sizeof(SPropValue), &lpMem);
1460         if (SUCCEEDED(hRet))
1461         {
1462             lpNew->value = lpMem;
1463 
1464             hRet = PropCopyMore(lpNew->value, lpProp, This->lpMore, lpMem);
1465             if (SUCCEEDED(hRet))
1466             {
1467                 list_add_tail(&This->values, &lpNew->entry);
1468                 This->ulNumValues++;
1469                 return lpNew;
1470             }
1471             This->lpFree(lpNew->value);
1472         }
1473         This->lpFree(lpNew);
1474     }
1475     return NULL;
1476 }
1477 
1478 /* Internal - Lock an IPropData object */
1479 static inline void IMAPIPROP_Lock(IPropDataImpl *This)
1480 {
1481     EnterCriticalSection(&This->cs);
1482 }
1483 
1484 /* Internal - Unlock an IPropData object */
1485 static inline void IMAPIPROP_Unlock(IPropDataImpl *This)
1486 {
1487     LeaveCriticalSection(&This->cs);
1488 }
1489 
1490 /* This one seems to be missing from mapidefs.h */
1491 #define CbNewSPropProblemArray(c) \
1492     (offsetof(SPropProblemArray,aProblem)+(c)*sizeof(SPropProblem))
1493 
1494 /**************************************************************************
1495  *  IPropData_QueryInterface {MAPI32}
1496  *
1497  * Inherited method from the IUnknown Interface.
1498  * See IUnknown_QueryInterface.
1499  */
1500 static HRESULT WINAPI IPropData_fnQueryInterface(LPPROPDATA iface, REFIID riid, LPVOID *ppvObj)
1501 {
1502     IPropDataImpl *This = impl_from_IPropData(iface);
1503 
1504     TRACE("(%p,%s,%p)\n", This, debugstr_guid(riid), ppvObj);
1505 
1506     if (!ppvObj || !riid)
1507         return MAPI_E_INVALID_PARAMETER;
1508 
1509     *ppvObj = NULL;
1510 
1511     if(IsEqualIID(riid, &IID_IUnknown) ||
1512        IsEqualIID(riid, &IID_IMAPIProp) ||
1513        IsEqualIID(riid, &IID_IMAPIPropData))
1514     {
1515         *ppvObj = &This->IPropData_iface;
1516         IPropData_AddRef(iface);
1517         TRACE("returning %p\n", *ppvObj);
1518         return S_OK;
1519     }
1520 
1521     TRACE("returning E_NOINTERFACE\n");
1522     return MAPI_E_INTERFACE_NOT_SUPPORTED;
1523 }
1524 
1525 /**************************************************************************
1526  *  IPropData_AddRef {MAPI32}
1527  *
1528  * Inherited method from the IUnknown Interface.
1529  * See IUnknown_AddRef.
1530  */
1531 static ULONG WINAPI IPropData_fnAddRef(LPPROPDATA iface)
1532 {
1533     IPropDataImpl *This = impl_from_IPropData(iface);
1534 
1535     TRACE("(%p)->(count before=%u)\n", This, This->lRef);
1536 
1537     return InterlockedIncrement(&This->lRef);
1538 }
1539 
1540 /**************************************************************************
1541  *  IPropData_Release {MAPI32}
1542  *
1543  * Inherited method from the IUnknown Interface.
1544  * See IUnknown_Release.
1545  */
1546 static ULONG WINAPI IPropData_fnRelease(LPPROPDATA iface)
1547 {
1548     IPropDataImpl *This = impl_from_IPropData(iface);
1549     LONG lRef;
1550 
1551     TRACE("(%p)->(count before=%u)\n", This, This->lRef);
1552 
1553     lRef = InterlockedDecrement(&This->lRef);
1554     if (!lRef)
1555     {
1556         TRACE("Destroying IPropData (%p)\n",This);
1557 
1558         /* Note: No need to lock, since no other thread is referencing iface */
1559         while (!list_empty(&This->values))
1560         {
1561             struct list *head = list_head(&This->values);
1562             LPIPropDataItem current = LIST_ENTRY(head, IPropDataItem, entry);
1563             list_remove(head);
1564             This->lpFree(current->value);
1565             This->lpFree(current);
1566         }
1567         This->cs.DebugInfo->Spare[0] = 0;
1568         DeleteCriticalSection(&This->cs);
1569         This->lpFree(This);
1570     }
1571     return (ULONG)lRef;
1572 }
1573 
1574 /**************************************************************************
1575  *  IPropData_GetLastError {MAPI32}
1576  *
1577  * Get information about the last error that occurred in an IMAPIProp object.
1578  *
1579  * PARAMS
1580  *  iface    [I] IMAPIProp object that experienced the error
1581  *  hRes     [I] Result of the call that returned an error
1582  *  ulFlags  [I] 0=return Ascii strings, MAPI_UNICODE=return Unicode strings
1583  *  lppError [O] Destination for detailed error information
1584  *
1585  * RETURNS
1586  *  Success: S_OK. *lppError contains details about the last error.
1587  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid,
1588  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails.
1589  *
1590  * NOTES
1591  *  - If this function succeeds, the returned information in *lppError must be
1592  *  freed using MAPIFreeBuffer() once the caller is finished with it.
1593  *  - It is possible for this function to succeed and set *lppError to NULL,
1594  *  if there is no further information to report about hRes.
1595  */
1596 static HRESULT WINAPI IPropData_fnGetLastError(LPPROPDATA iface, HRESULT hRes, ULONG ulFlags,
1597                                                LPMAPIERROR *lppError)
1598 {
1599     TRACE("(%p,0x%08X,0x%08X,%p)\n", iface, hRes, ulFlags, lppError);
1600 
1601     if (!lppError  || SUCCEEDED(hRes) || (ulFlags & ~MAPI_UNICODE))
1602         return MAPI_E_INVALID_PARAMETER;
1603 
1604     *lppError = NULL;
1605     return S_OK;
1606 }
1607 
1608 /**************************************************************************
1609  *  IPropData_SaveChanges {MAPI32}
1610  *
1611  * Update any changes made to a transactional IMAPIProp object.
1612  *
1613  * PARAMS
1614  *  iface    [I] IMAPIProp object to update
1615  *  ulFlags  [I] Flags controlling the update.
1616  *
1617  * RETURNS
1618  *  Success: S_OK. Any outstanding changes are committed to the object.
1619  *  Failure: An HRESULT error code describing the error.
1620  */
1621 static HRESULT WINAPI IPropData_fnSaveChanges(LPPROPDATA iface, ULONG ulFlags)
1622 {
1623     TRACE("(%p,0x%08X)\n", iface, ulFlags);
1624 
1625      /* Since this object is not transacted we do not need to implement this */
1626      /* FIXME: Should we set the access levels to clean? */
1627     return S_OK;
1628 }
1629 
1630 /**************************************************************************
1631  *  IPropData_GetProps {MAPI32}
1632  *
1633  * Get property values from an IMAPIProp object.
1634  *
1635  * PARAMS
1636  *  iface    [I] IMAPIProp object to get the property values from
1637  *  lpTags   [I] Property tags of property values to be retrieved
1638  *  ulFlags  [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1639  *                 unspecified types
1640  *  lpCount  [O] Destination for number of properties returned
1641  *  lppProps [O] Destination for returned property values
1642  *
1643  * RETURNS
1644  *  Success: S_OK. *lppProps and *lpCount are updated.
1645  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1646  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
1647  *           MAPI_W_ERRORS_RETURNED if not all properties were retrieved
1648  *           successfully.
1649  * NOTES
1650  *  - If MAPI_W_ERRORS_RETURNED is returned, any properties that could not be
1651  *    retrieved from iface are present in lppProps with their type
1652  *    changed to PT_ERROR and Id unchanged.
1653  */
1654 static HRESULT WINAPI IPropData_fnGetProps(LPPROPDATA iface, LPSPropTagArray lpTags, ULONG ulFlags,
1655                                            ULONG *lpCount, LPSPropValue *lppProps)
1656 {
1657     IPropDataImpl *This = impl_from_IPropData(iface);
1658     ULONG i;
1659     HRESULT hRet = S_OK;
1660 
1661     TRACE("(%p,%p,0x%08x,%p,%p) stub\n", iface, lpTags, ulFlags,
1662           lpCount, lppProps);
1663 
1664     if (!iface || ulFlags & ~MAPI_UNICODE || !lpTags || *lpCount || !lppProps)
1665         return MAPI_E_INVALID_PARAMETER;
1666 
1667     FIXME("semi-stub, flags not supported\n");
1668 
1669     *lpCount = lpTags->cValues;
1670     *lppProps = NULL;
1671 
1672     if (*lpCount)
1673     {
1674         hRet = MAPIAllocateBuffer(*lpCount * sizeof(SPropValue), (LPVOID*)lppProps);
1675         if (FAILED(hRet))
1676             return hRet;
1677 
1678         IMAPIPROP_Lock(This);
1679 
1680         for (i = 0; i < lpTags->cValues; i++)
1681         {
1682             HRESULT hRetTmp = E_INVALIDARG;
1683             LPIPropDataItem item;
1684 
1685             item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
1686 
1687             if (item)
1688                 hRetTmp = PropCopyMore(&(*lppProps)[i], item->value,
1689                                        This->lpMore, *lppProps);
1690             if (FAILED(hRetTmp))
1691             {
1692                 hRet = MAPI_W_ERRORS_RETURNED;
1693                 (*lppProps)[i].ulPropTag =
1694                     CHANGE_PROP_TYPE(lpTags->aulPropTag[i], PT_ERROR);
1695             }
1696         }
1697 
1698         IMAPIPROP_Unlock(This);
1699     }
1700     return hRet;
1701 }
1702 
1703 /**************************************************************************
1704  *  MAPIProp_GetPropList {MAPI32}
1705  *
1706  * Get the list of property tags for all values in an IMAPIProp object.
1707  *
1708  * PARAMS
1709  *  iface   [I] IMAPIProp object to get the property tag list from
1710  *  ulFlags [I] Return 0=Ascii MAPI_UNICODE=Unicode strings for
1711  *              unspecified types
1712  *  lppTags [O] Destination for the retrieved property tag list
1713  *
1714  * RETURNS
1715  *  Success: S_OK. *lppTags contains the tags for all available properties.
1716  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1717  *           MAPI_E_BAD_CHARWIDTH, if Ascii or Unicode strings are requested
1718  *           and that type of string is not supported.
1719  */
1720 static HRESULT WINAPI IPropData_fnGetPropList(LPPROPDATA iface, ULONG ulFlags,
1721                                               LPSPropTagArray *lppTags)
1722 {
1723     IPropDataImpl *This = impl_from_IPropData(iface);
1724     ULONG i;
1725     HRESULT hRet;
1726 
1727     TRACE("(%p,0x%08x,%p) stub\n", iface, ulFlags, lppTags);
1728 
1729     if (!iface || ulFlags & ~MAPI_UNICODE || !lppTags)
1730         return MAPI_E_INVALID_PARAMETER;
1731 
1732     FIXME("semi-stub, flags not supported\n");
1733 
1734     *lppTags = NULL;
1735 
1736     IMAPIPROP_Lock(This);
1737 
1738     hRet = MAPIAllocateBuffer(CbNewSPropTagArray(This->ulNumValues),
1739                               (LPVOID*)lppTags);
1740     if (SUCCEEDED(hRet))
1741     {
1742         struct list *cursor;
1743 
1744         i = 0;
1745         LIST_FOR_EACH(cursor, &This->values)
1746         {
1747             LPIPropDataItem current = LIST_ENTRY(cursor, IPropDataItem, entry);
1748             (*lppTags)->aulPropTag[i] = current->value->ulPropTag;
1749             i++;
1750         }
1751         (*lppTags)->cValues = This->ulNumValues;
1752     }
1753 
1754     IMAPIPROP_Unlock(This);
1755     return hRet;
1756 }
1757 
1758 /**************************************************************************
1759  *  IPropData_OpenProperty {MAPI32}
1760  *
1761  * Not documented at this time.
1762  *
1763  * RETURNS
1764  *  An HRESULT success/failure code.
1765  */
1766 static HRESULT WINAPI IPropData_fnOpenProperty(LPPROPDATA iface, ULONG ulPropTag, LPCIID iid,
1767                                                ULONG ulOpts, ULONG ulFlags, LPUNKNOWN *lpUnk)
1768 {
1769     FIXME("(%p,%u,%s,%u,0x%08x,%p) stub\n", iface, ulPropTag,
1770           debugstr_guid(iid), ulOpts, ulFlags, lpUnk);
1771     return MAPI_E_NO_SUPPORT;
1772 }
1773 
1774 
1775 /**************************************************************************
1776  *  IPropData_SetProps {MAPI32}
1777  *
1778  * Add or edit the property values in an IMAPIProp object.
1779  *
1780  * PARAMS
1781  *  iface    [I] IMAPIProp object to get the property tag list from
1782  *  ulValues [I] Number of properties in lpProps
1783  *  lpProps  [I] Property values to set
1784  *  lppProbs [O] Optional destination for any problems that occurred
1785  *
1786  * RETURNS
1787  *  Success: S_OK. The properties in lpProps are added to iface if they don't
1788  *           exist, or changed to the values in lpProps if they do
1789  *  Failure: An HRESULT error code describing the error
1790  */
1791 static HRESULT WINAPI IPropData_fnSetProps(LPPROPDATA iface, ULONG ulValues, LPSPropValue lpProps,
1792                                            LPSPropProblemArray *lppProbs)
1793 {
1794     IPropDataImpl *This = impl_from_IPropData(iface);
1795     HRESULT hRet = S_OK;
1796     ULONG i;
1797 
1798     TRACE("(%p,%u,%p,%p)\n", iface, ulValues, lpProps, lppProbs);
1799 
1800     if (!iface || !lpProps)
1801       return MAPI_E_INVALID_PARAMETER;
1802 
1803     for (i = 0; i < ulValues; i++)
1804     {
1805         if (FBadProp(&lpProps[i]) ||
1806             PROP_TYPE(lpProps[i].ulPropTag) == PT_OBJECT ||
1807             PROP_TYPE(lpProps[i].ulPropTag) == PT_NULL)
1808           return MAPI_E_INVALID_PARAMETER;
1809     }
1810 
1811     IMAPIPROP_Lock(This);
1812 
1813     /* FIXME: Under what circumstances is lpProbs created? */
1814     for (i = 0; i < ulValues; i++)
1815     {
1816         LPIPropDataItem item = IMAPIPROP_GetValue(This, lpProps[i].ulPropTag);
1817 
1818         if (item)
1819         {
1820             HRESULT hRetTmp;
1821             LPVOID lpMem = NULL;
1822 
1823             /* Found, so update the existing value */
1824             if (item->value->ulPropTag != lpProps[i].ulPropTag)
1825                 FIXME("semi-stub, overwriting type (not coercing)\n");
1826 
1827             hRetTmp = This->lpAlloc(sizeof(SPropValue), &lpMem);
1828             if (SUCCEEDED(hRetTmp))
1829             {
1830                 hRetTmp = PropCopyMore(lpMem, &lpProps[i], This->lpMore, lpMem);
1831                 if (SUCCEEDED(hRetTmp))
1832                 {
1833                     This->lpFree(item->value);
1834                     item->value = lpMem;
1835                     continue;
1836                 }
1837                 This->lpFree(lpMem);
1838             }
1839             hRet = hRetTmp;
1840         }
1841         else
1842         {
1843             /* Add new value */
1844             if (!IMAPIPROP_AddValue(This, &lpProps[i]))
1845                 hRet = MAPI_E_NOT_ENOUGH_MEMORY;
1846         }
1847     }
1848 
1849     IMAPIPROP_Unlock(This);
1850     return hRet;
1851 }
1852 
1853 /**************************************************************************
1854  *  IPropData_DeleteProps {MAPI32}
1855  *
1856  * Delete one or more property values from an IMAPIProp object.
1857  *
1858  * PARAMS
1859  *  iface    [I] IMAPIProp object to remove property values from.
1860  *  lpTags   [I] Collection of property Id's to remove from iface.
1861  *  lppProbs [O] Destination for problems encountered, if any.
1862  *
1863  * RETURNS
1864  *  Success: S_OK. Any properties in iface matching property Id's in lpTags have
1865  *           been deleted. If lppProbs is non-NULL it contains details of any
1866  *           errors that occurred.
1867  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
1868  *           E_ACCESSDENIED, if this object was created using CreateIProp() and
1869  *           a subsequent call to IPropData_SetObjAccess() was made specifying
1870  *           IPROP_READONLY as the access type.
1871  *
1872  * NOTES
1873  *  - lppProbs will not be populated for cases where a property Id is present
1874  *    in lpTags but not in iface.
1875  *  - lppProbs should be deleted with MAPIFreeBuffer() if returned.
1876  */
1877 static HRESULT WINAPI IPropData_fnDeleteProps(LPPROPDATA iface, LPSPropTagArray lpTags,
1878                                               LPSPropProblemArray *lppProbs)
1879 {
1880     IPropDataImpl *This = impl_from_IPropData(iface);
1881     ULONG i, numProbs = 0;
1882     HRESULT hRet = S_OK;
1883 
1884     TRACE("(%p,%p,%p)\n", iface, lpTags, lppProbs);
1885 
1886     if (!iface || !lpTags)
1887         return MAPI_E_INVALID_PARAMETER;
1888 
1889     if (lppProbs)
1890         *lppProbs = NULL;
1891 
1892     for (i = 0; i < lpTags->cValues; i++)
1893     {
1894         if (FBadPropTag(lpTags->aulPropTag[i]) ||
1895             PROP_TYPE(lpTags->aulPropTag[i]) == PT_OBJECT ||
1896             PROP_TYPE(lpTags->aulPropTag[i]) == PT_NULL)
1897           return MAPI_E_INVALID_PARAMETER;
1898     }
1899 
1900     IMAPIPROP_Lock(This);
1901 
1902     if (This->ulObjAccess != IPROP_READWRITE)
1903     {
1904         IMAPIPROP_Unlock(This);
1905         return E_ACCESSDENIED;
1906     }
1907 
1908     for (i = 0; i < lpTags->cValues; i++)
1909     {
1910         LPIPropDataItem item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
1911 
1912         if (item)
1913         {
1914             if (item->ulAccess & IPROP_READWRITE)
1915             {
1916                 /* Everything hunky-dory, remove the item */
1917                 list_remove(&item->entry);
1918                 This->lpFree(item->value); /* Also frees value pointers */
1919                 This->lpFree(item);
1920                 This->ulNumValues--;
1921             }
1922             else if (lppProbs)
1923             {
1924                  /* Can't write the value. Create/populate problems array */
1925                  if (!*lppProbs)
1926                  {
1927                      /* Create problems array */
1928                      ULONG ulSize = CbNewSPropProblemArray(lpTags->cValues - i);
1929                      HRESULT hRetTmp = MAPIAllocateBuffer(ulSize, (LPVOID*)lppProbs);
1930                      if (FAILED(hRetTmp))
1931                          hRet = hRetTmp;
1932                  }
1933                  if (*lppProbs)
1934                  {
1935                      LPSPropProblem lpProb = &(*lppProbs)->aProblem[numProbs];
1936                      lpProb->ulIndex = i;
1937                      lpProb->ulPropTag = lpTags->aulPropTag[i];
1938                      lpProb->scode = E_ACCESSDENIED;
1939                      numProbs++;
1940                  }
1941             }
1942         }
1943     }
1944     if (lppProbs && *lppProbs)
1945         (*lppProbs)->cProblem = numProbs;
1946 
1947     IMAPIPROP_Unlock(This);
1948     return hRet;
1949 }
1950 
1951 
1952 /**************************************************************************
1953  *  IPropData_CopyTo {MAPI32}
1954  *
1955  * Not documented at this time.
1956  *
1957  * RETURNS
1958  *  An HRESULT success/failure code.
1959  */
1960 static HRESULT WINAPI IPropData_fnCopyTo(LPPROPDATA iface, ULONG niids, LPCIID lpiidExcl,
1961                                          LPSPropTagArray lpPropsExcl, ULONG ulParam,
1962                                          LPMAPIPROGRESS lpIProgress, LPCIID lpIfaceIid,
1963                                          LPVOID lpDstObj, ULONG ulFlags,
1964                                          LPSPropProblemArray *lppProbs)
1965 {
1966     FIXME("(%p,%u,%p,%p,%x,%p,%s,%p,0x%08X,%p) stub\n", iface, niids,
1967           lpiidExcl, lpPropsExcl, ulParam, lpIProgress,
1968           debugstr_guid(lpIfaceIid), lpDstObj, ulFlags, lppProbs);
1969     return MAPI_E_NO_SUPPORT;
1970 }
1971 
1972 /**************************************************************************
1973  *  IPropData_CopyProps {MAPI32}
1974  *
1975  * Not documented at this time.
1976  *
1977  * RETURNS
1978  *  An HRESULT success/failure code.
1979  */
1980 static HRESULT WINAPI IPropData_fnCopyProps(LPPROPDATA iface, LPSPropTagArray lpInclProps,
1981                                             ULONG ulParam, LPMAPIPROGRESS lpIProgress,
1982                                             LPCIID lpIface, LPVOID lpDstObj, ULONG ulFlags,
1983                                             LPSPropProblemArray *lppProbs)
1984 {
1985     FIXME("(%p,%p,%x,%p,%s,%p,0x%08X,%p) stub\n", iface, lpInclProps,
1986           ulParam, lpIProgress, debugstr_guid(lpIface), lpDstObj, ulFlags,
1987           lppProbs);
1988     return MAPI_E_NO_SUPPORT;
1989 }
1990 
1991 /**************************************************************************
1992  *  IPropData_GetNamesFromIDs {MAPI32}
1993  *
1994  * Get the names of properties from their identifiers.
1995  *
1996  * PARAMS
1997  *  iface       [I]   IMAPIProp object to operate on
1998  *  lppPropTags [I/O] Property identifiers to get the names for, or NULL to
1999  *                    get all names
2000  *  iid         [I]   Property set identifier, or NULL
2001  *  ulFlags     [I]   MAPI_NO_IDS=Don't return numeric named properties,
2002  *                    or MAPI_NO_STRINGS=Don't return strings
2003  *  lpCount     [O]   Destination for number of properties returned
2004  *  lpppNames   [O]   Destination for returned names
2005  *
2006  * RETURNS
2007  *  Success: S_OK. *lppPropTags and lpppNames contain the returned
2008  *           name/identifiers.
2009  *  Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2010  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2011  *           MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2012  *           successfully.
2013  */
2014 static HRESULT WINAPI IPropData_fnGetNamesFromIDs(LPPROPDATA iface, LPSPropTagArray *lppPropTags,
2015                                                   LPGUID iid, ULONG ulFlags, ULONG *lpCount,
2016                                                   LPMAPINAMEID **lpppNames)
2017 {
2018     FIXME("(%p,%p,%s,0x%08X,%p,%p) stub\n", iface, lppPropTags,
2019           debugstr_guid(iid), ulFlags, lpCount, lpppNames);
2020     return MAPI_E_NO_SUPPORT;
2021 }
2022 
2023 /**************************************************************************
2024  *  IPropData_GetIDsFromNames {MAPI32}
2025  *
2026  * Get property identifiers associated with one or more named properties.
2027  *
2028  * PARAMS
2029  *  iface       [I] IMAPIProp object to operate on
2030  *  ulNames     [I] Number of names in lppNames
2031  *  lppNames    [I] Names to query or create, or NULL to query all names
2032  *  ulFlags     [I] Pass MAPI_CREATE to create new named properties
2033  *  lppPropTags [O] Destination for queried or created property identifiers
2034  *
2035  * RETURNS
2036  *  Success: S_OK. *lppPropTags contains the property tags created or requested.
2037  *  Failure: MAPI_E_NO_SUPPORT, if the object does not support named properties,
2038  *           MAPI_E_TOO_BIG, if the object cannot process the number of
2039  *           properties involved.
2040  *           MAPI_E_NOT_ENOUGH_MEMORY, if memory allocation fails, or
2041  *           MAPI_W_ERRORS_RETURNED if not all properties were retrieved
2042  *           successfully.
2043  */
2044 static HRESULT WINAPI IPropData_fnGetIDsFromNames(LPPROPDATA iface, ULONG ulNames,
2045                                                   LPMAPINAMEID *lppNames, ULONG ulFlags,
2046                                                   LPSPropTagArray *lppPropTags)
2047 {
2048     FIXME("(%p,%d,%p,0x%08X,%p) stub\n",
2049           iface, ulNames, lppNames, ulFlags, lppPropTags);
2050     return MAPI_E_NO_SUPPORT;
2051 }
2052 
2053 /**************************************************************************
2054  *  IPropData_HrSetObjAccess {MAPI32}
2055  *
2056  * Set the access level of an IPropData object.
2057  *
2058  * PARAMS
2059  *  iface    [I] IPropData object to set the access on
2060  *  ulAccess [I] Either IPROP_READONLY or IPROP_READWRITE for read or
2061  *               read/write access respectively.
2062  *
2063  * RETURNS
2064  *  Success: S_OK. The objects access level is changed.
2065  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2066  */
2067 static HRESULT WINAPI
2068 IPropData_fnHrSetObjAccess(LPPROPDATA iface, ULONG ulAccess)
2069 {
2070     IPropDataImpl *This = impl_from_IPropData(iface);
2071 
2072     TRACE("(%p,%x)\n", iface, ulAccess);
2073 
2074     if (!iface || ulAccess < IPROP_READONLY || ulAccess > IPROP_READWRITE)
2075         return MAPI_E_INVALID_PARAMETER;
2076 
2077     IMAPIPROP_Lock(This);
2078 
2079     This->ulObjAccess = ulAccess;
2080 
2081     IMAPIPROP_Unlock(This);
2082     return S_OK;
2083 }
2084 
2085 /* Internal - determine if an access value is bad */
2086 static inline BOOL PROP_IsBadAccess(ULONG ulAccess)
2087 {
2088     switch (ulAccess)
2089     {
2090     case IPROP_READONLY|IPROP_CLEAN:
2091     case IPROP_READONLY|IPROP_DIRTY:
2092     case IPROP_READWRITE|IPROP_CLEAN:
2093     case IPROP_READWRITE|IPROP_DIRTY:
2094         return FALSE;
2095     }
2096     return TRUE;
2097 }
2098 
2099 /**************************************************************************
2100  *  IPropData_HrSetPropAccess {MAPI32}
2101  *
2102  * Set the access levels for a group of property values in an IPropData object.
2103  *
2104  * PARAMS
2105  *  iface    [I] IPropData object to set access levels in.
2106  *  lpTags   [I] List of property Id's to set access for.
2107  *  lpAccess [O] Access level for each property in lpTags.
2108  *
2109  * RETURNS
2110  *  Success: S_OK. The access level of each property value in lpTags that is
2111  *           present in iface is changed.
2112  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid.
2113  *
2114  * NOTES
2115  *  - Each access level in lpAccess must contain at least one of IPROP_READONLY
2116  *    or IPROP_READWRITE, but not both, and also IPROP_CLEAN or IPROP_DIRTY,
2117  *    but not both. No other bits should be set.
2118  *  - If a property Id in lpTags is not present in iface, it is ignored.
2119  */
2120 static HRESULT WINAPI
2121 IPropData_fnHrSetPropAccess(LPPROPDATA iface, LPSPropTagArray lpTags,
2122                             ULONG *lpAccess)
2123 {
2124     IPropDataImpl *This = impl_from_IPropData(iface);
2125     ULONG i;
2126 
2127     TRACE("(%p,%p,%p)\n", iface, lpTags, lpAccess);
2128 
2129     if (!iface || !lpTags || !lpAccess)
2130         return MAPI_E_INVALID_PARAMETER;
2131 
2132     for (i = 0; i < lpTags->cValues; i++)
2133     {
2134         if (FBadPropTag(lpTags->aulPropTag[i]) || PROP_IsBadAccess(lpAccess[i]))
2135             return MAPI_E_INVALID_PARAMETER;
2136     }
2137 
2138     IMAPIPROP_Lock(This);
2139 
2140     for (i = 0; i < lpTags->cValues; i++)
2141     {
2142         LPIPropDataItem item = IMAPIPROP_GetValue(This, lpTags->aulPropTag[i]);
2143 
2144         if (item)
2145             item->ulAccess = lpAccess[i];
2146     }
2147 
2148     IMAPIPROP_Unlock(This);
2149     return S_OK;
2150 }
2151 
2152 /**************************************************************************
2153  *  IPropData_HrGetPropAccess {MAPI32}
2154  *
2155  * Get the access levels for a group of property values in an IPropData object.
2156  *
2157  * PARAMS
2158  *  iface     [I] IPropData object to get access levels from.
2159  *  lppTags   [O] Destination for the list of property Id's in iface.
2160  *  lppAccess [O] Destination for access level for each property in lppTags.
2161  *
2162  * RETURNS
2163  *  Success: S_OK. lppTags and lppAccess contain the property Id's and the
2164  *           Access level of each property value in iface.
2165  *  Failure: MAPI_E_INVALID_PARAMETER, if any parameter is invalid, or
2166  *           MAPI_E_NOT_ENOUGH_MEMORY if memory allocation fails.
2167  *
2168  * NOTES
2169  *  - *lppTags and *lppAccess should be freed with MAPIFreeBuffer() by the caller.
2170  */
2171 static HRESULT WINAPI
2172 IPropData_fnHrGetPropAccess(LPPROPDATA iface, LPSPropTagArray *lppTags,
2173                             ULONG **lppAccess)
2174 {
2175     IPropDataImpl *This = impl_from_IPropData(iface);
2176     LPVOID lpMem;
2177     HRESULT hRet;
2178     ULONG i;
2179 
2180     TRACE("(%p,%p,%p) stub\n", iface, lppTags, lppAccess);
2181 
2182     if (!iface || !lppTags || !lppAccess)
2183         return MAPI_E_INVALID_PARAMETER;
2184 
2185     *lppTags = NULL;
2186     *lppAccess = NULL;
2187 
2188     IMAPIPROP_Lock(This);
2189 
2190     hRet = This->lpAlloc(CbNewSPropTagArray(This->ulNumValues), &lpMem);
2191     if (SUCCEEDED(hRet))
2192     {
2193         *lppTags = lpMem;
2194 
2195         hRet = This->lpAlloc(This->ulNumValues * sizeof(ULONG), &lpMem);
2196         if (SUCCEEDED(hRet))
2197         {
2198             struct list *cursor;
2199 
2200             *lppAccess = lpMem;
2201             (*lppTags)->cValues = This->ulNumValues;
2202 
2203             i = 0;
2204             LIST_FOR_EACH(cursor, &This->values)
2205             {
2206                 LPIPropDataItem item = LIST_ENTRY(cursor, IPropDataItem, entry);
2207                 (*lppTags)->aulPropTag[i] = item->value->ulPropTag;
2208                 (*lppAccess)[i] = item->ulAccess;
2209                 i++;
2210             }
2211             IMAPIPROP_Unlock(This);
2212             return S_OK;
2213         }
2214         This->lpFree(*lppTags);
2215         *lppTags = 0;
2216     }
2217     IMAPIPROP_Unlock(This);
2218     return MAPI_E_NOT_ENOUGH_MEMORY;
2219 }
2220 
2221 /**************************************************************************
2222  *  IPropData_HrAddObjProps {MAPI32}
2223  *
2224  * Not documented at this time.
2225  *
2226  * RETURNS
2227  *  An HRESULT success/failure code.
2228  */
2229 static HRESULT WINAPI
2230 IPropData_fnHrAddObjProps(LPPROPDATA iface, LPSPropTagArray lpTags,
2231                           LPSPropProblemArray *lppProbs)
2232 {
2233 #if 0
2234     ULONG i;
2235     HRESULT hRet;
2236     LPSPropValue lpValues;
2237 #endif
2238 
2239     FIXME("(%p,%p,%p) stub\n", iface, lpTags, lppProbs);
2240 
2241     if (!iface || !lpTags)
2242         return MAPI_E_INVALID_PARAMETER;
2243 
2244     /* FIXME: Below is the obvious implementation, adding all the properties
2245      *        in lpTags to the object. However, it doesn't appear that this
2246      *        is what this function does.
2247      */
2248     return S_OK;
2249 #if 0
2250     if (!lpTags->cValues)
2251         return S_OK;
2252 
2253     lpValues = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
2254                          lpTags->cValues * sizeof(SPropValue));
2255     if (!lpValues)
2256         return MAPI_E_NOT_ENOUGH_MEMORY;
2257 
2258     for (i = 0; i < lpTags->cValues; i++)
2259         lpValues[i].ulPropTag = lpTags->aulPropTag[i];
2260 
2261     hRet = IPropData_SetProps(iface, lpTags->cValues, lpValues, lppProbs);
2262     HeapFree(GetProcessHeap(), 0, lpValues);
2263     return hRet;
2264 #endif
2265 }
2266 
2267 static const IPropDataVtbl IPropDataImpl_vtbl =
2268 {
2269     IPropData_fnQueryInterface,
2270     IPropData_fnAddRef,
2271     IPropData_fnRelease,
2272     IPropData_fnGetLastError,
2273     IPropData_fnSaveChanges,
2274     IPropData_fnGetProps,
2275     IPropData_fnGetPropList,
2276     IPropData_fnOpenProperty,
2277     IPropData_fnSetProps,
2278     IPropData_fnDeleteProps,
2279     IPropData_fnCopyTo,
2280     IPropData_fnCopyProps,
2281     IPropData_fnGetNamesFromIDs,
2282     IPropData_fnGetIDsFromNames,
2283     IPropData_fnHrSetObjAccess,
2284     IPropData_fnHrSetPropAccess,
2285     IPropData_fnHrGetPropAccess,
2286     IPropData_fnHrAddObjProps
2287 };
2288 
2289 /*************************************************************************
2290  * CreateIProp@24 (MAPI32.60)
2291  *
2292  * Create an IPropData object.
2293  *
2294  * PARAMS
2295  *  iid         [I] GUID of the object to create. Use &IID_IMAPIPropData or NULL
2296  *  lpAlloc     [I] Memory allocation function. Use MAPIAllocateBuffer()
2297  *  lpMore      [I] Linked memory allocation function. Use MAPIAllocateMore()
2298  *  lpFree      [I] Memory free function. Use MAPIFreeBuffer()
2299  *  lpReserved  [I] Reserved, set to NULL
2300  *  lppPropData [O] Destination for created IPropData object
2301  *
2302  * RETURNS
2303  *  Success: S_OK. *lppPropData contains the newly created object.
2304  *  Failure: MAPI_E_INTERFACE_NOT_SUPPORTED, if iid is non-NULL and not supported,
2305  *           MAPI_E_INVALID_PARAMETER, if any parameter is invalid
2306  */
2307 SCODE WINAPI CreateIProp(LPCIID iid, ALLOCATEBUFFER *lpAlloc,
2308                          ALLOCATEMORE *lpMore, FREEBUFFER *lpFree,
2309                          LPVOID lpReserved, LPPROPDATA *lppPropData)
2310 {
2311     IPropDataImpl *lpPropData;
2312     SCODE scode;
2313 
2314     TRACE("(%s,%p,%p,%p,%p,%p)\n", debugstr_guid(iid), lpAlloc, lpMore, lpFree,
2315           lpReserved, lppPropData);
2316 
2317     if (lppPropData)
2318         *lppPropData = NULL;
2319 
2320     if (iid && !IsEqualGUID(iid, &IID_IMAPIPropData))
2321         return MAPI_E_INTERFACE_NOT_SUPPORTED;
2322 
2323     if (!lpAlloc || !lpMore || !lpFree || lpReserved || !lppPropData)
2324         return MAPI_E_INVALID_PARAMETER;
2325 
2326     scode = lpAlloc(sizeof(IPropDataImpl), (LPVOID*)&lpPropData);
2327 
2328     if (SUCCEEDED(scode))
2329     {
2330         lpPropData->IPropData_iface.lpVtbl = &IPropDataImpl_vtbl;
2331         lpPropData->lRef = 1;
2332         lpPropData->lpAlloc = lpAlloc;
2333         lpPropData->lpMore = lpMore;
2334         lpPropData->lpFree = lpFree;
2335         lpPropData->ulObjAccess = IPROP_READWRITE;
2336         lpPropData->ulNumValues = 0;
2337         list_init(&lpPropData->values);
2338         InitializeCriticalSection(&lpPropData->cs);
2339         lpPropData->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": IPropDataImpl.cs");
2340         *lppPropData = &lpPropData->IPropData_iface;
2341     }
2342     return scode;
2343 }
2344