VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/ApplianceImplIO.cpp@ 41026

Last change on this file since 41026 was 40329, checked in by vboxsync, 13 years ago

Main/Runtime/VBoxManage: SHA256 support comes with OVF 2.0

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 43.5 KB
Line 
1/* $Id: ApplianceImplIO.cpp 40329 2012-03-02 16:13:50Z vboxsync $ */
2/** @file
3 *
4 * IO helper for IAppliance COM class implementations.
5 */
6
7/*
8 * Copyright (C) 2010 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19/******************************************************************************
20 * Header Files *
21 ******************************************************************************/
22
23#include "ProgressImpl.h"
24#include "ApplianceImpl.h"
25#include "ApplianceImplPrivate.h"
26
27#include <iprt/tar.h>
28#include <iprt/sha.h>
29#include <iprt/path.h>
30#include <iprt/asm.h>
31#include <iprt/stream.h>
32#include <iprt/circbuf.h>
33#include <VBox/vd.h>
34
35/******************************************************************************
36 * Structures and Typedefs *
37 ******************************************************************************/
38
39typedef struct FILESTORAGEINTERNAL
40{
41 /** File handle. */
42 RTFILE file;
43 /** Completion callback. */
44 PFNVDCOMPLETED pfnCompleted;
45} FILESTORAGEINTERNAL, *PFILESTORAGEINTERNAL;
46
47typedef struct TARSTORAGEINTERNAL
48{
49 /** Tar handle. */
50 RTTARFILE file;
51 /** Completion callback. */
52 PFNVDCOMPLETED pfnCompleted;
53} TARSTORAGEINTERNAL, *PTARSTORAGEINTERNAL;
54
55typedef struct SHASTORAGEINTERNAL
56{
57 /** Completion callback. */
58 PFNVDCOMPLETED pfnCompleted;
59 /** Storage handle for the next callback in chain. */
60 void *pvStorage;
61 /** Current file open mode. */
62 uint32_t fOpenMode;
63 /** Our own storage handle. */
64 PSHASTORAGE pShaStorage;
65 /** Circular buffer used for transferring data from/to the worker thread. */
66 PRTCIRCBUF pCircBuf;
67 /** Current absolute position (regardless of the real read/written data). */
68 uint64_t cbCurAll;
69 /** Current real position in the file. */
70 uint64_t cbCurFile;
71 /** Handle of the worker thread. */
72 RTTHREAD pWorkerThread;
73 /** Status of the worker thread. */
74 volatile uint32_t u32Status;
75 /** Event for signaling a new status. */
76 RTSEMEVENT newStatusEvent;
77 /** Event for signaling a finished task of the worker thread. */
78 RTSEMEVENT workFinishedEvent;
79 /** SHA1/SHA256 calculation context. */
80 union
81 {
82 RTSHA1CONTEXT Sha1;
83 RTSHA256CONTEXT Sha256;
84 } ctx;
85 /** Write mode only: Memory buffer for writing zeros. */
86 void *pvZeroBuf;
87 /** Write mode only: Size of the zero memory buffer. */
88 size_t cbZeroBuf;
89 /** Read mode only: Indicate if we reached end of file. */
90 volatile bool fEOF;
91// uint64_t calls;
92// uint64_t waits;
93} SHASTORAGEINTERNAL, *PSHASTORAGEINTERNAL;
94
95/******************************************************************************
96 * Defined Constants And Macros *
97 ******************************************************************************/
98
99#define STATUS_WAIT UINT32_C(0)
100#define STATUS_WRITE UINT32_C(1)
101#define STATUS_WRITING UINT32_C(2)
102#define STATUS_READ UINT32_C(3)
103#define STATUS_READING UINT32_C(4)
104#define STATUS_END UINT32_C(5)
105
106/* Enable for getting some flow history. */
107#if 0
108# define DEBUG_PRINT_FLOW() RTPrintf("%s\n", __FUNCTION__)
109#else
110# define DEBUG_PRINT_FLOW() do {} while(0)
111#endif
112
113/******************************************************************************
114 * Internal Functions *
115 ******************************************************************************/
116
117/******************************************************************************
118 * Internal: RTFile interface
119 ******************************************************************************/
120
121static int fileOpenCallback(void * /* pvUser */, const char *pszLocation, uint32_t fOpen,
122 PFNVDCOMPLETED pfnCompleted, void **ppInt)
123{
124 /* Validate input. */
125 AssertPtrReturn(ppInt, VERR_INVALID_POINTER);
126 AssertPtrNullReturn(pfnCompleted, VERR_INVALID_PARAMETER);
127
128 DEBUG_PRINT_FLOW();
129
130 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)RTMemAllocZ(sizeof(FILESTORAGEINTERNAL));
131 if (!pInt)
132 return VERR_NO_MEMORY;
133
134 pInt->pfnCompleted = pfnCompleted;
135
136 int rc = RTFileOpen(&pInt->file, pszLocation, fOpen);
137
138 if (RT_FAILURE(rc))
139 RTMemFree(pInt);
140 else
141 *ppInt = pInt;
142
143 return rc;
144}
145
146static int fileCloseCallback(void * /* pvUser */, void *pvStorage)
147{
148 /* Validate input. */
149 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
150
151 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
152
153 DEBUG_PRINT_FLOW();
154
155 int rc = RTFileClose(pInt->file);
156
157 /* Cleanup */
158 RTMemFree(pInt);
159
160 return rc;
161}
162
163static int fileDeleteCallback(void * /* pvUser */, const char *pcszFilename)
164{
165 DEBUG_PRINT_FLOW();
166
167 return RTFileDelete(pcszFilename);
168}
169
170static int fileMoveCallback(void * /* pvUser */, const char *pcszSrc, const char *pcszDst, unsigned fMove)
171{
172 DEBUG_PRINT_FLOW();
173
174 return RTFileMove(pcszSrc, pcszDst, fMove);
175}
176
177static int fileGetFreeSpaceCallback(void * /* pvUser */, const char *pcszFilename, int64_t *pcbFreeSpace)
178{
179 /* Validate input. */
180 AssertPtrReturn(pcszFilename, VERR_INVALID_POINTER);
181 AssertPtrReturn(pcbFreeSpace, VERR_INVALID_POINTER);
182
183 DEBUG_PRINT_FLOW();
184
185 return VERR_NOT_IMPLEMENTED;
186}
187
188static int fileGetModificationTimeCallback(void * /* pvUser */, const char *pcszFilename, PRTTIMESPEC pModificationTime)
189{
190 /* Validate input. */
191 AssertPtrReturn(pcszFilename, VERR_INVALID_POINTER);
192 AssertPtrReturn(pModificationTime, VERR_INVALID_POINTER);
193
194 DEBUG_PRINT_FLOW();
195
196 return VERR_NOT_IMPLEMENTED;
197}
198
199static int fileGetSizeCallback(void * /* pvUser */, void *pvStorage, uint64_t *pcbSize)
200{
201 /* Validate input. */
202 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
203
204 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
205
206 DEBUG_PRINT_FLOW();
207
208 return RTFileGetSize(pInt->file, pcbSize);
209}
210
211static int fileSetSizeCallback(void * /* pvUser */, void *pvStorage, uint64_t cbSize)
212{
213 /* Validate input. */
214 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
215
216 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
217
218 DEBUG_PRINT_FLOW();
219
220 return RTFileSetSize(pInt->file, cbSize);
221}
222
223static int fileWriteSyncCallback(void * /* pvUser */, void *pvStorage, uint64_t uOffset,
224 const void *pvBuf, size_t cbWrite, size_t *pcbWritten)
225{
226 /* Validate input. */
227 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
228
229 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
230
231 return RTFileWriteAt(pInt->file, uOffset, pvBuf, cbWrite, pcbWritten);
232}
233
234static int fileReadSyncCallback(void * /* pvUser */, void *pvStorage, uint64_t uOffset,
235 void *pvBuf, size_t cbRead, size_t *pcbRead)
236{
237 /* Validate input. */
238 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
239
240// DEBUG_PRINT_FLOW();
241
242 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
243
244 return RTFileReadAt(pInt->file, uOffset, pvBuf, cbRead, pcbRead);
245}
246
247static int fileFlushSyncCallback(void * /* pvUser */, void *pvStorage)
248{
249 /* Validate input. */
250 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
251
252 DEBUG_PRINT_FLOW();
253
254 PFILESTORAGEINTERNAL pInt = (PFILESTORAGEINTERNAL)pvStorage;
255
256 return RTFileFlush(pInt->file);
257}
258
259/******************************************************************************
260 * Internal: RTTar interface
261 ******************************************************************************/
262
263static int tarOpenCallback(void *pvUser, const char *pszLocation, uint32_t fOpen,
264 PFNVDCOMPLETED pfnCompleted, void **ppInt)
265{
266 /* Validate input. */
267 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
268 AssertPtrReturn(ppInt, VERR_INVALID_POINTER);
269 AssertPtrNullReturn(pfnCompleted, VERR_INVALID_PARAMETER);
270// AssertReturn(!(fOpen & RTFILE_O_READWRITE), VERR_INVALID_PARAMETER);
271
272 RTTAR tar = (RTTAR)pvUser;
273
274 DEBUG_PRINT_FLOW();
275
276 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)RTMemAllocZ(sizeof(TARSTORAGEINTERNAL));
277 if (!pInt)
278 return VERR_NO_MEMORY;
279
280 pInt->pfnCompleted = pfnCompleted;
281
282 int rc = VINF_SUCCESS;
283
284 if ( fOpen & RTFILE_O_READ
285 && !(fOpen & RTFILE_O_WRITE))
286 {
287 /* Read only is a little bit more complicated than writing, cause we
288 * need streaming functionality. First try to open the file on the
289 * current file position. If this is the file the caller requested, we
290 * are fine. If not seek to the next file in the stream and check
291 * again. This is repeated until EOF of the OVA. */
292 /*
293 *
294 *
295 * TODO: recheck this with more VDMKs (or what else) in an test OVA.
296 *
297 *
298 */
299 bool fFound = false;
300 for(;;)
301 {
302 char *pszFilename = 0;
303 rc = RTTarCurrentFile(tar, &pszFilename);
304 if (RT_SUCCESS(rc))
305 {
306 fFound = !strcmp(pszFilename, RTPathFilename(pszLocation));
307 RTStrFree(pszFilename);
308 if (fFound)
309 break;
310 else
311 {
312 rc = RTTarSeekNextFile(tar);
313 if (RT_FAILURE(rc))
314 break;
315 }
316 }else
317 break;
318 }
319 if (fFound)
320 rc = RTTarFileOpenCurrentFile(tar, &pInt->file, 0, fOpen);
321 }
322 else
323 rc = RTTarFileOpen(tar, &pInt->file, RTPathFilename(pszLocation), fOpen);
324
325 if (RT_FAILURE(rc))
326 RTMemFree(pInt);
327 else
328 *ppInt = pInt;
329
330 return rc;
331}
332
333static int tarCloseCallback(void *pvUser, void *pvStorage)
334{
335 /* Validate input. */
336 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
337 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
338
339 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)pvStorage;
340
341 DEBUG_PRINT_FLOW();
342
343 int rc = RTTarFileClose(pInt->file);
344
345 /* Cleanup */
346 RTMemFree(pInt);
347
348 return rc;
349}
350
351static int tarDeleteCallback(void *pvUser, const char *pcszFilename)
352{
353 /* Validate input. */
354 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
355 AssertPtrReturn(pcszFilename, VERR_INVALID_POINTER);
356
357 DEBUG_PRINT_FLOW();
358
359 return VERR_NOT_IMPLEMENTED;
360}
361
362static int tarMoveCallback(void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned /* fMove */)
363{
364 /* Validate input. */
365 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
366 AssertPtrReturn(pcszSrc, VERR_INVALID_POINTER);
367 AssertPtrReturn(pcszDst, VERR_INVALID_POINTER);
368
369 DEBUG_PRINT_FLOW();
370
371 return VERR_NOT_IMPLEMENTED;
372}
373
374static int tarGetFreeSpaceCallback(void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace)
375{
376 /* Validate input. */
377 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
378 AssertPtrReturn(pcszFilename, VERR_INVALID_POINTER);
379 AssertPtrReturn(pcbFreeSpace, VERR_INVALID_POINTER);
380
381 DEBUG_PRINT_FLOW();
382
383 return VERR_NOT_IMPLEMENTED;
384}
385
386static int tarGetModificationTimeCallback(void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime)
387{
388 /* Validate input. */
389 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
390 AssertPtrReturn(pcszFilename, VERR_INVALID_POINTER);
391 AssertPtrReturn(pModificationTime, VERR_INVALID_POINTER);
392
393 DEBUG_PRINT_FLOW();
394
395 return VERR_NOT_IMPLEMENTED;
396}
397
398static int tarGetSizeCallback(void *pvUser, void *pvStorage, uint64_t *pcbSize)
399{
400 /* Validate input. */
401 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
402 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
403
404 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)pvStorage;
405
406 DEBUG_PRINT_FLOW();
407
408 return RTTarFileGetSize(pInt->file, pcbSize);
409}
410
411static int tarSetSizeCallback(void *pvUser, void *pvStorage, uint64_t cbSize)
412{
413 /* Validate input. */
414 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
415 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
416
417 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)pvStorage;
418
419 DEBUG_PRINT_FLOW();
420
421 return RTTarFileSetSize(pInt->file, cbSize);
422}
423
424static int tarWriteSyncCallback(void *pvUser, void *pvStorage, uint64_t uOffset,
425 const void *pvBuf, size_t cbWrite, size_t *pcbWritten)
426{
427 /* Validate input. */
428 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
429 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
430
431 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)pvStorage;
432
433 DEBUG_PRINT_FLOW();
434
435 return RTTarFileWriteAt(pInt->file, uOffset, pvBuf, cbWrite, pcbWritten);
436}
437
438static int tarReadSyncCallback(void *pvUser, void *pvStorage, uint64_t uOffset,
439 void *pvBuf, size_t cbRead, size_t *pcbRead)
440{
441 /* Validate input. */
442 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
443 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
444
445 PTARSTORAGEINTERNAL pInt = (PTARSTORAGEINTERNAL)pvStorage;
446
447// DEBUG_PRINT_FLOW();
448
449 return RTTarFileReadAt(pInt->file, uOffset, pvBuf, cbRead, pcbRead);
450}
451
452static int tarFlushSyncCallback(void *pvUser, void *pvStorage)
453{
454 /* Validate input. */
455 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
456 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
457
458 DEBUG_PRINT_FLOW();
459
460 return VERR_NOT_IMPLEMENTED;
461}
462
463/******************************************************************************
464 * Internal: RTSha interface
465 ******************************************************************************/
466
467DECLCALLBACK(int) shaCalcWorkerThread(RTTHREAD /* aThread */, void *pvUser)
468{
469 /* Validate input. */
470 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
471
472 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvUser;
473
474 PVDINTERFACEIO pIfIo = VDIfIoGet(pInt->pShaStorage->pVDImageIfaces);
475 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
476
477 int rc = VINF_SUCCESS;
478 bool fLoop = true;
479 while(fLoop)
480 {
481 /* What should we do next? */
482 uint32_t u32Status = ASMAtomicReadU32(&pInt->u32Status);
483// RTPrintf("status: %d\n", u32Status);
484 switch (u32Status)
485 {
486 case STATUS_WAIT:
487 {
488 /* Wait for new work. */
489 rc = RTSemEventWait(pInt->newStatusEvent, 100);
490 if ( RT_FAILURE(rc)
491 && rc != VERR_TIMEOUT)
492 fLoop = false;
493 break;
494 }
495 case STATUS_WRITE:
496 {
497 ASMAtomicCmpXchgU32(&pInt->u32Status, STATUS_WRITING, STATUS_WRITE);
498 size_t cbAvail = RTCircBufUsed(pInt->pCircBuf);
499 size_t cbMemAllRead = 0;
500 /* First loop over all the free memory in the circular
501 * memory buffer (could be turn around at the end). */
502 for(;;)
503 {
504 if ( cbMemAllRead == cbAvail
505 || fLoop == false)
506 break;
507 char *pcBuf;
508 size_t cbMemToRead = cbAvail - cbMemAllRead;
509 size_t cbMemRead = 0;
510 /* Try to acquire all the used space of the circular buffer. */
511 RTCircBufAcquireReadBlock(pInt->pCircBuf, cbMemToRead, (void**)&pcBuf, &cbMemRead);
512 size_t cbAllWritten = 0;
513 /* Second, write as long as used memory is there. The write
514 * method could also split the writes up into to smaller
515 * parts. */
516 for(;;)
517 {
518 if (cbAllWritten == cbMemRead)
519 break;
520 size_t cbToWrite = cbMemRead - cbAllWritten;
521 size_t cbWritten = 0;
522 rc = vdIfIoFileWriteSync(pIfIo, pInt->pvStorage, pInt->cbCurFile, &pcBuf[cbAllWritten], cbToWrite, &cbWritten);
523// RTPrintf ("%lu %lu %lu %Rrc\n", pInt->cbCurFile, cbToRead, cbRead, rc);
524 if (RT_FAILURE(rc))
525 {
526 fLoop = false;
527 break;
528 }
529 cbAllWritten += cbWritten;
530 pInt->cbCurFile += cbWritten;
531 }
532 /* Update the SHA1/SHA256 context with the next data block. */
533 if ( RT_SUCCESS(rc)
534 && pInt->pShaStorage->fCreateDigest)
535 {
536 if (pInt->pShaStorage->fSha256)
537 RTSha256Update(&pInt->ctx.Sha256, pcBuf, cbAllWritten);
538 else
539 RTSha1Update(&pInt->ctx.Sha1, pcBuf, cbAllWritten);
540 }
541 /* Mark the block as empty. */
542 RTCircBufReleaseReadBlock(pInt->pCircBuf, cbAllWritten);
543 cbMemAllRead += cbAllWritten;
544 }
545 /* Reset the thread status and signal the main thread that we
546 * are finished. Use CmpXchg, so we not overwrite other states
547 * which could be signaled in the meantime. */
548 if (ASMAtomicCmpXchgU32(&pInt->u32Status, STATUS_WAIT, STATUS_WRITING))
549 rc = RTSemEventSignal(pInt->workFinishedEvent);
550 break;
551 }
552 case STATUS_READ:
553 {
554 ASMAtomicCmpXchgU32(&pInt->u32Status, STATUS_READING, STATUS_READ);
555 size_t cbAvail = RTCircBufFree(pInt->pCircBuf);
556 size_t cbMemAllWrite = 0;
557 /* First loop over all the available memory in the circular
558 * memory buffer (could be turn around at the end). */
559 for(;;)
560 {
561 if ( cbMemAllWrite == cbAvail
562 || fLoop == false)
563 break;
564 char *pcBuf;
565 size_t cbMemToWrite = cbAvail - cbMemAllWrite;
566 size_t cbMemWrite = 0;
567 /* Try to acquire all the free space of the circular buffer. */
568 RTCircBufAcquireWriteBlock(pInt->pCircBuf, cbMemToWrite, (void**)&pcBuf, &cbMemWrite);
569 /* Second, read as long as we filled all the memory. The
570 * read method could also split the reads up into to
571 * smaller parts. */
572 size_t cbAllRead = 0;
573 for(;;)
574 {
575 if (cbAllRead == cbMemWrite)
576 break;
577 size_t cbToRead = cbMemWrite - cbAllRead;
578 size_t cbRead = 0;
579 rc = vdIfIoFileReadSync(pIfIo, pInt->pvStorage, pInt->cbCurFile, &pcBuf[cbAllRead], cbToRead, &cbRead);
580// RTPrintf ("%lu %lu %lu %Rrc\n", pInt->cbCurFile, cbToRead, cbRead, rc);
581 if (RT_FAILURE(rc))
582 {
583 fLoop = false;
584 break;
585 }
586 /* This indicates end of file. Stop reading. */
587 if (cbRead == 0)
588 {
589 fLoop = false;
590 ASMAtomicWriteBool(&pInt->fEOF, true);
591 break;
592 }
593 cbAllRead += cbRead;
594 pInt->cbCurFile += cbRead;
595 }
596 /* Update the SHA1/SHA256 context with the next data block. */
597 if ( RT_SUCCESS(rc)
598 && pInt->pShaStorage->fCreateDigest)
599 {
600 if (pInt->pShaStorage->fSha256)
601 RTSha256Update(&pInt->ctx.Sha256, pcBuf, cbAllRead);
602 else
603 RTSha1Update(&pInt->ctx.Sha1, pcBuf, cbAllRead);
604 }
605 /* Mark the block as full. */
606 RTCircBufReleaseWriteBlock(pInt->pCircBuf, cbAllRead);
607 cbMemAllWrite += cbAllRead;
608 }
609 /* Reset the thread status and signal the main thread that we
610 * are finished. Use CmpXchg, so we not overwrite other states
611 * which could be signaled in the meantime. */
612 if (ASMAtomicCmpXchgU32(&pInt->u32Status, STATUS_WAIT, STATUS_READING))
613 rc = RTSemEventSignal(pInt->workFinishedEvent);
614 break;
615 }
616 case STATUS_END:
617 {
618 /* End signaled */
619 fLoop = false;
620 break;
621 }
622 }
623 }
624 /* Cleanup any status changes to indicate we are finished. */
625 ASMAtomicWriteU32(&pInt->u32Status, STATUS_END);
626 rc = RTSemEventSignal(pInt->workFinishedEvent);
627 return rc;
628}
629
630DECLINLINE(int) shaSignalManifestThread(PSHASTORAGEINTERNAL pInt, uint32_t uStatus)
631{
632 ASMAtomicWriteU32(&pInt->u32Status, uStatus);
633 return RTSemEventSignal(pInt->newStatusEvent);
634}
635
636DECLINLINE(int) shaWaitForManifestThreadFinished(PSHASTORAGEINTERNAL pInt)
637{
638// RTPrintf("start\n");
639 int rc = VINF_SUCCESS;
640 for(;;)
641 {
642// RTPrintf(" wait\n");
643 uint32_t u32Status = ASMAtomicReadU32(&pInt->u32Status);
644 if (!( u32Status == STATUS_WRITE
645 || u32Status == STATUS_WRITING
646 || u32Status == STATUS_READ
647 || u32Status == STATUS_READING))
648 break;
649 rc = RTSemEventWait(pInt->workFinishedEvent, 100);
650 }
651 if (rc == VERR_TIMEOUT)
652 rc = VINF_SUCCESS;
653 return rc;
654}
655
656DECLINLINE(int) shaFlushCurBuf(PSHASTORAGEINTERNAL pInt)
657{
658 int rc = VINF_SUCCESS;
659 if (pInt->fOpenMode & RTFILE_O_WRITE)
660 {
661 /* Let the write worker thread start immediately. */
662 rc = shaSignalManifestThread(pInt, STATUS_WRITE);
663 if (RT_FAILURE(rc))
664 return rc;
665
666 /* Wait until the write worker thread has finished. */
667 rc = shaWaitForManifestThreadFinished(pInt);
668 }
669
670 return rc;
671}
672
673static int shaOpenCallback(void *pvUser, const char *pszLocation, uint32_t fOpen,
674 PFNVDCOMPLETED pfnCompleted, void **ppInt)
675{
676 /* Validate input. */
677 AssertPtrReturn(pvUser, VERR_INVALID_PARAMETER);
678 AssertPtrReturn(pszLocation, VERR_INVALID_POINTER);
679 AssertPtrNullReturn(pfnCompleted, VERR_INVALID_PARAMETER);
680 AssertPtrReturn(ppInt, VERR_INVALID_POINTER);
681 AssertReturn((fOpen & RTFILE_O_READWRITE) != RTFILE_O_READWRITE, VERR_INVALID_PARAMETER); /* No read/write allowed */
682
683 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
684 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
685 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
686
687 DEBUG_PRINT_FLOW();
688
689 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)RTMemAllocZ(sizeof(SHASTORAGEINTERNAL));
690 if (!pInt)
691 return VERR_NO_MEMORY;
692
693 int rc = VINF_SUCCESS;
694 do
695 {
696 pInt->pfnCompleted = pfnCompleted;
697 pInt->pShaStorage = pShaStorage;
698 pInt->fEOF = false;
699 pInt->fOpenMode = fOpen;
700 pInt->u32Status = STATUS_WAIT;
701
702 /* Circular buffer in the read case. */
703 rc = RTCircBufCreate(&pInt->pCircBuf, _1M * 2);
704 if (RT_FAILURE(rc))
705 break;
706
707 if (fOpen & RTFILE_O_WRITE)
708 {
709 /* The zero buffer is used for appending empty parts at the end of the
710 * file (or our buffer) in setSize or when uOffset in writeSync is
711 * increased in steps bigger than a byte. */
712 pInt->cbZeroBuf = _1K;
713 pInt->pvZeroBuf = RTMemAllocZ(pInt->cbZeroBuf);
714 if (!pInt->pvZeroBuf)
715 {
716 rc = VERR_NO_MEMORY;
717 break;
718 }
719 }
720
721 /* Create an event semaphore to indicate a state change for the worker
722 * thread. */
723 rc = RTSemEventCreate(&pInt->newStatusEvent);
724 if (RT_FAILURE(rc))
725 break;
726 /* Create an event semaphore to indicate a finished calculation of the
727 worker thread. */
728 rc = RTSemEventCreate(&pInt->workFinishedEvent);
729 if (RT_FAILURE(rc))
730 break;
731 /* Create the worker thread. */
732 rc = RTThreadCreate(&pInt->pWorkerThread, shaCalcWorkerThread, pInt, 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, RTTHREADFLAGS_WAITABLE, "SHA-Worker");
733 if (RT_FAILURE(rc))
734 break;
735
736 if (pShaStorage->fCreateDigest)
737 {
738 /* Create a SHA1/SHA256 context the worker thread will work with. */
739 if (pShaStorage->fSha256)
740 RTSha256Init(&pInt->ctx.Sha256);
741 else
742 RTSha1Init(&pInt->ctx.Sha1);
743 }
744
745 /* Open the file. */
746 rc = vdIfIoFileOpen(pIfIo, pszLocation, fOpen, pInt->pfnCompleted,
747 &pInt->pvStorage);
748 if (RT_FAILURE(rc))
749 break;
750
751 if (fOpen & RTFILE_O_READ)
752 {
753 /* Immediately let the worker thread start the reading. */
754 rc = shaSignalManifestThread(pInt, STATUS_READ);
755 }
756 }
757 while(0);
758
759 if (RT_FAILURE(rc))
760 {
761 if (pInt->pWorkerThread)
762 {
763 shaSignalManifestThread(pInt, STATUS_END);
764 RTThreadWait(pInt->pWorkerThread, RT_INDEFINITE_WAIT, 0);
765 }
766 if (pInt->workFinishedEvent)
767 RTSemEventDestroy(pInt->workFinishedEvent);
768 if (pInt->newStatusEvent)
769 RTSemEventDestroy(pInt->newStatusEvent);
770 if (pInt->pCircBuf)
771 RTCircBufDestroy(pInt->pCircBuf);
772 if (pInt->pvZeroBuf)
773 RTMemFree(pInt->pvZeroBuf);
774 RTMemFree(pInt);
775 }
776 else
777 *ppInt = pInt;
778
779 return rc;
780}
781
782static int shaCloseCallback(void *pvUser, void *pvStorage)
783{
784 /* Validate input. */
785 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
786 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
787
788 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
789 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
790 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
791
792 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
793
794 DEBUG_PRINT_FLOW();
795
796 int rc = VINF_SUCCESS;
797
798 /* Make sure all pending writes are flushed */
799 rc = shaFlushCurBuf(pInt);
800
801 if (pInt->pWorkerThread)
802 {
803 /* Signal the worker thread to end himself */
804 rc = shaSignalManifestThread(pInt, STATUS_END);
805 /* Worker thread stopped? */
806 rc = RTThreadWait(pInt->pWorkerThread, RT_INDEFINITE_WAIT, 0);
807 }
808
809 if ( RT_SUCCESS(rc)
810 && pShaStorage->fCreateDigest)
811 {
812 /* Finally calculate & format the SHA1/SHA256 sum */
813 unsigned char auchDig[RTSHA256_HASH_SIZE];
814 char *pszDigest;
815 size_t cbDigest;
816 if (pShaStorage->fSha256)
817 {
818 RTSha256Final(&pInt->ctx.Sha256, auchDig);
819 cbDigest = RTSHA256_DIGEST_LEN;
820 }
821 else
822 {
823 RTSha1Final(&pInt->ctx.Sha1, auchDig);
824 cbDigest = RTSHA1_DIGEST_LEN;
825 }
826 rc = RTStrAllocEx(&pszDigest, cbDigest + 1);
827 if (RT_SUCCESS(rc))
828 {
829 if (pShaStorage->fSha256)
830 rc = RTSha256ToString(auchDig, pszDigest, cbDigest + 1);
831 else
832 rc = RTSha1ToString(auchDig, pszDigest, cbDigest + 1);
833 if (RT_SUCCESS(rc))
834 pShaStorage->strDigest = pszDigest;
835 RTStrFree(pszDigest);
836 }
837 }
838
839 /* Close the file */
840 rc = vdIfIoFileClose(pIfIo, pInt->pvStorage);
841
842// RTPrintf("%lu %lu\n", pInt->calls, pInt->waits);
843
844 /* Cleanup */
845 if (pInt->workFinishedEvent)
846 RTSemEventDestroy(pInt->workFinishedEvent);
847 if (pInt->newStatusEvent)
848 RTSemEventDestroy(pInt->newStatusEvent);
849 if (pInt->pCircBuf)
850 RTCircBufDestroy(pInt->pCircBuf);
851 if (pInt->pvZeroBuf)
852 RTMemFree(pInt->pvZeroBuf);
853 RTMemFree(pInt);
854
855 return rc;
856}
857
858static int shaDeleteCallback(void *pvUser, const char *pcszFilename)
859{
860 /* Validate input. */
861 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
862
863 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
864 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
865 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
866
867 DEBUG_PRINT_FLOW();
868
869 return vdIfIoFileDelete(pIfIo, pcszFilename);
870}
871
872static int shaMoveCallback(void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned fMove)
873{
874 /* Validate input. */
875 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
876
877 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
878 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
879 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
880
881
882 DEBUG_PRINT_FLOW();
883
884 return vdIfIoFileMove(pIfIo, pcszSrc, pcszDst, fMove);
885}
886
887static int shaGetFreeSpaceCallback(void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace)
888{
889 /* Validate input. */
890 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
891
892 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
893 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
894 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
895
896 DEBUG_PRINT_FLOW();
897
898 return vdIfIoFileGetFreeSpace(pIfIo, pcszFilename, pcbFreeSpace);
899}
900
901static int shaGetModificationTimeCallback(void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime)
902{
903 /* Validate input. */
904 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
905
906 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
907 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
908 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
909
910 DEBUG_PRINT_FLOW();
911
912 return vdIfIoFileGetModificationTime(pIfIo, pcszFilename, pModificationTime);
913}
914
915
916static int shaGetSizeCallback(void *pvUser, void *pvStorage, uint64_t *pcbSize)
917{
918 /* Validate input. */
919 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
920 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
921
922 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
923 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
924 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
925
926 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
927
928 DEBUG_PRINT_FLOW();
929
930 uint64_t cbSize;
931 int rc = vdIfIoFileGetSize(pIfIo, pInt->pvStorage, &cbSize);
932 if (RT_FAILURE(rc))
933 return rc;
934
935 *pcbSize = RT_MAX(pInt->cbCurAll, cbSize);
936
937 return VINF_SUCCESS;
938}
939
940static int shaSetSizeCallback(void *pvUser, void *pvStorage, uint64_t cbSize)
941{
942 /* Validate input. */
943 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
944 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
945
946 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
947 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
948 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
949
950 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
951
952 DEBUG_PRINT_FLOW();
953
954 return vdIfIoFileSetSize(pIfIo, pInt->pvStorage, cbSize);
955}
956
957static int shaWriteSyncCallback(void *pvUser, void *pvStorage, uint64_t uOffset,
958 const void *pvBuf, size_t cbWrite, size_t *pcbWritten)
959{
960 /* Validate input. */
961 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
962 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
963
964 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
965 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
966 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
967
968 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
969
970 DEBUG_PRINT_FLOW();
971
972 /* Check that the write is linear */
973 AssertMsgReturn(pInt->cbCurAll <= uOffset, ("Backward seeking is not allowed (uOffset: %7lu cbCurAll: %7lu)!", uOffset, pInt->cbCurAll), VERR_INVALID_PARAMETER);
974
975 int rc = VINF_SUCCESS;
976
977 /* Check if we have to add some free space at the end, before we start the
978 * real write. */
979 if (pInt->cbCurAll < uOffset)
980 {
981 size_t cbSize = (size_t)(uOffset - pInt->cbCurAll);
982 size_t cbAllWritten = 0;
983 for(;;)
984 {
985 /* Finished? */
986 if (cbAllWritten == cbSize)
987 break;
988 size_t cbToWrite = RT_MIN(pInt->cbZeroBuf, cbSize - cbAllWritten);
989 size_t cbWritten = 0;
990 rc = shaWriteSyncCallback(pvUser, pvStorage, pInt->cbCurAll,
991 pInt->pvZeroBuf, cbToWrite, &cbWritten);
992 if (RT_FAILURE(rc))
993 break;
994 cbAllWritten += cbWritten;
995 }
996 if (RT_FAILURE(rc))
997 return rc;
998 }
999// RTPrintf("Write uOffset: %7lu cbWrite: %7lu = %7lu\n", uOffset, cbWrite, uOffset + cbWrite);
1000
1001 size_t cbAllWritten = 0;
1002 for(;;)
1003 {
1004 /* Finished? */
1005 if (cbAllWritten == cbWrite)
1006 break;
1007 size_t cbAvail = RTCircBufFree(pInt->pCircBuf);
1008 if ( cbAvail == 0
1009 && pInt->fEOF)
1010 return VERR_EOF;
1011 /* If there isn't enough free space make sure the worker thread is
1012 * writing some data. */
1013 if ((cbWrite - cbAllWritten) > cbAvail)
1014 {
1015 rc = shaSignalManifestThread(pInt, STATUS_WRITE);
1016 if(RT_FAILURE(rc))
1017 break;
1018 /* If there is _no_ free space available, we have to wait until it is. */
1019 if (cbAvail == 0)
1020 {
1021 rc = shaWaitForManifestThreadFinished(pInt);
1022 if (RT_FAILURE(rc))
1023 break;
1024 cbAvail = RTCircBufFree(pInt->pCircBuf);
1025// RTPrintf("############## wait %lu %lu %lu \n", cbRead, cbAllRead, cbAvail);
1026// pInt->waits++;
1027 }
1028 }
1029 size_t cbToWrite = RT_MIN(cbWrite - cbAllWritten, cbAvail);
1030 char *pcBuf;
1031 size_t cbMemWritten = 0;
1032 /* Acquire a block for writing from our circular buffer. */
1033 RTCircBufAcquireWriteBlock(pInt->pCircBuf, cbToWrite, (void**)&pcBuf, &cbMemWritten);
1034 memcpy(pcBuf, &((char*)pvBuf)[cbAllWritten], cbMemWritten);
1035 /* Mark the block full. */
1036 RTCircBufReleaseWriteBlock(pInt->pCircBuf, cbMemWritten);
1037 cbAllWritten += cbMemWritten;
1038 pInt->cbCurAll += cbMemWritten;
1039 }
1040
1041 if (pcbWritten)
1042 *pcbWritten = cbAllWritten;
1043
1044 /* Signal the thread to write more data in the mean time. */
1045 if ( RT_SUCCESS(rc)
1046 && RTCircBufUsed(pInt->pCircBuf) >= (RTCircBufSize(pInt->pCircBuf) / 2))
1047 rc = shaSignalManifestThread(pInt, STATUS_WRITE);
1048
1049 return rc;
1050}
1051
1052static int shaReadSyncCallback(void *pvUser, void *pvStorage, uint64_t uOffset,
1053 void *pvBuf, size_t cbRead, size_t *pcbRead)
1054{
1055 /* Validate input. */
1056 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
1057 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
1058
1059 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
1060 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
1061 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
1062
1063// DEBUG_PRINT_FLOW();
1064
1065 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
1066
1067 int rc = VINF_SUCCESS;
1068
1069// pInt->calls++;
1070// RTPrintf("Read uOffset: %7lu cbRead: %7lu = %7lu\n", uOffset, cbRead, uOffset + cbRead);
1071
1072 /* Check if we jump forward in the file. If so we have to read the
1073 * remaining stuff in the gap anyway (SHA1/SHA256; streaming). */
1074 if (pInt->cbCurAll < uOffset)
1075 {
1076 rc = shaReadSyncCallback(pvUser, pvStorage, pInt->cbCurAll, 0,
1077 (size_t)(uOffset - pInt->cbCurAll), 0);
1078 if (RT_FAILURE(rc))
1079 return rc;
1080// RTPrintf("Gap Read uOffset: %7lu cbRead: %7lu = %7lu\n", uOffset, cbRead, uOffset + cbRead);
1081 }
1082
1083 size_t cbAllRead = 0;
1084 for(;;)
1085 {
1086 /* Finished? */
1087 if (cbAllRead == cbRead)
1088 break;
1089 size_t cbAvail = RTCircBufUsed(pInt->pCircBuf);
1090 if ( cbAvail == 0
1091 && pInt->fEOF
1092 && !RTCircBufIsWriting(pInt->pCircBuf))
1093 {
1094 break;
1095 }
1096 /* If there isn't enough data make sure the worker thread is fetching
1097 * more. */
1098 if ((cbRead - cbAllRead) > cbAvail)
1099 {
1100 rc = shaSignalManifestThread(pInt, STATUS_READ);
1101 if(RT_FAILURE(rc))
1102 break;
1103 /* If there is _no_ data available, we have to wait until it is. */
1104 if (cbAvail == 0)
1105 {
1106 rc = shaWaitForManifestThreadFinished(pInt);
1107 if (RT_FAILURE(rc))
1108 break;
1109 cbAvail = RTCircBufUsed(pInt->pCircBuf);
1110// RTPrintf("############## wait %lu %lu %lu \n", cbRead, cbAllRead, cbAvail);
1111// pInt->waits++;
1112 }
1113 }
1114 size_t cbToRead = RT_MIN(cbRead - cbAllRead, cbAvail);
1115 char *pcBuf;
1116 size_t cbMemRead = 0;
1117 /* Acquire a block for reading from our circular buffer. */
1118 RTCircBufAcquireReadBlock(pInt->pCircBuf, cbToRead, (void**)&pcBuf, &cbMemRead);
1119 if (pvBuf) /* Make it possible to blind read data (for skipping) */
1120 memcpy(&((char*)pvBuf)[cbAllRead], pcBuf, cbMemRead);
1121 /* Mark the block as empty again. */
1122 RTCircBufReleaseReadBlock(pInt->pCircBuf, cbMemRead);
1123 cbAllRead += cbMemRead;
1124
1125 pInt->cbCurAll += cbMemRead;
1126 }
1127
1128 if (pcbRead)
1129 *pcbRead = cbAllRead;
1130
1131 if (rc == VERR_EOF)
1132 rc = VINF_SUCCESS;
1133
1134 /* Signal the thread to read more data in the mean time. */
1135 if ( RT_SUCCESS(rc)
1136 && RTCircBufFree(pInt->pCircBuf) >= (RTCircBufSize(pInt->pCircBuf) / 2))
1137 rc = shaSignalManifestThread(pInt, STATUS_READ);
1138
1139 return rc;
1140}
1141
1142static int shaFlushSyncCallback(void *pvUser, void *pvStorage)
1143{
1144 /* Validate input. */
1145 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
1146 AssertPtrReturn(pvStorage, VERR_INVALID_POINTER);
1147
1148 PSHASTORAGE pShaStorage = (PSHASTORAGE)pvUser;
1149 PVDINTERFACEIO pIfIo = VDIfIoGet(pShaStorage->pVDImageIfaces);
1150 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
1151
1152 DEBUG_PRINT_FLOW();
1153
1154 PSHASTORAGEINTERNAL pInt = (PSHASTORAGEINTERNAL)pvStorage;
1155
1156 /* Check if there is still something in the buffer. If yes, flush it. */
1157 int rc = shaFlushCurBuf(pInt);
1158 if (RT_FAILURE(rc))
1159 return rc;
1160
1161 return vdIfIoFileFlushSync(pIfIo, pInt->pvStorage);
1162}
1163
1164/******************************************************************************
1165 * Public Functions *
1166 ******************************************************************************/
1167
1168PVDINTERFACEIO ShaCreateInterface()
1169{
1170 PVDINTERFACEIO pCallbacks = (PVDINTERFACEIO)RTMemAllocZ(sizeof(VDINTERFACEIO));
1171 if (!pCallbacks)
1172 return NULL;
1173
1174 pCallbacks->pfnOpen = shaOpenCallback;
1175 pCallbacks->pfnClose = shaCloseCallback;
1176 pCallbacks->pfnDelete = shaDeleteCallback;
1177 pCallbacks->pfnMove = shaMoveCallback;
1178 pCallbacks->pfnGetFreeSpace = shaGetFreeSpaceCallback;
1179 pCallbacks->pfnGetModificationTime = shaGetModificationTimeCallback;
1180 pCallbacks->pfnGetSize = shaGetSizeCallback;
1181 pCallbacks->pfnSetSize = shaSetSizeCallback;
1182 pCallbacks->pfnReadSync = shaReadSyncCallback;
1183 pCallbacks->pfnWriteSync = shaWriteSyncCallback;
1184 pCallbacks->pfnFlushSync = shaFlushSyncCallback;
1185
1186 return pCallbacks;
1187}
1188
1189PVDINTERFACEIO FileCreateInterface()
1190{
1191 PVDINTERFACEIO pCallbacks = (PVDINTERFACEIO)RTMemAllocZ(sizeof(VDINTERFACEIO));
1192 if (!pCallbacks)
1193 return NULL;
1194
1195 pCallbacks->pfnOpen = fileOpenCallback;
1196 pCallbacks->pfnClose = fileCloseCallback;
1197 pCallbacks->pfnDelete = fileDeleteCallback;
1198 pCallbacks->pfnMove = fileMoveCallback;
1199 pCallbacks->pfnGetFreeSpace = fileGetFreeSpaceCallback;
1200 pCallbacks->pfnGetModificationTime = fileGetModificationTimeCallback;
1201 pCallbacks->pfnGetSize = fileGetSizeCallback;
1202 pCallbacks->pfnSetSize = fileSetSizeCallback;
1203 pCallbacks->pfnReadSync = fileReadSyncCallback;
1204 pCallbacks->pfnWriteSync = fileWriteSyncCallback;
1205 pCallbacks->pfnFlushSync = fileFlushSyncCallback;
1206
1207 return pCallbacks;
1208}
1209
1210PVDINTERFACEIO TarCreateInterface()
1211{
1212 PVDINTERFACEIO pCallbacks = (PVDINTERFACEIO)RTMemAllocZ(sizeof(VDINTERFACEIO));
1213 if (!pCallbacks)
1214 return NULL;
1215
1216 pCallbacks->pfnOpen = tarOpenCallback;
1217 pCallbacks->pfnClose = tarCloseCallback;
1218 pCallbacks->pfnDelete = tarDeleteCallback;
1219 pCallbacks->pfnMove = tarMoveCallback;
1220 pCallbacks->pfnGetFreeSpace = tarGetFreeSpaceCallback;
1221 pCallbacks->pfnGetModificationTime = tarGetModificationTimeCallback;
1222 pCallbacks->pfnGetSize = tarGetSizeCallback;
1223 pCallbacks->pfnSetSize = tarSetSizeCallback;
1224 pCallbacks->pfnReadSync = tarReadSyncCallback;
1225 pCallbacks->pfnWriteSync = tarWriteSyncCallback;
1226 pCallbacks->pfnFlushSync = tarFlushSyncCallback;
1227
1228 return pCallbacks;
1229}
1230
1231int ShaReadBuf(const char *pcszFilename, void **ppvBuf, size_t *pcbSize, PVDINTERFACEIO pIfIo, void *pvUser)
1232{
1233 /* Validate input. */
1234 AssertPtrReturn(ppvBuf, VERR_INVALID_POINTER);
1235 AssertPtrReturn(pcbSize, VERR_INVALID_POINTER);
1236 AssertPtrReturn(pIfIo, VERR_INVALID_POINTER);
1237
1238 void *pvStorage;
1239 int rc = pIfIo->pfnOpen(pvUser, pcszFilename,
1240 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE, 0,
1241 &pvStorage);
1242 if (RT_FAILURE(rc))
1243 return rc;
1244
1245 void *pvTmpBuf = 0;
1246 void *pvBuf = 0;
1247 uint64_t cbTmpSize = _1M;
1248 size_t cbAllRead = 0;
1249 do
1250 {
1251 pvTmpBuf = RTMemAlloc(cbTmpSize);
1252 if (!pvTmpBuf)
1253 {
1254 rc = VERR_NO_MEMORY;
1255 break;
1256 }
1257
1258 for(;;)
1259 {
1260 size_t cbRead = 0;
1261 rc = pIfIo->pfnReadSync(pvUser, pvStorage, cbAllRead, pvTmpBuf, cbTmpSize, &cbRead);
1262 if ( RT_FAILURE(rc)
1263 || cbRead == 0)
1264 break;
1265 pvBuf = RTMemRealloc(pvBuf, cbAllRead + cbRead);
1266 if (!pvBuf)
1267 {
1268 rc = VERR_NO_MEMORY;
1269 break;
1270 }
1271 memcpy(&((char*)pvBuf)[cbAllRead], pvTmpBuf, cbRead);
1272 cbAllRead += cbRead;
1273 }
1274 }while(0);
1275
1276 pIfIo->pfnClose(pvUser, pvStorage);
1277
1278 if (rc == VERR_EOF)
1279 rc = VINF_SUCCESS;
1280
1281 if (pvTmpBuf)
1282 RTMemFree(pvTmpBuf);
1283
1284 if (RT_SUCCESS(rc))
1285 {
1286 *ppvBuf = pvBuf;
1287 *pcbSize = cbAllRead;
1288 }else
1289 {
1290 if (pvBuf)
1291 RTMemFree(pvBuf);
1292 }
1293
1294 return rc;
1295}
1296
1297int ShaWriteBuf(const char *pcszFilename, void *pvBuf, size_t cbSize, PVDINTERFACEIO pIfIo, void *pvUser)
1298{
1299 /* Validate input. */
1300 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1301 AssertReturn(cbSize, VERR_INVALID_PARAMETER);
1302 AssertPtrReturn(pIfIo, VERR_INVALID_POINTER);
1303
1304 void *pvStorage;
1305 int rc = pIfIo->pfnOpen(pvUser, pcszFilename,
1306 RTFILE_O_CREATE | RTFILE_O_WRITE | RTFILE_O_DENY_ALL, 0,
1307 &pvStorage);
1308 if (RT_FAILURE(rc))
1309 return rc;
1310
1311 size_t cbAllWritten = 0;
1312 for(;;)
1313 {
1314 if (cbAllWritten >= cbSize)
1315 break;
1316 size_t cbToWrite = cbSize - cbAllWritten;
1317 size_t cbWritten = 0;
1318 rc = pIfIo->pfnWriteSync(pvUser, pvStorage, cbAllWritten, &((char*)pvBuf)[cbAllWritten], cbToWrite, &cbWritten);
1319 if (RT_FAILURE(rc))
1320 break;
1321 cbAllWritten += cbWritten;
1322 }
1323
1324 pIfIo->pfnClose(pvUser, pvStorage);
1325
1326 return rc;
1327}
1328
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette