Release 20030408.
[wine/gsoc-2012-control.git] / dlls / shlwapi / istream.c
blob3510c6e53fbf12bf339083e4528f2adf8a8d6105
1 /*
2 * SHLWAPI IStream functions
4 * Copyright 2002 Jon Griffiths
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 #include <string.h>
22 #define NONAMELESSUNION
23 #define NONAMELESSSTRUCT
24 #include "winbase.h"
25 #include "winerror.h"
26 #include "winnls.h"
27 #define NO_SHLWAPI_REG
28 #define NO_SHLWAPI_PATH
29 #include "shlwapi.h"
30 #include "wine/debug.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(shell);
34 /* Layout of ISHFileStream object */
35 typedef struct
37 ICOM_VFIELD(IStream);
38 ULONG ref;
39 HANDLE hFile;
40 DWORD dwMode;
41 LPOLESTR lpszPath;
42 DWORD type;
43 DWORD grfStateBits;
44 } ISHFileStream;
46 static HRESULT WINAPI IStream_fnCommit(IStream*,DWORD);
49 /**************************************************************************
50 * IStream_fnQueryInterface
52 static HRESULT WINAPI IStream_fnQueryInterface(IStream *iface, REFIID riid, LPVOID *ppvObj)
54 ICOM_THIS(ISHFileStream, iface);
56 TRACE("(%p,%s,%p)\n", This, debugstr_guid(riid), ppvObj);
58 *ppvObj = NULL;
60 if(IsEqualIID(riid, &IID_IUnknown) ||
61 IsEqualIID(riid, &IID_IStream))
63 *ppvObj = This;
65 IStream_AddRef((IStream*)*ppvObj);
66 return S_OK;
68 return E_NOINTERFACE;
71 /**************************************************************************
72 * IStream_fnAddRef
74 static ULONG WINAPI IStream_fnAddRef(IStream *iface)
76 ICOM_THIS(ISHFileStream, iface);
78 TRACE("(%p)\n", This);
79 return InterlockedIncrement(&This->ref);
82 /**************************************************************************
83 * IStream_fnRelease
85 static ULONG WINAPI IStream_fnRelease(IStream *iface)
87 ICOM_THIS(ISHFileStream, iface);
88 ULONG ulRet;
90 TRACE("(%p)\n", This);
92 if (!(ulRet = InterlockedDecrement(&This->ref)))
94 IStream_fnCommit(iface, 0); /* If ever buffered, this will be needed */
95 LocalFree((HLOCAL)This->lpszPath);
96 CloseHandle(This->hFile);
97 HeapFree(GetProcessHeap(), 0, This);
99 return ulRet;
102 /**************************************************************************
103 * IStream_fnRead
105 static HRESULT WINAPI IStream_fnRead(IStream *iface, void* pv, ULONG cb, ULONG* pcbRead)
107 ICOM_THIS(ISHFileStream, iface);
108 HRESULT hRet = S_OK;
109 DWORD dwRead = 0;
111 TRACE("(%p,%p,0x%08lx,%p)\n", This, pv, cb, pcbRead);
113 if (!pv)
114 hRet = STG_E_INVALIDPOINTER;
115 else if (!ReadFile(This->hFile, pv, cb, &dwRead, NULL))
117 hRet = (HRESULT)GetLastError();
118 if(hRet > 0)
119 hRet = HRESULT_FROM_WIN32(hRet);
121 if (pcbRead)
122 *pcbRead = dwRead;
123 return hRet;
126 /**************************************************************************
127 * IStream_fnWrite
129 static HRESULT WINAPI IStream_fnWrite(IStream *iface, const void* pv, ULONG cb, ULONG* pcbWritten)
131 ICOM_THIS(ISHFileStream, iface);
132 HRESULT hRet = S_OK;
133 DWORD dwWritten = 0;
135 TRACE("(%p,%p,0x%08lx,%p)\n", This, pv, cb, pcbWritten);
137 if (!pv)
138 hRet = STG_E_INVALIDPOINTER;
139 else if (!(This->dwMode & STGM_WRITE))
140 hRet = E_FAIL;
141 else if (!WriteFile(This->hFile, pv, cb, &dwWritten, NULL))
143 hRet = (HRESULT)GetLastError();
144 if(hRet > 0)
145 hRet = HRESULT_FROM_WIN32(hRet);
147 if (pcbWritten)
148 *pcbWritten = dwWritten;
149 return hRet;
152 /**************************************************************************
153 * IStream_fnSeek
155 static HRESULT WINAPI IStream_fnSeek(IStream *iface, LARGE_INTEGER dlibMove,
156 DWORD dwOrigin, ULARGE_INTEGER* pNewPos)
158 ICOM_THIS(ISHFileStream, iface);
159 DWORD dwPos;
161 TRACE("(%p,%ld,%ld,%p)\n", This, dlibMove.s.LowPart, dwOrigin, pNewPos);
163 IStream_fnCommit(iface, 0); /* If ever buffered, this will be needed */
164 dwPos = SetFilePointer(This->hFile, dlibMove.s.LowPart, NULL, dwOrigin);
166 if (pNewPos)
168 pNewPos->s.HighPart = 0;
169 pNewPos->s.LowPart = dwPos;
171 return S_OK;
174 /**************************************************************************
175 * IStream_fnSetSize
177 static HRESULT WINAPI IStream_fnSetSize(IStream *iface, ULARGE_INTEGER libNewSize)
179 ICOM_THIS(ISHFileStream, iface);
181 TRACE("(%p,%ld)\n", This, libNewSize.s.LowPart);
182 IStream_fnCommit(iface, 0); /* If ever buffered, this will be needed */
183 return E_NOTIMPL;
186 /**************************************************************************
187 * IStream_fnCopyTo
189 static HRESULT WINAPI IStream_fnCopyTo(IStream *iface, IStream* pstm, ULARGE_INTEGER cb,
190 ULARGE_INTEGER* pcbRead, ULARGE_INTEGER* pcbWritten)
192 ICOM_THIS(ISHFileStream, iface);
193 char copyBuff[1024];
194 ULONGLONG ulSize;
195 HRESULT hRet = S_OK;
197 TRACE("(%p,%p,%ld,%p,%p)\n", This, pstm, cb.s.LowPart, pcbRead, pcbWritten);
199 if (pcbRead)
200 pcbRead->QuadPart = 0;
201 if (pcbWritten)
202 pcbWritten->QuadPart = 0;
204 if (!pstm)
205 return STG_E_INVALIDPOINTER;
207 IStream_fnCommit(iface, 0); /* If ever buffered, this will be needed */
209 /* Copy data */
210 ulSize = cb.QuadPart;
211 while (ulSize)
213 ULONG ulLeft, ulAmt;
215 ulLeft = ulSize > sizeof(copyBuff) ? sizeof(copyBuff) : ulSize;
217 /* Read */
218 hRet = IStream_fnRead(iface, copyBuff, ulLeft, &ulAmt);
219 if (pcbRead)
220 pcbRead->QuadPart += ulAmt;
221 if (FAILED(hRet) || ulAmt != ulLeft)
222 break;
224 /* Write */
225 hRet = IStream_fnWrite(pstm, copyBuff, ulLeft, &ulAmt);
226 if (pcbWritten)
227 pcbWritten->QuadPart += ulAmt;
228 if (FAILED(hRet) || ulAmt != ulLeft)
229 break;
231 ulSize -= ulLeft;
233 return hRet;
236 /**************************************************************************
237 * IStream_fnCommit
239 static HRESULT WINAPI IStream_fnCommit(IStream *iface, DWORD grfCommitFlags)
241 ICOM_THIS(ISHFileStream, iface);
243 TRACE("(%p,%ld)\n", This, grfCommitFlags);
244 /* Currently unbuffered: This function is not needed */
245 return S_OK;
248 /**************************************************************************
249 * IStream_fnRevert
251 static HRESULT WINAPI IStream_fnRevert(IStream *iface)
253 ICOM_THIS(ISHFileStream, iface);
255 TRACE("(%p)\n", This);
256 return E_NOTIMPL;
259 /**************************************************************************
260 * IStream_fnLockUnlockRegion
262 static HRESULT WINAPI IStream_fnLockUnlockRegion(IStream *iface, ULARGE_INTEGER libOffset,
263 ULARGE_INTEGER cb, DWORD dwLockType)
265 ICOM_THIS(ISHFileStream, iface);
266 TRACE("(%p,%ld,%ld,%ld)\n", This, libOffset.s.LowPart, cb.s.LowPart, dwLockType);
267 return E_NOTIMPL;
270 /*************************************************************************
271 * IStream_fnStat
273 static HRESULT WINAPI IStream_fnStat(IStream *iface, STATSTG* lpStat,
274 DWORD grfStatFlag)
276 ICOM_THIS(ISHFileStream, iface);
277 BY_HANDLE_FILE_INFORMATION fi;
278 HRESULT hRet = S_OK;
280 TRACE("(%p,%p,%ld)\n", This, lpStat, grfStatFlag);
282 if (!grfStatFlag)
283 hRet = STG_E_INVALIDPOINTER;
284 else
286 memset(&fi, 0, sizeof(fi));
287 GetFileInformationByHandle(This->hFile, &fi);
289 if (grfStatFlag & STATFLAG_NONAME)
290 lpStat->pwcsName = NULL;
291 else
292 lpStat->pwcsName = StrDupW(This->lpszPath);
293 lpStat->type = This->type;
294 lpStat->cbSize.s.LowPart = fi.nFileSizeLow;
295 lpStat->cbSize.s.HighPart = fi.nFileSizeHigh;
296 lpStat->mtime = fi.ftLastWriteTime;
297 lpStat->ctime = fi.ftCreationTime;
298 lpStat->atime = fi.ftLastAccessTime;
299 lpStat->grfMode = This->dwMode;
300 lpStat->grfLocksSupported = 0;
301 memcpy(&lpStat->clsid, &IID_IStream, sizeof(CLSID));
302 lpStat->grfStateBits = This->grfStateBits;
303 lpStat->reserved = 0;
305 return hRet;
308 /*************************************************************************
309 * IStream_fnClone
311 static HRESULT WINAPI IStream_fnClone(IStream *iface, IStream** ppstm)
313 ICOM_THIS(ISHFileStream, iface);
315 TRACE("(%p)\n",This);
316 if (ppstm)
317 *ppstm = NULL;
318 return E_NOTIMPL;
321 static struct ICOM_VTABLE(IStream) SHLWAPI_fsVTable =
323 ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE
324 IStream_fnQueryInterface,
325 IStream_fnAddRef,
326 IStream_fnRelease,
327 IStream_fnRead,
328 IStream_fnWrite,
329 IStream_fnSeek,
330 IStream_fnSetSize,
331 IStream_fnCopyTo,
332 IStream_fnCommit,
333 IStream_fnRevert,
334 IStream_fnLockUnlockRegion,
335 IStream_fnLockUnlockRegion,
336 IStream_fnStat,
337 IStream_fnClone
340 /**************************************************************************
341 * IStream_Create
343 * Internal helper: Create and initialise a new file stream object.
345 static IStream *IStream_Create(LPCWSTR lpszPath, HANDLE hFile, DWORD dwMode)
347 ISHFileStream* fileStream;
349 fileStream = (ISHFileStream*)HeapAlloc(GetProcessHeap(), 0, sizeof(ISHFileStream));
351 if (fileStream)
353 ICOM_VTBL(fileStream) = &SHLWAPI_fsVTable;
354 fileStream->ref = 1;
355 fileStream->hFile = hFile;
356 fileStream->dwMode = dwMode;
357 fileStream->lpszPath = StrDupW(lpszPath);
358 fileStream->type = 0; /* FIXME */
359 fileStream->grfStateBits = 0; /* FIXME */
361 TRACE ("Returning %p\n", fileStream);
362 return (IStream *)fileStream;
365 /*************************************************************************
366 * SHCreateStreamOnFileEx [SHLWAPI.@]
368 * Create a stream on a file.
370 * PARAMS
371 * lpszPath [I] Path of file to create stream on
372 * dwMode [I] Mode to create stream in
373 * dwAttributes [I] Attributes of the file
374 * bCreate [I] Whether to create the file if it doesn't exist
375 * lpTemplate [I] Reserved, must be NULL
376 * lppStream [O] Destination for created stream
378 * RETURNS
379 * Success: S_OK. lppStream contains the new stream object
380 * Failure: E_INVALIDARG if any parameter is invalid, or an HRESULT error code
382 * NOTES
383 * This function is available in Unicode only.
385 HRESULT WINAPI SHCreateStreamOnFileEx(LPCWSTR lpszPath, DWORD dwMode,
386 DWORD dwAttributes, BOOL bCreate,
387 IStream *lpTemplate, IStream **lppStream)
389 DWORD dwAccess, dwShare, dwCreate;
390 HANDLE hFile;
392 TRACE("(%s,%ld,0x%08lX,%d,%p,%p)\n", debugstr_w(lpszPath), dwMode,
393 dwAttributes, bCreate, lpTemplate, lppStream);
395 if (!lpszPath || !lppStream || lpTemplate)
396 return E_INVALIDARG;
398 *lppStream = NULL;
400 if (dwMode & ~(STGM_WRITE|STGM_READWRITE|STGM_SHARE_DENY_NONE|STGM_SHARE_DENY_READ|STGM_CREATE))
402 WARN("Invalid mode 0x%08lX\n", dwMode);
403 return E_INVALIDARG;
406 /* Access */
407 switch (dwMode & (STGM_WRITE|STGM_READWRITE))
409 case STGM_READWRITE|STGM_WRITE:
410 case STGM_READWRITE:
411 dwAccess = GENERIC_READ|GENERIC_WRITE;
412 break;
413 case STGM_WRITE:
414 dwAccess = GENERIC_WRITE;
415 break;
416 default:
417 dwAccess = GENERIC_READ;
418 break;
421 /* Sharing */
422 switch (dwMode & STGM_SHARE_DENY_READ)
424 case STGM_SHARE_DENY_READ:
425 dwShare = FILE_SHARE_WRITE;
426 break;
427 case STGM_SHARE_DENY_WRITE:
428 dwShare = FILE_SHARE_READ;
429 break;
430 case STGM_SHARE_EXCLUSIVE:
431 dwShare = 0;
432 break;
433 default:
434 dwShare = FILE_SHARE_READ|FILE_SHARE_WRITE;
437 /* FIXME: Creation Flags, MSDN is fuzzy on the mapping... */
438 if (dwMode == STGM_FAILIFTHERE)
439 dwCreate = bCreate ? CREATE_NEW : OPEN_EXISTING;
440 else if (dwMode & STGM_CREATE)
441 dwCreate = CREATE_ALWAYS;
442 else
443 dwCreate = OPEN_ALWAYS;
445 /* Open HANDLE to file */
446 hFile = CreateFileW(lpszPath, dwAccess, dwShare, NULL, dwCreate,
447 dwAttributes, 0);
449 if(hFile == INVALID_HANDLE_VALUE)
451 HRESULT hRet = (HRESULT)GetLastError();
452 if(hRet > 0)
453 hRet = HRESULT_FROM_WIN32(hRet);
454 return hRet;
457 *lppStream = IStream_Create(lpszPath, hFile, dwMode);
459 if(!*lppStream)
461 CloseHandle(hFile);
462 return E_OUTOFMEMORY;
464 return S_OK;
467 /*************************************************************************
468 * SHCreateStreamOnFileW [SHLWAPI.@]
470 * See SHCreateStreamOnFileA.
472 HRESULT WINAPI SHCreateStreamOnFileW(LPCWSTR lpszPath, DWORD dwMode,
473 IStream **lppStream)
475 DWORD dwAttr;
477 TRACE("(%s,%ld,%p)\n", debugstr_w(lpszPath), dwMode, lppStream);
479 if (!lpszPath || !lppStream)
480 return E_INVALIDARG;
482 dwAttr = GetFileAttributesW(lpszPath);
483 if (dwAttr == -1u)
484 dwAttr = 0;
486 return SHCreateStreamOnFileEx(lpszPath, dwMode|STGM_WRITE, dwAttr,
487 TRUE, NULL, lppStream);
490 /*************************************************************************
491 * SHCreateStreamOnFileA [SHLWAPI.@]
493 * Create a stream on a file.
495 * PARAMS
496 * lpszPath [I] Path of file to create stream on
497 * dwMode [I] Mode to create stream in
498 * lppStream [O] Destination for created IStream object
500 * RETURNS
501 * Success: S_OK. lppStream contains the new IStream object
502 * Failure: E_INVALIDARG if any parameter is invalid, or an HRESULT error code
504 HRESULT WINAPI SHCreateStreamOnFileA(LPCSTR lpszPath, DWORD dwMode,
505 IStream **lppStream)
507 WCHAR szPath[MAX_PATH];
509 TRACE("(%s,%ld,%p)\n", debugstr_a(lpszPath), dwMode, lppStream);
511 if (!lpszPath)
512 return E_INVALIDARG;
513 MultiByteToWideChar(0, 0, lpszPath, -1, szPath, MAX_PATH);
514 return SHCreateStreamOnFileW(szPath, dwMode, lppStream);
517 /*************************************************************************
518 * @ [SHLWAPI.184]
520 * Call IStream_Read() on a stream.
522 * PARAMS
523 * lpStream [I] IStream object
524 * lpvDest [O] Destination for data read
525 * ulSize [I] Size of data to read
527 * RETURNS
528 * Success: S_OK. ulSize bytes have been read from the stream into lpvDest.
529 * Failure: An HRESULT error code, or E_FAIL if the read succeeded but the
530 * number of bytes read does not match.
532 HRESULT WINAPI SHLWAPI_184(IStream *lpStream, LPVOID lpvDest, ULONG ulSize)
534 ULONG ulRead;
535 HRESULT hRet;
537 TRACE("(%p,%p,%ld)\n", lpStream, lpvDest, ulSize);
539 hRet = IStream_Read(lpStream, lpvDest, ulSize, &ulRead);
541 if (SUCCEEDED(hRet) && ulRead != ulSize)
542 hRet = E_FAIL;
543 return hRet;
546 /*************************************************************************
547 * @ [SHLWAPI.166]
549 * Determine if a stream has 0 length.
551 * PARAMS
552 * lpStream [I] IStream object
554 * RETURNS
555 * TRUE: If the stream has 0 length
556 * FALSE: Otherwise.
558 BOOL WINAPI SHLWAPI_166(IStream *lpStream)
560 STATSTG statstg;
561 BOOL bRet = TRUE;
563 TRACE("(%p)\n", lpStream);
565 memset(&statstg, 0, sizeof(statstg));
567 if(SUCCEEDED(IStream_Stat(lpStream, &statstg, 1)))
569 if(statstg.cbSize.QuadPart)
570 bRet = FALSE; /* Non-Zero */
572 else
574 DWORD dwDummy;
576 /* Try to read from the stream */
577 if(SUCCEEDED(SHLWAPI_184(lpStream, &dwDummy, sizeof(dwDummy))))
579 LARGE_INTEGER zero;
580 zero.QuadPart = 0;
582 IStream_Seek(lpStream, zero, 0, NULL);
583 bRet = FALSE; /* Non-Zero */
586 return bRet;
589 /*************************************************************************
590 * @ [SHLWAPI.212]
592 * Call IStream_Write() on a stream.
594 * PARAMS
595 * lpStream [I] IStream object
596 * lpvSrc [I] Source for data to write
597 * ulSize [I] Size of data
599 * RETURNS
600 * Success: S_OK. ulSize bytes have been written to the stream from lpvSrc.
601 * Failure: An HRESULT error code, or E_FAIL if the write succeeded but the
602 * number of bytes written does not match.
604 HRESULT WINAPI SHLWAPI_212(IStream *lpStream, LPCVOID lpvSrc, ULONG ulSize)
606 ULONG ulWritten;
607 HRESULT hRet;
609 TRACE("(%p,%p,%ld)\n", lpStream, lpvSrc, ulSize);
611 hRet = IStream_Write(lpStream, lpvSrc, ulSize, &ulWritten);
613 if (SUCCEEDED(hRet) && ulWritten != ulSize)
614 hRet = E_FAIL;
616 return hRet;
619 /*************************************************************************
620 * @ [SHLWAPI.213]
622 * Seek to the start of a stream.
624 * PARAMS
625 * lpStream [I] IStream object
627 * RETURNS
628 * Success: S_OK. The current position within the stream is updated
629 * Failure: An HRESULT error code.
631 HRESULT WINAPI SHLWAPI_213(IStream *lpStream)
633 LARGE_INTEGER zero;
634 TRACE("(%p)\n", lpStream);
635 zero.QuadPart = 0;
636 return IStream_Seek(lpStream, zero, 0, NULL);
639 /*************************************************************************
640 * @ [SHLWAPI.214]
642 * Get the size of a stream.
644 * PARAMS
645 * lpStream [I] IStream object
646 * lpulSize [O] Destination for size
648 * RETURNS
649 * Success: S_OK. lpulSize contains the size of the stream.
650 * Failure: An HRESULT error code.
652 HRESULT WINAPI SHLWAPI_214(IStream *lpStream, ULARGE_INTEGER* lpulSize)
654 STATSTG statstg;
655 HRESULT hRet;
657 TRACE("(%p,%p)\n", lpStream, lpulSize);
659 memset(&statstg, 0, sizeof(statstg));
661 hRet = IStream_Stat(lpStream, &statstg, 1);
663 if (SUCCEEDED(hRet) && lpulSize)
664 *lpulSize = statstg.cbSize;
665 return hRet;