VirtualBox

source: vbox/trunk/src/VBox/Runtime/generic/RTFileCopyPartEx-generic.cpp@ 78382

Last change on this file since 78382 was 77832, checked in by vboxsync, 6 years ago

IPRT: Spealiced RTFileCopyPartEx and associated prep/cleanup functions for linux copy_file_range(). bugref:9172

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.3 KB
Line 
1/* $Id: RTFileCopyPartEx-generic.cpp 77832 2019-03-21 20:41:23Z vboxsync $ */
2/** @file
3 * IPRT - RTFileCopyPartEx, generic implementation.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/file.h>
32#include "internal/iprt.h"
33
34#include <iprt/alloca.h>
35#include <iprt/assert.h>
36#include <iprt/err.h>
37#include <iprt/mem.h>
38
39
40#ifndef IPRT_FALLBACK_VERSION
41RTDECL(int) RTFileCopyPartPrep(PRTFILECOPYPARTBUFSTATE pBufState, uint64_t cbToCopy)
42#else
43static int rtFileCopyPartPrepFallback(PRTFILECOPYPARTBUFSTATE pBufState, uint64_t cbToCopy)
44#endif
45{
46 /*
47 * Allocate a fitting buffer.
48 *
49 * We're a little careful with using RTMemPageAlloc here as it does require
50 * kernel calls on some hosts. If we could be sure the heap was able to
51 * handle it most of the time, we would be a lot more aggressive.
52 */
53 size_t cbBuf = 0;
54 void *pvBuf = NULL;
55 if ( cbToCopy >= _512K
56 && (pvBuf = RTMemPageAlloc(cbBuf = _128K)) != NULL)
57 pBufState->iAllocType = 2;
58 else if ( cbToCopy >= _128K
59 && (pvBuf = RTMemTmpAlloc(cbBuf = _128K)) != NULL)
60 pBufState->iAllocType = 1;
61 else if ( cbToCopy < _128K
62 && cbToCopy >= _4K
63 && (pvBuf = RTMemTmpAlloc(cbBuf = RT_ALIGN_Z(cbToCopy, 32))) != NULL)
64 pBufState->iAllocType = 1;
65 else if ( cbToCopy >= _4K
66 && (pvBuf = RTMemTmpAlloc(cbBuf = _4K)) != NULL)
67 pBufState->iAllocType = 1;
68 else
69 pBufState->iAllocType = 0;
70 pBufState->pbBuf = (uint8_t *)pvBuf;
71 pBufState->cbBuf = cbBuf;
72 pBufState->uMagic = RTFILECOPYPARTBUFSTATE_MAGIC;
73 return VINF_SUCCESS;
74}
75
76
77#ifndef IPRT_FALLBACK_VERSION
78RTDECL(void) RTFileCopyPartCleanup(PRTFILECOPYPARTBUFSTATE pBufState)
79#else
80static void rtFileCopyPartCleanupFallback(PRTFILECOPYPARTBUFSTATE pBufState)
81#endif
82{
83 AssertReturnVoid(pBufState->uMagic == RTFILECOPYPARTBUFSTATE_MAGIC);
84 if (pBufState->iAllocType == 1)
85 RTMemTmpFree(pBufState->pbBuf);
86 else if (pBufState->iAllocType == 2)
87 RTMemPageFree(pBufState->pbBuf, pBufState->cbBuf);
88 pBufState->pbBuf = NULL;
89 pBufState->cbBuf = 0;
90 pBufState->uMagic = ~RTFILECOPYPARTBUFSTATE_MAGIC;
91}
92
93
94#ifndef IPRT_FALLBACK_VERSION
95RTDECL(int) RTFileCopyPartEx(RTFILE hFileSrc, RTFOFF offSrc, RTFILE hFileDst, RTFOFF offDst, uint64_t cbToCopy,
96 uint32_t fFlags, PRTFILECOPYPARTBUFSTATE pBufState, uint64_t *pcbCopied)
97#else
98static int rtFileCopyPartExFallback(RTFILE hFileSrc, RTFOFF offSrc, RTFILE hFileDst, RTFOFF offDst, uint64_t cbToCopy,
99 uint32_t fFlags, PRTFILECOPYPARTBUFSTATE pBufState, uint64_t *pcbCopied)
100#endif
101{
102 /*
103 * Validate input.
104 */
105 if (pcbCopied)
106 *pcbCopied = 0;
107 AssertReturn(offSrc >= 0, VERR_NEGATIVE_SEEK);
108 AssertReturn(offDst >= 0, VERR_NEGATIVE_SEEK);
109 AssertReturn(!fFlags, VERR_INVALID_FLAGS);
110 AssertReturn(pBufState->uMagic == RTFILECOPYPARTBUFSTATE_MAGIC, VERR_INVALID_FLAGS);
111
112 /*
113 * If nothing to copy, return right away.
114 */
115 if (!cbToCopy)
116 return VINF_SUCCESS;
117
118 if (pBufState->iAllocType == 0)
119 {
120 pBufState->cbBuf = (size_t)RT_MIN(_4K, cbToCopy);
121 pBufState->pbBuf = (uint8_t *)alloca(pBufState->cbBuf);
122 AssertReturn(pBufState->pbBuf, VERR_NO_MEMORY);
123 }
124
125 /*
126 * Do the copying.
127 */
128 uint64_t cbCopied = 0;
129 int rc = VINF_SUCCESS;
130 do
131 {
132 size_t cbThisCopy = (size_t)RT_MIN(cbToCopy - cbCopied, pBufState->cbBuf);
133 size_t cbActual = 0;
134 rc = RTFileReadAt(hFileSrc, offSrc + cbCopied, pBufState->pbBuf, cbThisCopy, &cbActual);
135 if (RT_FAILURE(rc))
136 break;
137 if (cbActual == 0)
138 {
139 if (!pcbCopied)
140 rc = VERR_EOF;
141 break;
142 }
143
144 rc = RTFileWriteAt(hFileDst, offDst + cbCopied, pBufState->pbBuf, cbActual, NULL);
145 if (RT_FAILURE(rc))
146 break;
147
148 cbCopied += cbActual;
149 } while (cbCopied < cbToCopy);
150
151 if (pcbCopied)
152 *pcbCopied = cbCopied;
153
154 return rc;
155}
156
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