Release 20050930.
[wine/gsoc-2012-control.git] / dlls / shlwapi / regstream.c
blob50b6c45d8ac760c01a843f7ec39dc655e966d00e
1 /*
2 * SHLWAPI Registry Stream functions
4 * Copyright 1999 Juergen Schmied
5 * Copyright 2002 Jon Griffiths
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include <stdarg.h>
23 #include <string.h>
25 #define COBJMACROS
27 #include "winerror.h"
28 #include "windef.h"
29 #include "winbase.h"
30 #include "objbase.h"
31 #include "winreg.h"
33 #include "wine/debug.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(shell);
37 typedef struct
39 const IStreamVtbl *lpVtbl;
40 LONG ref;
41 HKEY hKey;
42 LPBYTE pbBuffer;
43 DWORD dwLength;
44 DWORD dwPos;
45 } ISHRegStream;
47 /**************************************************************************
48 * IStream_fnQueryInterface
50 static HRESULT WINAPI IStream_fnQueryInterface(IStream *iface, REFIID riid, LPVOID *ppvObj)
52 ISHRegStream *This = (ISHRegStream *)iface;
54 TRACE("(%p)->(\n\tIID:\t%s,%p)\n",This,debugstr_guid(riid),ppvObj);
56 *ppvObj = NULL;
58 if(IsEqualIID(riid, &IID_IUnknown)) /*IUnknown*/
59 *ppvObj = This;
60 else if(IsEqualIID(riid, &IID_IStream)) /*IStream*/
61 *ppvObj = This;
63 if(*ppvObj)
65 IStream_AddRef((IStream*)*ppvObj);
66 TRACE("-- Interface: (%p)->(%p)\n",ppvObj,*ppvObj);
67 return S_OK;
69 TRACE("-- Interface: E_NOINTERFACE\n");
70 return E_NOINTERFACE;
73 /**************************************************************************
74 * IStream_fnAddRef
76 static ULONG WINAPI IStream_fnAddRef(IStream *iface)
78 ISHRegStream *This = (ISHRegStream *)iface;
79 ULONG refCount = InterlockedIncrement(&This->ref);
81 TRACE("(%p)->(ref before=%lu)\n",This, refCount - 1);
83 return refCount;
86 /**************************************************************************
87 * IStream_fnRelease
89 static ULONG WINAPI IStream_fnRelease(IStream *iface)
91 ISHRegStream *This = (ISHRegStream *)iface;
92 ULONG refCount = InterlockedDecrement(&This->ref);
94 TRACE("(%p)->(ref before=%lu)\n",This, refCount + 1);
96 if (!refCount)
98 TRACE(" destroying SHReg IStream (%p)\n",This);
100 HeapFree(GetProcessHeap(),0,This->pbBuffer);
102 if (This->hKey)
103 RegCloseKey(This->hKey);
105 HeapFree(GetProcessHeap(),0,This);
106 return 0;
109 return refCount;
112 /**************************************************************************
113 * IStream_fnRead
115 static HRESULT WINAPI IStream_fnRead (IStream * iface, void* pv, ULONG cb, ULONG* pcbRead)
117 ISHRegStream *This = (ISHRegStream *)iface;
119 DWORD dwBytesToRead, dwBytesLeft;
121 TRACE("(%p)->(%p,0x%08lx,%p)\n",This, pv, cb, pcbRead);
123 if (!pv)
124 return STG_E_INVALIDPOINTER;
126 dwBytesLeft = This->dwLength - This->dwPos;
128 if ( 0 >= dwBytesLeft ) /* end of buffer */
129 return S_FALSE;
131 dwBytesToRead = ( cb > dwBytesLeft) ? dwBytesLeft : cb;
133 memmove ( pv, (This->pbBuffer) + (This->dwPos), dwBytesToRead);
135 This->dwPos += dwBytesToRead; /* adjust pointer */
137 if (pcbRead)
138 *pcbRead = dwBytesToRead;
140 return S_OK;
143 /**************************************************************************
144 * IStream_fnWrite
146 static HRESULT WINAPI IStream_fnWrite (IStream * iface, const void* pv, ULONG cb, ULONG* pcbWritten)
148 ISHRegStream *This = (ISHRegStream *)iface;
150 TRACE("(%p)\n",This);
152 if (pcbWritten)
153 *pcbWritten = 0;
155 return E_NOTIMPL;
158 /**************************************************************************
159 * IStream_fnSeek
161 static HRESULT WINAPI IStream_fnSeek (IStream * iface, LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER* plibNewPosition)
163 ISHRegStream *This = (ISHRegStream *)iface;
165 TRACE("(%p)\n",This);
167 if (plibNewPosition)
168 plibNewPosition->QuadPart = 0;
169 return E_NOTIMPL;
172 /**************************************************************************
173 * IStream_fnSetSize
175 static HRESULT WINAPI IStream_fnSetSize (IStream * iface, ULARGE_INTEGER libNewSize)
177 ISHRegStream *This = (ISHRegStream *)iface;
179 TRACE("(%p)\n",This);
180 return E_NOTIMPL;
183 /**************************************************************************
184 * IStream_fnCopyTo
186 static HRESULT WINAPI IStream_fnCopyTo (IStream * iface, IStream* pstm, ULARGE_INTEGER cb, ULARGE_INTEGER* pcbRead, ULARGE_INTEGER* pcbWritten)
188 ISHRegStream *This = (ISHRegStream *)iface;
190 TRACE("(%p)\n",This);
191 if (pcbRead)
192 pcbRead->QuadPart = 0;
193 if (pcbWritten)
194 pcbWritten->QuadPart = 0;
195 return E_NOTIMPL;
198 /**************************************************************************
199 * IStream_fnCommit
201 static HRESULT WINAPI IStream_fnCommit (IStream * iface, DWORD grfCommitFlags)
203 ISHRegStream *This = (ISHRegStream *)iface;
205 TRACE("(%p)\n",This);
207 return E_NOTIMPL;
210 /**************************************************************************
211 * IStream_fnRevert
213 static HRESULT WINAPI IStream_fnRevert (IStream * iface)
215 ISHRegStream *This = (ISHRegStream *)iface;
217 TRACE("(%p)\n",This);
219 return E_NOTIMPL;
222 /**************************************************************************
223 * IStream_fnLockUnlockRegion
225 static HRESULT WINAPI IStream_fnLockUnlockRegion (IStream * iface, ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
227 ISHRegStream *This = (ISHRegStream *)iface;
229 TRACE("(%p)\n",This);
231 return E_NOTIMPL;
234 /*************************************************************************
235 * IStream_fnStat
237 static HRESULT WINAPI IStream_fnStat (IStream * iface, STATSTG* pstatstg, DWORD grfStatFlag)
239 ISHRegStream *This = (ISHRegStream *)iface;
241 TRACE("(%p)\n",This);
243 return E_NOTIMPL;
246 /*************************************************************************
247 * IStream_fnClone
249 static HRESULT WINAPI IStream_fnClone (IStream * iface, IStream** ppstm)
251 ISHRegStream *This = (ISHRegStream *)iface;
253 TRACE("(%p)\n",This);
254 if (ppstm)
255 *ppstm = NULL;
256 return E_NOTIMPL;
259 static const IStreamVtbl rstvt =
261 IStream_fnQueryInterface,
262 IStream_fnAddRef,
263 IStream_fnRelease,
264 IStream_fnRead,
265 IStream_fnWrite,
266 IStream_fnSeek,
267 IStream_fnSetSize,
268 IStream_fnCopyTo,
269 IStream_fnCommit,
270 IStream_fnRevert,
271 IStream_fnLockUnlockRegion,
272 IStream_fnLockUnlockRegion,
273 IStream_fnStat,
274 IStream_fnClone
277 /* Methods overridden by the dummy stream */
279 /**************************************************************************
280 * IStream_fnAddRefDummy
282 static ULONG WINAPI IStream_fnAddRefDummy(IStream *iface)
284 ISHRegStream *This = (ISHRegStream *)iface;
285 TRACE("(%p)\n", This);
286 return 2;
289 /**************************************************************************
290 * IStream_fnReleaseDummy
292 static ULONG WINAPI IStream_fnReleaseDummy(IStream *iface)
294 ISHRegStream *This = (ISHRegStream *)iface;
295 TRACE("(%p)\n", This);
296 return 1;
299 /**************************************************************************
300 * IStream_fnReadDummy
302 static HRESULT WINAPI IStream_fnReadDummy(IStream *iface, LPVOID pv, ULONG cb, ULONG* pcbRead)
304 if (pcbRead)
305 *pcbRead = 0;
306 return E_NOTIMPL;
309 static const IStreamVtbl DummyRegStreamVTable =
311 IStream_fnQueryInterface,
312 IStream_fnAddRefDummy, /* Overridden */
313 IStream_fnReleaseDummy, /* Overridden */
314 IStream_fnReadDummy, /* Overridden */
315 IStream_fnWrite,
316 IStream_fnSeek,
317 IStream_fnSetSize,
318 IStream_fnCopyTo,
319 IStream_fnCommit,
320 IStream_fnRevert,
321 IStream_fnLockUnlockRegion,
322 IStream_fnLockUnlockRegion,
323 IStream_fnStat,
324 IStream_fnClone
327 /* Dummy registry stream object */
328 static ISHRegStream rsDummyRegStream =
330 &DummyRegStreamVTable,
332 NULL,
333 NULL,
338 /**************************************************************************
339 * IStream_Create
341 * Internal helper: Create and initialise a new registry stream object.
343 static IStream *IStream_Create(HKEY hKey, LPBYTE pbBuffer, DWORD dwLength)
345 ISHRegStream* regStream;
347 regStream = HeapAlloc(GetProcessHeap(), 0, sizeof(ISHRegStream));
349 if (regStream)
351 regStream->lpVtbl = &rstvt;
352 regStream->ref = 1;
353 regStream->hKey = hKey;
354 regStream->pbBuffer = pbBuffer;
355 regStream->dwLength = dwLength;
356 regStream->dwPos = 0;
358 TRACE ("Returning %p\n", regStream);
359 return (IStream *)regStream;
362 /*************************************************************************
363 * SHOpenRegStream2A [SHLWAPI.@]
365 * Create a stream to read binary registry data.
367 * PARAMS
368 * hKey [I] Registry handle
369 * pszSubkey [I] The sub key name
370 * pszValue [I] The value name under the sub key
371 * dwMode [I] Unused
373 * RETURNS
374 * Success: An IStream interface referring to the registry data
375 * Failure: NULL, if the registry key could not be opened or is not binary.
377 IStream * WINAPI SHOpenRegStream2A(HKEY hKey, LPCSTR pszSubkey,
378 LPCSTR pszValue,DWORD dwMode)
380 HKEY hStrKey = NULL;
381 LPBYTE lpBuff = NULL;
382 DWORD dwLength, dwType;
384 TRACE("(%p,%s,%s,0x%08lx)\n", hKey, pszSubkey, pszValue, dwMode);
386 /* Open the key, read in binary data and create stream */
387 if (!RegOpenKeyExA (hKey, pszSubkey, 0, KEY_READ, &hStrKey) &&
388 !RegQueryValueExA (hStrKey, pszValue, 0, 0, 0, &dwLength) &&
389 (lpBuff = HeapAlloc (GetProcessHeap(), 0, dwLength)) &&
390 !RegQueryValueExA (hStrKey, pszValue, 0, &dwType, lpBuff, &dwLength) &&
391 dwType == REG_BINARY)
392 return IStream_Create(hStrKey, lpBuff, dwLength);
394 HeapFree (GetProcessHeap(), 0, lpBuff);
395 if (hStrKey)
396 RegCloseKey(hStrKey);
397 return NULL;
400 /*************************************************************************
401 * SHOpenRegStream2W [SHLWAPI.@]
403 * See SHOpenRegStream2A.
405 IStream * WINAPI SHOpenRegStream2W(HKEY hKey, LPCWSTR pszSubkey,
406 LPCWSTR pszValue, DWORD dwMode)
408 HKEY hStrKey = NULL;
409 LPBYTE lpBuff = NULL;
410 DWORD dwLength, dwType;
412 TRACE("(%p,%s,%s,0x%08lx)\n", hKey, debugstr_w(pszSubkey),
413 debugstr_w(pszValue), dwMode);
415 /* Open the key, read in binary data and create stream */
416 if (!RegOpenKeyExW (hKey, pszSubkey, 0, KEY_READ, &hStrKey) &&
417 !RegQueryValueExW (hStrKey, pszValue, 0, 0, 0, &dwLength) &&
418 (lpBuff = HeapAlloc (GetProcessHeap(), 0, dwLength)) &&
419 !RegQueryValueExW (hStrKey, pszValue, 0, &dwType, lpBuff, &dwLength) &&
420 dwType == REG_BINARY)
421 return IStream_Create(hStrKey, lpBuff, dwLength);
423 HeapFree (GetProcessHeap(), 0, lpBuff);
424 if (hStrKey)
425 RegCloseKey(hStrKey);
426 return NULL;
429 /*************************************************************************
430 * SHOpenRegStreamA [SHLWAPI.@]
432 * Create a stream to read binary registry data.
434 * PARAMS
435 * hKey [I] Registry handle
436 * pszSubkey [I] The sub key name
437 * pszValue [I] The value name under the sub key
438 * dwMode [I] STGM mode for opening the file
440 * RETURNS
441 * Success: An IStream interface referring to the registry data
442 * Failure: If the registry key could not be opened or is not binary,
443 * A dummy (empty) IStream object is returned.
445 IStream * WINAPI SHOpenRegStreamA(HKEY hkey, LPCSTR pszSubkey,
446 LPCSTR pszValue, DWORD dwMode)
448 IStream *iStream;
450 TRACE("(%p,%s,%s,0x%08lx)\n", hkey, pszSubkey, pszValue, dwMode);
452 iStream = SHOpenRegStream2A(hkey, pszSubkey, pszValue, dwMode);
453 return iStream ? iStream : (IStream *)&rsDummyRegStream;
456 /*************************************************************************
457 * SHOpenRegStreamW [SHLWAPI.@]
459 * See SHOpenRegStreamA.
461 IStream * WINAPI SHOpenRegStreamW(HKEY hkey, LPCWSTR pszSubkey,
462 LPCWSTR pszValue, DWORD dwMode)
464 IStream *iStream;
466 TRACE("(%p,%s,%s,0x%08lx)\n", hkey, debugstr_w(pszSubkey),
467 debugstr_w(pszValue), dwMode);
468 iStream = SHOpenRegStream2W(hkey, pszSubkey, pszValue, dwMode);
469 return iStream ? iStream : (IStream *)&rsDummyRegStream;
472 /*************************************************************************
473 * @ [SHLWAPI.12]
475 * Create an IStream object on a block of memory.
477 * PARAMS
478 * lpbData [I] Memory block to create the IStream object on
479 * dwDataLen [I] Length of data block
481 * RETURNS
482 * Success: A pointer to the IStream object.
483 * Failure: NULL, if any parameters are invalid or an error occurs.
485 * NOTES
486 * A copy of the memory pointed to by lpbData is made, and is freed
487 * when the stream is released.
489 IStream * WINAPI SHCreateMemStream(LPBYTE lpbData, DWORD dwDataLen)
491 IStream *iStrmRet = NULL;
493 TRACE("(%p,%ld)\n", lpbData, dwDataLen);
495 if (lpbData)
497 LPBYTE lpbDup = HeapAlloc(GetProcessHeap(), 0, dwDataLen);
499 if (lpbDup)
501 memcpy(lpbDup, lpbData, dwDataLen);
502 iStrmRet = IStream_Create(NULL, lpbDup, dwDataLen);
504 if (!iStrmRet)
505 HeapFree(GetProcessHeap(), 0, lpbDup);
508 return iStrmRet;
511 /*************************************************************************
512 * SHCreateStreamWrapper [SHLWAPI.@]
514 * Create an IStream object on a block of memory.
516 * PARAMS
517 * lpbData [I] Memory block to create the IStream object on
518 * dwDataLen [I] Length of data block
519 * dwReserved [I] Reserved, Must be 0.
520 * lppStream [O] Destination for IStream object
522 * RETURNS
523 * Success: S_OK. lppStream contains the new IStream object.
524 * Failure: E_INVALIDARG, if any parameters are invalid,
525 * E_OUTOFMEMORY if memory allocation fails.
527 * NOTES
528 * The stream assumes ownership of the memory passed to it.
530 HRESULT WINAPI SHCreateStreamWrapper(LPBYTE lpbData, DWORD dwDataLen,
531 DWORD dwReserved, IStream **lppStream)
533 IStream* lpStream;
535 if (lppStream)
536 *lppStream = NULL;
538 if(dwReserved || !lppStream)
539 return E_INVALIDARG;
541 lpStream = IStream_Create(NULL, lpbData, dwDataLen);
543 if(!lpStream)
544 return E_OUTOFMEMORY;
546 IStream_QueryInterface(lpStream, &IID_IStream, (void**)lppStream);
547 IStream_Release(lpStream);
548 return S_OK;