xref: /reactos/dll/win32/shlwapi/clist.c (revision d3fd5bdd)
1 /*
2  * SHLWAPI DataBlock List functions
3  *
4  * Copyright 2002 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 #include <stdarg.h>
21 #include <string.h>
22 
23 #define COBJMACROS
24 
25 #include "windef.h"
26 #include "winbase.h"
27 #include "winuser.h"
28 #include "objbase.h"
29 #include "shlobj.h"
30 #include "wine/debug.h"
31 
32 WINE_DEFAULT_DEBUG_CHANNEL(shell);
33 
34 /* dwSignature for contained DATABLOCK_HEADER items */
35 #define CLIST_ID_CONTAINER (~0U)
36 
37 /*************************************************************************
38  * NextItem
39  *
40  * Internal helper: move a DataBlock pointer to the next item.
41  */
NextItem(LPDBLIST lpList)42 static inline LPDATABLOCK_HEADER NextItem(LPDBLIST lpList)
43 {
44   char* address = (char*)lpList;
45   address += lpList->cbSize;
46   return (LPDATABLOCK_HEADER)address;
47 }
48 
49 /*************************************************************************
50  *      @	[SHLWAPI.20]
51  *
52  * Insert a new item into a DataBlock list.
53  *
54  * PARAMS
55  *  lppList   [0] Pointer to the List
56  *  lpNewItem [I] The new item to add to the list
57  *
58  * RETURNS
59  *  Success: S_OK. The item is added to the list.
60  *  Failure: An HRESULT error code.
61  *
62  * NOTES
63  *  If the size of the element to be inserted is less than the size of a
64  *  DATABLOCK_HEADER node, or the Id for the item is CLIST_ID_CONTAINER,
65  *  the call returns S_OK but does not actually add the element.
66  *  See SHWriteDataBlockList.
67  */
SHAddDataBlock(LPDBLIST * lppList,const DATABLOCK_HEADER * lpNewItem)68 BOOL WINAPI SHAddDataBlock(LPDBLIST* lppList, const DATABLOCK_HEADER *lpNewItem)
69 {
70   LPDATABLOCK_HEADER lpInsertAt = NULL;
71   ULONG ulSize;
72 
73   TRACE("(%p,%p)\n", lppList, lpNewItem);
74 
75   if(!lppList || !lpNewItem)
76     return FALSE;
77 
78   if (lpNewItem->cbSize < sizeof(DATABLOCK_HEADER) ||
79       lpNewItem->dwSignature == CLIST_ID_CONTAINER)
80     return FALSE;
81 
82   ulSize = lpNewItem->cbSize;
83 
84   if(ulSize & 0x3)
85   {
86     /* Tune size to a ULONG boundary, add space for container element */
87     ulSize = ((ulSize + 0x3) & 0xFFFFFFFC) + sizeof(DATABLOCK_HEADER);
88     TRACE("Creating container item, new size = %d\n", ulSize);
89   }
90 
91   if(!*lppList)
92   {
93     /* An empty list. Allocate space for terminal ulSize also */
94     *lppList = LocalAlloc(LMEM_ZEROINIT, ulSize + sizeof(ULONG));
95     lpInsertAt = *lppList;
96   }
97   else
98   {
99     /* Append to the end of the list */
100     ULONG ulTotalSize = 0;
101     LPDATABLOCK_HEADER lpIter = *lppList;
102 
103     /* Iterate to the end of the list, calculating the total size */
104     while (lpIter->cbSize)
105     {
106       ulTotalSize += lpIter->cbSize;
107       lpIter = NextItem(lpIter);
108     }
109 
110     /* Increase the size of the list */
111     lpIter = LocalReAlloc(*lppList, ulTotalSize + ulSize+sizeof(ULONG),
112                                           LMEM_ZEROINIT | LMEM_MOVEABLE);
113     if(lpIter)
114     {
115       *lppList = lpIter;
116       lpInsertAt = (LPDATABLOCK_HEADER)((char*)lpIter + ulTotalSize); /* At end */
117     }
118   }
119 
120   if(lpInsertAt)
121   {
122     /* Copy in the new item */
123     LPDATABLOCK_HEADER lpDest = lpInsertAt;
124 
125     if(ulSize != lpNewItem->cbSize)
126     {
127       lpInsertAt->cbSize = ulSize;
128       lpInsertAt->dwSignature = CLIST_ID_CONTAINER;
129       lpDest++;
130     }
131     memcpy(lpDest, lpNewItem, lpNewItem->cbSize);
132 
133     /* Terminate the list */
134     lpInsertAt = NextItem(lpInsertAt);
135     lpInsertAt->cbSize = 0;
136 
137     return TRUE;
138   }
139   return FALSE;
140 }
141 
142 /*************************************************************************
143  *      @	[SHLWAPI.17]
144  *
145  * Write a DataBlock list to an IStream object.
146  *
147  * PARAMS
148  *  lpStream  [I] IStream object to write the list to
149  *  lpList    [I] List of items to write
150  *
151  * RETURNS
152  *  Success: S_OK. The object is written to the stream.
153  *  Failure: An HRESULT error code
154  *
155  * NOTES
156  *  Ordinals 17,18,19,20,21 and 22 are related and together provide a compact
157  *  list structure (a "DataBlock List"), which may be stored and retrieved from
158  *  an IStream object.
159  *
160  *  The exposed API consists of:
161  *
162  *  - SHWriteDataBlockList() - Write a DataBlock list to a stream,
163  *  - SHReadDataBlockList() - Read and create a list from a stream,
164  *  - SHFreeDataBlockList() - Free a list,
165  *  - SHAddDataBlock() - Insert a new item into a list,
166  *  - SHRemoveDataBlock() - Remove an item from a list,
167  *  - SHFindDataBlock() - Find an item in a list.
168  *
169  *  The DataBlock list is stored packed into a memory array. Each element has a
170  *  size and an associated ID. Elements must be less than 64k if the list is
171  *  to be subsequently read from a stream.
172  *
173  *  Elements are aligned on DWORD boundaries. If an elements data size is not
174  *  a DWORD size multiple, the element is wrapped by inserting a surrounding
175  *  element with an Id of 0xFFFFFFFF, and size sufficient to pad to a DWORD boundary.
176  *
177  *  These functions are slow for large objects and long lists.
178  */
SHWriteDataBlockList(IStream * lpStream,LPDBLIST lpList)179 HRESULT WINAPI SHWriteDataBlockList(IStream* lpStream, LPDBLIST lpList)
180 {
181   ULONG ulSize;
182   HRESULT hRet = S_OK;
183 
184   TRACE("(%p,%p)\n", lpStream, lpList);
185 
186   if(lpList)
187   {
188     while (lpList->cbSize)
189     {
190       LPDATABLOCK_HEADER lpItem = lpList;
191 
192       if(lpList->dwSignature == CLIST_ID_CONTAINER)
193         lpItem++;
194 
195       hRet = IStream_Write(lpStream,lpItem,lpItem->cbSize,&ulSize);
196       if (FAILED(hRet))
197         return hRet;
198 
199       if(lpItem->cbSize != ulSize)
200         return STG_E_MEDIUMFULL;
201 
202       lpList = NextItem(lpList);
203     }
204   }
205 
206   if(SUCCEEDED(hRet))
207   {
208     ULONG ulDummy;
209     ulSize = 0;
210 
211     /* Write a terminating list entry with zero size */
212     hRet = IStream_Write(lpStream, &ulSize,sizeof(ulSize),&ulDummy);
213   }
214 
215   return hRet;
216 }
217 
218 /*************************************************************************
219  *      @	[SHLWAPI.18]
220  *
221  * Read and create a DataBlock list from an IStream object.
222  *
223  * PARAMS
224  *  lpStream  [I] Stream to read the list from
225  *  lppList   [0] Pointer to receive the new List
226  *
227  * RETURNS
228  *  Success: S_OK
229  *  Failure: An HRESULT error code
230  *
231  * NOTES
232  *  When read from a file, list objects are limited in size to 64k.
233  *  See SHWriteDataBlockList.
234  */
SHReadDataBlockList(IStream * lpStream,LPDBLIST * lppList)235 HRESULT WINAPI SHReadDataBlockList(IStream* lpStream, LPDBLIST* lppList)
236 {
237   DATABLOCK_HEADER bBuff[128]; /* Temporary storage for new list item */
238   ULONG ulBuffSize = sizeof(bBuff);
239   LPDATABLOCK_HEADER pItem = bBuff;
240   ULONG ulRead, ulSize;
241   HRESULT hRet = S_OK;
242 
243   TRACE("(%p,%p)\n", lpStream, lppList);
244 
245   if(*lppList)
246   {
247     /* Free any existing list */
248     LocalFree(*lppList);
249     *lppList = NULL;
250   }
251 
252   do
253   {
254     /* Read the size of the next item */
255     hRet = IStream_Read(lpStream, &ulSize,sizeof(ulSize),&ulRead);
256 
257     if(FAILED(hRet) || ulRead != sizeof(ulSize) || !ulSize)
258       break; /* Read failed or read zero size (the end of the list) */
259 
260     if(ulSize > 0xFFFF)
261     {
262       LARGE_INTEGER liZero;
263       ULARGE_INTEGER ulPos;
264 
265       liZero.QuadPart = 0;
266 
267       /* Back the stream up; this object is too big for the list */
268       if(SUCCEEDED(IStream_Seek(lpStream, liZero, STREAM_SEEK_CUR, &ulPos)))
269       {
270         liZero.QuadPart = ulPos.QuadPart - sizeof(ULONG);
271         IStream_Seek(lpStream, liZero, STREAM_SEEK_SET, NULL);
272       }
273       break;
274     }
275     else if (ulSize >= sizeof(DATABLOCK_HEADER))
276     {
277       /* Add this new item to the list */
278       if(ulSize > ulBuffSize)
279       {
280         /* We need more buffer space, allocate it */
281         LPDATABLOCK_HEADER lpTemp;
282 
283         if (pItem == bBuff)
284           lpTemp = LocalAlloc(LMEM_ZEROINIT, ulSize);
285         else
286           lpTemp = LocalReAlloc(pItem, ulSize, LMEM_ZEROINIT|LMEM_MOVEABLE);
287 
288         if(!lpTemp)
289         {
290           hRet = E_OUTOFMEMORY;
291           break;
292         }
293         ulBuffSize = ulSize;
294         pItem = lpTemp;
295       }
296 
297       pItem->cbSize = ulSize;
298       ulSize -= sizeof(pItem->cbSize); /* already read this member */
299 
300       /* Read the item Id and data */
301       hRet = IStream_Read(lpStream, &pItem->dwSignature, ulSize, &ulRead);
302 
303       if(FAILED(hRet) || ulRead != ulSize)
304         break;
305 
306       SHAddDataBlock(lppList, pItem); /* Insert Item */
307     }
308   } while(1);
309 
310   /* If we allocated space, free it */
311   if(pItem != bBuff)
312     LocalFree(pItem);
313 
314   return hRet;
315 }
316 
317 /*************************************************************************
318  *      @	[SHLWAPI.19]
319  *
320  * Free a DataBlock list.
321  *
322  * PARAMS
323  *  lpList [I] List to free
324  *
325  * RETURNS
326  *  Nothing.
327  *
328  * NOTES
329  *  See SHWriteDataBlockList.
330  */
SHFreeDataBlockList(LPDBLIST lpList)331 VOID WINAPI SHFreeDataBlockList(LPDBLIST lpList)
332 {
333   TRACE("(%p)\n", lpList);
334 
335   if (lpList)
336     LocalFree(lpList);
337 }
338 
339 /*************************************************************************
340  *      @	[SHLWAPI.21]
341  *
342  * Remove an item from a DataBlock list.
343  *
344  * PARAMS
345  *  lppList     [O] List to remove the item from
346  *  dwSignature [I] Id of item to remove
347  *
348  * RETURNS
349  *  Success: TRUE.
350  *  Failure: FALSE, If any parameters are invalid, or the item was not found.
351  *
352  * NOTES
353  *  See SHWriteDataBlockList.
354  */
SHRemoveDataBlock(LPDBLIST * lppList,DWORD dwSignature)355 BOOL WINAPI SHRemoveDataBlock(LPDBLIST* lppList, DWORD dwSignature)
356 {
357   LPDATABLOCK_HEADER lpList = NULL;
358   LPDATABLOCK_HEADER lpItem = NULL;
359   LPDATABLOCK_HEADER lpNext;
360   ULONG ulNewSize;
361 
362   TRACE("(%p,%d)\n", lppList, dwSignature);
363 
364   if(lppList && (lpList = *lppList))
365   {
366     /* Search for item in list */
367     while (lpList->cbSize)
368     {
369       if(lpList->dwSignature == dwSignature ||
370         (lpList->dwSignature == CLIST_ID_CONTAINER && lpList[1].dwSignature == dwSignature))
371       {
372         lpItem = lpList; /* Found */
373         break;
374       }
375       lpList = NextItem(lpList);
376     }
377   }
378 
379   if(!lpItem)
380     return FALSE;
381 
382   lpList = lpNext = NextItem(lpItem);
383 
384   /* Locate the end of the list */
385   while (lpList->cbSize)
386     lpList = NextItem(lpList);
387 
388   /* Resize the list */
389   ulNewSize = LocalSize(*lppList) - lpItem->cbSize;
390 
391   /* Copy following elements over lpItem */
392   memmove(lpItem, lpNext, (char *)lpList - (char *)lpNext + sizeof(ULONG));
393 
394   if(ulNewSize <= sizeof(ULONG))
395   {
396     LocalFree(*lppList);
397     *lppList = NULL; /* Removed the last element */
398   }
399   else
400   {
401     lpList = LocalReAlloc(*lppList, ulNewSize, LMEM_ZEROINIT|LMEM_MOVEABLE);
402     if(lpList)
403       *lppList = lpList;
404   }
405   return TRUE;
406 }
407 
408 /*************************************************************************
409  *      @	[SHLWAPI.22]
410  *
411  * Find an item in a DataBlock list.
412  *
413  * PARAMS
414  *  lpList      [I] List to search
415  *  dwSignature [I] Id of item to find
416  *
417  * RETURNS
418  *  Success: A pointer to the list item found
419  *  Failure: NULL
420  *
421  * NOTES
422  *  See SHWriteDataBlockList.
423  */
SHFindDataBlock(LPDBLIST lpList,DWORD dwSignature)424 DATABLOCK_HEADER* WINAPI SHFindDataBlock(LPDBLIST lpList, DWORD dwSignature)
425 {
426   TRACE("(%p,%d)\n", lpList, dwSignature);
427 
428   if(lpList)
429   {
430     while(lpList->cbSize)
431     {
432       if(lpList->dwSignature == dwSignature)
433         return lpList; /* Matched */
434       else if(lpList->dwSignature == CLIST_ID_CONTAINER && lpList[1].dwSignature == dwSignature)
435         return lpList + 1; /* Contained item matches */
436 
437       lpList = NextItem(lpList);
438     }
439   }
440   return NULL;
441 }
442