xref: /aoo4110/main/tools/source/fsys/filecopy.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_tools.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #if defined WNT
28*b1cdbd2cSJim Jagielski #ifndef _SVWIN_H
29*b1cdbd2cSJim Jagielski #include <io.h>
30*b1cdbd2cSJim Jagielski #include <tools/svwin.h>
31*b1cdbd2cSJim Jagielski #endif
32*b1cdbd2cSJim Jagielski 
33*b1cdbd2cSJim Jagielski #elif defined(OS2)
34*b1cdbd2cSJim Jagielski #include <sys/types.h>
35*b1cdbd2cSJim Jagielski #include <sys/stat.h>
36*b1cdbd2cSJim Jagielski #include <fcntl.h>
37*b1cdbd2cSJim Jagielski #include <share.h>
38*b1cdbd2cSJim Jagielski #include <io.h>
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski #elif defined UNX
41*b1cdbd2cSJim Jagielski #include <fcntl.h>
42*b1cdbd2cSJim Jagielski #include <unistd.h>
43*b1cdbd2cSJim Jagielski #include <sys/stat.h>
44*b1cdbd2cSJim Jagielski 
45*b1cdbd2cSJim Jagielski #endif
46*b1cdbd2cSJim Jagielski 
47*b1cdbd2cSJim Jagielski #include <ctype.h>
48*b1cdbd2cSJim Jagielski #include <errno.h>
49*b1cdbd2cSJim Jagielski #include <stdlib.h>
50*b1cdbd2cSJim Jagielski #include <string.h>
51*b1cdbd2cSJim Jagielski 
52*b1cdbd2cSJim Jagielski #include <stdio.h>
53*b1cdbd2cSJim Jagielski #include "comdep.hxx"
54*b1cdbd2cSJim Jagielski #include <tools/fsys.hxx>
55*b1cdbd2cSJim Jagielski #include <tools/stream.hxx>
56*b1cdbd2cSJim Jagielski #include <osl/file.hxx>
57*b1cdbd2cSJim Jagielski 
58*b1cdbd2cSJim Jagielski using namespace ::osl;
59*b1cdbd2cSJim Jagielski 
60*b1cdbd2cSJim Jagielski /*************************************************************************
61*b1cdbd2cSJim Jagielski |*
62*b1cdbd2cSJim Jagielski |*    FileCopier::FileCopier()
63*b1cdbd2cSJim Jagielski |*
64*b1cdbd2cSJim Jagielski |*    Beschreibung      FSYS.SDW
65*b1cdbd2cSJim Jagielski |*    Ersterstellung    MI 13.04.94
66*b1cdbd2cSJim Jagielski |*    Letzte Aenderung  MI 13.04.94
67*b1cdbd2cSJim Jagielski |*
68*b1cdbd2cSJim Jagielski *************************************************************************/
69*b1cdbd2cSJim Jagielski 
FileCopier()70*b1cdbd2cSJim Jagielski FileCopier::FileCopier() :
71*b1cdbd2cSJim Jagielski 
72*b1cdbd2cSJim Jagielski 	nBytesTotal	( 0 ),
73*b1cdbd2cSJim Jagielski 	nBytesCopied( 0 ),
74*b1cdbd2cSJim Jagielski 	nBlockSize	( 4096 ),
75*b1cdbd2cSJim Jagielski 	pImp		( new FileCopier_Impl )
76*b1cdbd2cSJim Jagielski 
77*b1cdbd2cSJim Jagielski {
78*b1cdbd2cSJim Jagielski }
79*b1cdbd2cSJim Jagielski 
80*b1cdbd2cSJim Jagielski // -----------------------------------------------------------------------
81*b1cdbd2cSJim Jagielski 
FileCopier(const DirEntry & rSource,const DirEntry & rTarget)82*b1cdbd2cSJim Jagielski FileCopier::FileCopier( const DirEntry& rSource, const DirEntry& rTarget ) :
83*b1cdbd2cSJim Jagielski 
84*b1cdbd2cSJim Jagielski 	aSource		( rSource ),
85*b1cdbd2cSJim Jagielski 	aTarget		( rTarget ),
86*b1cdbd2cSJim Jagielski 	nBytesTotal	( 0 ),
87*b1cdbd2cSJim Jagielski 	nBytesCopied( 0 ),
88*b1cdbd2cSJim Jagielski 	nBlockSize	( 4096 ),
89*b1cdbd2cSJim Jagielski 	pImp		( new FileCopier_Impl )
90*b1cdbd2cSJim Jagielski 
91*b1cdbd2cSJim Jagielski {
92*b1cdbd2cSJim Jagielski }
93*b1cdbd2cSJim Jagielski 
94*b1cdbd2cSJim Jagielski // -----------------------------------------------------------------------
95*b1cdbd2cSJim Jagielski 
FileCopier(const FileCopier & rCopier)96*b1cdbd2cSJim Jagielski FileCopier::FileCopier( const FileCopier& rCopier ) :
97*b1cdbd2cSJim Jagielski 
98*b1cdbd2cSJim Jagielski 	aSource			( rCopier.aSource ),
99*b1cdbd2cSJim Jagielski 	aTarget			( rCopier.aTarget ),
100*b1cdbd2cSJim Jagielski 	nBytesTotal		( 0 ),
101*b1cdbd2cSJim Jagielski 	nBytesCopied	( 0 ),
102*b1cdbd2cSJim Jagielski 	aProgressLink	( rCopier.aProgressLink ),
103*b1cdbd2cSJim Jagielski 	nBlockSize		( 4096 ),
104*b1cdbd2cSJim Jagielski 	pImp			( new FileCopier_Impl )
105*b1cdbd2cSJim Jagielski 
106*b1cdbd2cSJim Jagielski {
107*b1cdbd2cSJim Jagielski }
108*b1cdbd2cSJim Jagielski 
109*b1cdbd2cSJim Jagielski /*************************************************************************
110*b1cdbd2cSJim Jagielski |*
111*b1cdbd2cSJim Jagielski |*    FileCopier::~FileCopier()
112*b1cdbd2cSJim Jagielski |*
113*b1cdbd2cSJim Jagielski |*    Beschreibung      FSYS.SDW
114*b1cdbd2cSJim Jagielski |*    Ersterstellung    MI 13.04.94
115*b1cdbd2cSJim Jagielski |*    Letzte Aenderung  MI 13.04.94
116*b1cdbd2cSJim Jagielski |*
117*b1cdbd2cSJim Jagielski *************************************************************************/
118*b1cdbd2cSJim Jagielski 
~FileCopier()119*b1cdbd2cSJim Jagielski FileCopier::~FileCopier()
120*b1cdbd2cSJim Jagielski {
121*b1cdbd2cSJim Jagielski 	delete pImp;
122*b1cdbd2cSJim Jagielski }
123*b1cdbd2cSJim Jagielski 
124*b1cdbd2cSJim Jagielski /*************************************************************************
125*b1cdbd2cSJim Jagielski |*
126*b1cdbd2cSJim Jagielski |*    FileCopier::operator =()
127*b1cdbd2cSJim Jagielski |*
128*b1cdbd2cSJim Jagielski |*    Beschreibung      FSYS.SDW
129*b1cdbd2cSJim Jagielski |*    Ersterstellung    MI 13.04.94
130*b1cdbd2cSJim Jagielski |*    Letzte Aenderung  MI 13.04.94
131*b1cdbd2cSJim Jagielski |*
132*b1cdbd2cSJim Jagielski *************************************************************************/
133*b1cdbd2cSJim Jagielski 
operator =(const FileCopier & rCopier)134*b1cdbd2cSJim Jagielski FileCopier& FileCopier::operator = ( const FileCopier &rCopier )
135*b1cdbd2cSJim Jagielski {
136*b1cdbd2cSJim Jagielski 	aSource = rCopier.aSource;
137*b1cdbd2cSJim Jagielski 	aTarget = rCopier.aTarget;
138*b1cdbd2cSJim Jagielski 	nBytesTotal = rCopier.nBytesTotal;
139*b1cdbd2cSJim Jagielski 	nBytesCopied = rCopier.nBytesCopied;
140*b1cdbd2cSJim Jagielski 	nBytesCopied = rCopier.nBytesCopied;
141*b1cdbd2cSJim Jagielski 	nBlockSize = rCopier.nBlockSize;
142*b1cdbd2cSJim Jagielski 	aProgressLink = rCopier.aProgressLink;
143*b1cdbd2cSJim Jagielski 	*pImp = *(rCopier.pImp);
144*b1cdbd2cSJim Jagielski 	return *this;
145*b1cdbd2cSJim Jagielski }
146*b1cdbd2cSJim Jagielski 
147*b1cdbd2cSJim Jagielski /*************************************************************************
148*b1cdbd2cSJim Jagielski |*
149*b1cdbd2cSJim Jagielski |*    FileCopier::Progress()
150*b1cdbd2cSJim Jagielski |*
151*b1cdbd2cSJim Jagielski |*    Beschreibung      FSYS.SDW
152*b1cdbd2cSJim Jagielski |*    Ersterstellung    MI 13.04.94
153*b1cdbd2cSJim Jagielski |*    Letzte Aenderung  MI 13.04.94
154*b1cdbd2cSJim Jagielski |*
155*b1cdbd2cSJim Jagielski *************************************************************************/
156*b1cdbd2cSJim Jagielski 
Progress()157*b1cdbd2cSJim Jagielski sal_Bool FileCopier::Progress()
158*b1cdbd2cSJim Jagielski {
159*b1cdbd2cSJim Jagielski 	if ( !aProgressLink )
160*b1cdbd2cSJim Jagielski 		return sal_True;
161*b1cdbd2cSJim Jagielski 	else
162*b1cdbd2cSJim Jagielski 	{
163*b1cdbd2cSJim Jagielski 		if ( aProgressLink.Call( this ) )
164*b1cdbd2cSJim Jagielski 			return sal_True;
165*b1cdbd2cSJim Jagielski 		return ( 0 == Error( ERRCODE_ABORT, 0, 0 ) );
166*b1cdbd2cSJim Jagielski 	}
167*b1cdbd2cSJim Jagielski }
168*b1cdbd2cSJim Jagielski 
169*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
170*b1cdbd2cSJim Jagielski 
Error(ErrCode eErr,const DirEntry * pSource,const DirEntry * pTarget)171*b1cdbd2cSJim Jagielski ErrCode FileCopier::Error( ErrCode eErr, const DirEntry* pSource, const DirEntry* pTarget )
172*b1cdbd2cSJim Jagielski {
173*b1cdbd2cSJim Jagielski 	// kein Fehler oder kein ErrorHandler?
174*b1cdbd2cSJim Jagielski 	if ( !eErr || !pImp->aErrorLink )
175*b1cdbd2cSJim Jagielski 		// => Error beibehalten
176*b1cdbd2cSJim Jagielski 		return eErr;
177*b1cdbd2cSJim Jagielski 
178*b1cdbd2cSJim Jagielski 	// sonst gesetzten ErrorHandler fragen
179*b1cdbd2cSJim Jagielski 	pImp->pErrSource = pSource;
180*b1cdbd2cSJim Jagielski 	pImp->pErrTarget = pTarget;
181*b1cdbd2cSJim Jagielski 	pImp->eErr = eErr;
182*b1cdbd2cSJim Jagielski 	ErrCode eRet = (ErrCode) pImp->aErrorLink.Call( this );
183*b1cdbd2cSJim Jagielski 	pImp->pErrSource = 0;
184*b1cdbd2cSJim Jagielski 	pImp->pErrTarget = 0;
185*b1cdbd2cSJim Jagielski 	return eRet;
186*b1cdbd2cSJim Jagielski }
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
189*b1cdbd2cSJim Jagielski 
GetErrorSource() const190*b1cdbd2cSJim Jagielski const DirEntry* FileCopier::GetErrorSource() const
191*b1cdbd2cSJim Jagielski {
192*b1cdbd2cSJim Jagielski 	return pImp->pErrSource;
193*b1cdbd2cSJim Jagielski }
194*b1cdbd2cSJim Jagielski 
195*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
196*b1cdbd2cSJim Jagielski 
GetErrorTarget() const197*b1cdbd2cSJim Jagielski const DirEntry* FileCopier::GetErrorTarget() const
198*b1cdbd2cSJim Jagielski {
199*b1cdbd2cSJim Jagielski 	return pImp->pErrTarget;
200*b1cdbd2cSJim Jagielski }
201*b1cdbd2cSJim Jagielski 
202*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
203*b1cdbd2cSJim Jagielski 
GetError() const204*b1cdbd2cSJim Jagielski ErrCode FileCopier::GetError() const
205*b1cdbd2cSJim Jagielski {
206*b1cdbd2cSJim Jagielski 	return pImp->eErr;
207*b1cdbd2cSJim Jagielski }
208*b1cdbd2cSJim Jagielski 
209*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
210*b1cdbd2cSJim Jagielski 
SetErrorHdl(const Link & rLink)211*b1cdbd2cSJim Jagielski void FileCopier::SetErrorHdl( const Link &rLink )
212*b1cdbd2cSJim Jagielski {
213*b1cdbd2cSJim Jagielski 	pImp->aErrorLink = rLink;
214*b1cdbd2cSJim Jagielski }
215*b1cdbd2cSJim Jagielski 
216*b1cdbd2cSJim Jagielski //---------------------------------------------------------------------------
217*b1cdbd2cSJim Jagielski 
GetErrorHdl() const218*b1cdbd2cSJim Jagielski const Link& FileCopier::GetErrorHdl() const
219*b1cdbd2cSJim Jagielski {
220*b1cdbd2cSJim Jagielski 	return pImp->aErrorLink ;
221*b1cdbd2cSJim Jagielski }
222*b1cdbd2cSJim Jagielski 
223*b1cdbd2cSJim Jagielski /*************************************************************************
224*b1cdbd2cSJim Jagielski |*
225*b1cdbd2cSJim Jagielski |*    FileCopier::Execute()
226*b1cdbd2cSJim Jagielski |*
227*b1cdbd2cSJim Jagielski |*    Beschreibung      FSYS.SDW
228*b1cdbd2cSJim Jagielski |*    Ersterstellung    MI 13.04.94
229*b1cdbd2cSJim Jagielski |*    Letzte Aenderung  PB 16.06.00
230*b1cdbd2cSJim Jagielski |*
231*b1cdbd2cSJim Jagielski *************************************************************************/
232*b1cdbd2cSJim Jagielski 
DoCopy_Impl(const DirEntry & rSource,const DirEntry & rTarget)233*b1cdbd2cSJim Jagielski FSysError FileCopier::DoCopy_Impl(
234*b1cdbd2cSJim Jagielski     const DirEntry &rSource, const DirEntry &rTarget )
235*b1cdbd2cSJim Jagielski {
236*b1cdbd2cSJim Jagielski 	FSysError eRet = FSYS_ERR_OK;
237*b1cdbd2cSJim Jagielski 	ErrCode eWarn = FSYS_ERR_OK;
238*b1cdbd2cSJim Jagielski 
239*b1cdbd2cSJim Jagielski 	// HPFS->FAT?
240*b1cdbd2cSJim Jagielski 	FSysPathStyle eSourceStyle = DirEntry::GetPathStyle( rSource.ImpGetTopPtr()->GetName() );
241*b1cdbd2cSJim Jagielski 	FSysPathStyle eTargetStyle = DirEntry::GetPathStyle( rTarget.ImpGetTopPtr()->GetName() );
242*b1cdbd2cSJim Jagielski 	sal_Bool bMakeShortNames = ( eSourceStyle == FSYS_STYLE_HPFS && eTargetStyle == FSYS_STYLE_FAT );
243*b1cdbd2cSJim Jagielski 
244*b1cdbd2cSJim Jagielski 	// Zieldateiname ggf. kuerzen
245*b1cdbd2cSJim Jagielski 	DirEntry aTgt;
246*b1cdbd2cSJim Jagielski 	if ( bMakeShortNames )
247*b1cdbd2cSJim Jagielski 	{
248*b1cdbd2cSJim Jagielski 		aTgt = rTarget.GetPath();
249*b1cdbd2cSJim Jagielski 		aTgt.MakeShortName( rTarget.GetName() );
250*b1cdbd2cSJim Jagielski 	}
251*b1cdbd2cSJim Jagielski 	else
252*b1cdbd2cSJim Jagielski 		aTgt = rTarget;
253*b1cdbd2cSJim Jagielski 
254*b1cdbd2cSJim Jagielski 	// kein Move wenn Namen gekuerzt werden muessten
255*b1cdbd2cSJim Jagielski 	if ( bMakeShortNames && FSYS_ACTION_MOVE == ( pImp->nActions & FSYS_ACTION_MOVE ) && aTgt != rTarget )
256*b1cdbd2cSJim Jagielski 		return ERRCODE_IO_NAMETOOLONG;
257*b1cdbd2cSJim Jagielski 
258*b1cdbd2cSJim Jagielski 	// source is directory?
259*b1cdbd2cSJim Jagielski 	FileStat aSourceFileStat( rSource );
260*b1cdbd2cSJim Jagielski 	if ( aSourceFileStat.IsKind( FSYS_KIND_DIR ) )
261*b1cdbd2cSJim Jagielski 	{
262*b1cdbd2cSJim Jagielski #ifdef OS2
263*b1cdbd2cSJim Jagielski 		CHAR szSource[CCHMAXPATHCOMP];
264*b1cdbd2cSJim Jagielski 		HOBJECT hSourceObject;
265*b1cdbd2cSJim Jagielski 
266*b1cdbd2cSJim Jagielski 		strcpy(szSource, ByteString(rSource.GetFull(), osl_getThreadTextEncoding()).GetBuffer());
267*b1cdbd2cSJim Jagielski 		hSourceObject = WinQueryObject(szSource);
268*b1cdbd2cSJim Jagielski 
269*b1cdbd2cSJim Jagielski 		if ( hSourceObject )
270*b1cdbd2cSJim Jagielski 		{
271*b1cdbd2cSJim Jagielski 			PSZ  pszSourceName;
272*b1cdbd2cSJim Jagielski 			PSZ  pszTargetName;
273*b1cdbd2cSJim Jagielski 			CHAR szTarget[CCHMAXPATHCOMP];
274*b1cdbd2cSJim Jagielski 			HOBJECT hTargetObject;
275*b1cdbd2cSJim Jagielski 			HOBJECT hReturn = NULLHANDLE;
276*b1cdbd2cSJim Jagielski 
277*b1cdbd2cSJim Jagielski 			strcpy(szTarget, ByteString(rTarget.GetFull(), osl_getThreadTextEncoding()).GetBuffer());
278*b1cdbd2cSJim Jagielski 			pszTargetName = strrchr(szTarget, '\\');
279*b1cdbd2cSJim Jagielski 			pszSourceName = strrchr(szSource, '\\');
280*b1cdbd2cSJim Jagielski 
281*b1cdbd2cSJim Jagielski 			hTargetObject = WinQueryObject(szTarget);
282*b1cdbd2cSJim Jagielski 
283*b1cdbd2cSJim Jagielski 			if ( hTargetObject )
284*b1cdbd2cSJim Jagielski 				WinDestroyObject(hTargetObject);
285*b1cdbd2cSJim Jagielski 
286*b1cdbd2cSJim Jagielski 			if ( pszTargetName && pszSourceName )
287*b1cdbd2cSJim Jagielski 			{
288*b1cdbd2cSJim Jagielski 				*pszTargetName = '\0';
289*b1cdbd2cSJim Jagielski 				pszSourceName++;
290*b1cdbd2cSJim Jagielski 				pszTargetName++;
291*b1cdbd2cSJim Jagielski 
292*b1cdbd2cSJim Jagielski 				if(strcmp(pszSourceName, pszTargetName) == 0)
293*b1cdbd2cSJim Jagielski 				{
294*b1cdbd2cSJim Jagielski 					hTargetObject = WinQueryObject(szTarget);
295*b1cdbd2cSJim Jagielski 
296*b1cdbd2cSJim Jagielski 					if(pImp->nActions & FSYS_ACTION_MOVE)
297*b1cdbd2cSJim Jagielski 					{
298*b1cdbd2cSJim Jagielski 						hReturn = WinMoveObject(hSourceObject, hTargetObject, 0);
299*b1cdbd2cSJim Jagielski 					}
300*b1cdbd2cSJim Jagielski 					else
301*b1cdbd2cSJim Jagielski 					{
302*b1cdbd2cSJim Jagielski 						hReturn = WinCopyObject(hSourceObject, hTargetObject, 0);
303*b1cdbd2cSJim Jagielski 					}
304*b1cdbd2cSJim Jagielski 					if ( bMakeShortNames && aTarget.Exists() )
305*b1cdbd2cSJim Jagielski 						aTarget.Kill();
306*b1cdbd2cSJim Jagielski 					return hReturn ? FSYS_ERR_OK : FSYS_ERR_UNKNOWN;
307*b1cdbd2cSJim Jagielski 				}
308*b1cdbd2cSJim Jagielski 			}
309*b1cdbd2cSJim Jagielski 		}
310*b1cdbd2cSJim Jagielski #endif
311*b1cdbd2cSJim Jagielski 		// recursive copy
312*b1cdbd2cSJim Jagielski 		eRet = Error( aTgt.MakeDir() ? FSYS_ERR_OK : FSYS_ERR_UNKNOWN, 0, &aTgt );
313*b1cdbd2cSJim Jagielski 		Dir aSourceDir( rSource, FSYS_KIND_DIR|FSYS_KIND_FILE );
314*b1cdbd2cSJim Jagielski 		for ( sal_uInt16 n = 0; ERRCODE_TOERROR(eRet) == FSYS_ERR_OK && n < aSourceDir.Count(); ++n )
315*b1cdbd2cSJim Jagielski 		{
316*b1cdbd2cSJim Jagielski 			const DirEntry &rSubSource = aSourceDir[n];
317*b1cdbd2cSJim Jagielski 			DirEntryFlag eFlag = rSubSource.GetFlag();
318*b1cdbd2cSJim Jagielski 			if ( eFlag != FSYS_FLAG_CURRENT && eFlag != FSYS_FLAG_PARENT )
319*b1cdbd2cSJim Jagielski 			{
320*b1cdbd2cSJim Jagielski 				DirEntry aSubTarget( aTgt );
321*b1cdbd2cSJim Jagielski 				aSubTarget += rSubSource.GetName();
322*b1cdbd2cSJim Jagielski 				eRet = DoCopy_Impl( rSubSource, aSubTarget );
323*b1cdbd2cSJim Jagielski 				if ( eRet && !eWarn )
324*b1cdbd2cSJim Jagielski 				eWarn = eRet;
325*b1cdbd2cSJim Jagielski 			}
326*b1cdbd2cSJim Jagielski 		}
327*b1cdbd2cSJim Jagielski 	}
328*b1cdbd2cSJim Jagielski 	else if ( aSourceFileStat.IsKind(FSYS_KIND_FILE) )
329*b1cdbd2cSJim Jagielski 	{
330*b1cdbd2cSJim Jagielski 		if ( ( FSYS_ACTION_KEEP_EXISTING == ( pImp->nActions & FSYS_ACTION_KEEP_EXISTING ) ) &&
331*b1cdbd2cSJim Jagielski 			 aTgt.Exists() )
332*b1cdbd2cSJim Jagielski 		{
333*b1cdbd2cSJim Jagielski 			// Do not overwrite existing file in target folder.
334*b1cdbd2cSJim Jagielski 			return ERRCODE_NONE;
335*b1cdbd2cSJim Jagielski 		}
336*b1cdbd2cSJim Jagielski 
337*b1cdbd2cSJim Jagielski 		// copy file
338*b1cdbd2cSJim Jagielski 		nBytesCopied = 0;
339*b1cdbd2cSJim Jagielski 		nBytesTotal = FileStat( rSource ).GetSize();
340*b1cdbd2cSJim Jagielski 
341*b1cdbd2cSJim Jagielski 		::rtl::OUString aFileName;
342*b1cdbd2cSJim Jagielski 		FileBase::getFileURLFromSystemPath( ::rtl::OUString(rSource.GetFull()), aFileName );
343*b1cdbd2cSJim Jagielski 		SvFileStream aSrc( aFileName, STREAM_READ|STREAM_NOCREATE|STREAM_SHARE_DENYNONE );
344*b1cdbd2cSJim Jagielski 
345*b1cdbd2cSJim Jagielski 		if ( !aSrc.GetError() )
346*b1cdbd2cSJim Jagielski 		{
347*b1cdbd2cSJim Jagielski #ifdef UNX
348*b1cdbd2cSJim Jagielski 			struct stat buf;
349*b1cdbd2cSJim Jagielski 			if ( fstat( aSrc.GetFileHandle(), &buf ) == -1 )
350*b1cdbd2cSJim Jagielski 				eRet = Error( FSYS_ERR_ACCESSDENIED, 0, &aTgt );
351*b1cdbd2cSJim Jagielski #endif
352*b1cdbd2cSJim Jagielski 			::rtl::OUString aTargetFileName;
353*b1cdbd2cSJim Jagielski 			FileBase::getFileURLFromSystemPath( ::rtl::OUString(aTgt.GetFull()), aTargetFileName );
354*b1cdbd2cSJim Jagielski 
355*b1cdbd2cSJim Jagielski 			SvFileStream aTargetStream( aTargetFileName, STREAM_WRITE | STREAM_TRUNC | STREAM_SHARE_DENYWRITE );
356*b1cdbd2cSJim Jagielski 			if ( !aTargetStream.GetError() )
357*b1cdbd2cSJim Jagielski 			{
358*b1cdbd2cSJim Jagielski #ifdef UNX
359*b1cdbd2cSJim Jagielski 				if ( fchmod( aTargetStream.GetFileHandle(), buf.st_mode ) == -1 )
360*b1cdbd2cSJim Jagielski 					eRet = Error( FSYS_ERR_ACCESSDENIED, 0, &aTgt );
361*b1cdbd2cSJim Jagielski #endif
362*b1cdbd2cSJim Jagielski 				size_t nAllocSize = 0, nSize = 0;
363*b1cdbd2cSJim Jagielski 				char *pBuf = 0;
364*b1cdbd2cSJim Jagielski 				while ( Progress() && nSize == nAllocSize && eRet == FSYS_ERR_OK )
365*b1cdbd2cSJim Jagielski 				{
366*b1cdbd2cSJim Jagielski 					// adjust the block-size
367*b1cdbd2cSJim Jagielski 					if ( nBlockSize > nAllocSize )
368*b1cdbd2cSJim Jagielski 					{
369*b1cdbd2cSJim Jagielski 						delete[] pBuf;
370*b1cdbd2cSJim Jagielski 						nAllocSize = nBlockSize;
371*b1cdbd2cSJim Jagielski 						pBuf = new char[nAllocSize];
372*b1cdbd2cSJim Jagielski 					}
373*b1cdbd2cSJim Jagielski 
374*b1cdbd2cSJim Jagielski 					// copy one block
375*b1cdbd2cSJim Jagielski 					nSize = aSrc.Read( pBuf, nBlockSize );
376*b1cdbd2cSJim Jagielski 					aTargetStream.Write( pBuf, nSize );
377*b1cdbd2cSJim Jagielski 					if ( aTargetStream.GetError() )
378*b1cdbd2cSJim Jagielski 						eRet = Error( aTargetStream.GetError(), 0, &aTgt );
379*b1cdbd2cSJim Jagielski 
380*b1cdbd2cSJim Jagielski 					// adjust counters
381*b1cdbd2cSJim Jagielski 					nBytesCopied += nSize;
382*b1cdbd2cSJim Jagielski 					if ( nBytesCopied > nBytesTotal )
383*b1cdbd2cSJim Jagielski 						nBytesTotal = nBytesCopied;
384*b1cdbd2cSJim Jagielski 				}
385*b1cdbd2cSJim Jagielski 				delete[] pBuf;
386*b1cdbd2cSJim Jagielski 			}
387*b1cdbd2cSJim Jagielski 			else
388*b1cdbd2cSJim Jagielski 				eRet = Error( aTargetStream.GetError(), 0, &aTgt );
389*b1cdbd2cSJim Jagielski 
390*b1cdbd2cSJim Jagielski             // unvollstaendiges File wieder loeschen
391*b1cdbd2cSJim Jagielski             aTargetStream.Close();
392*b1cdbd2cSJim Jagielski 
393*b1cdbd2cSJim Jagielski 			if ( nBytesCopied != nBytesTotal )
394*b1cdbd2cSJim Jagielski 			{
395*b1cdbd2cSJim Jagielski 				aTgt.Kill();
396*b1cdbd2cSJim Jagielski 			}
397*b1cdbd2cSJim Jagielski 		}
398*b1cdbd2cSJim Jagielski 		else
399*b1cdbd2cSJim Jagielski 			eRet = Error( aSrc.GetError(), &rSource, 0 );
400*b1cdbd2cSJim Jagielski 	}
401*b1cdbd2cSJim Jagielski 	else if ( aSourceFileStat.IsKind(FSYS_KIND_NONE) )
402*b1cdbd2cSJim Jagielski 		eRet = Error( ERRCODE_IO_NOTEXISTS, &rSource, 0 );
403*b1cdbd2cSJim Jagielski 	else
404*b1cdbd2cSJim Jagielski 		eRet = Error( ERRCODE_IO_NOTSUPPORTED, &rSource, 0 );
405*b1cdbd2cSJim Jagielski 
406*b1cdbd2cSJim Jagielski #ifdef WNT
407*b1cdbd2cSJim Jagielski 	// Set LastWriteTime and Attributes of the target identical with the source
408*b1cdbd2cSJim Jagielski 
409*b1cdbd2cSJim Jagielski 	if ( FSYS_ERR_OK == ERRCODE_TOERROR(eRet) )
410*b1cdbd2cSJim Jagielski 	{
411*b1cdbd2cSJim Jagielski 		WIN32_FIND_DATA	fdSource;
412*b1cdbd2cSJim Jagielski 		ByteString aFullSource(aSource.GetFull(), osl_getThreadTextEncoding());
413*b1cdbd2cSJim Jagielski 		ByteString aFullTarget(aTgt.GetFull(), osl_getThreadTextEncoding());
414*b1cdbd2cSJim Jagielski 		HANDLE	hFind = FindFirstFile( aFullSource.GetBuffer() , &fdSource );
415*b1cdbd2cSJim Jagielski 		if ( hFind != INVALID_HANDLE_VALUE )
416*b1cdbd2cSJim Jagielski 		{
417*b1cdbd2cSJim Jagielski 			FindClose( hFind );
418*b1cdbd2cSJim Jagielski 
419*b1cdbd2cSJim Jagielski 			HANDLE hFile = CreateFile( aFullTarget.GetBuffer(), GENERIC_WRITE,
420*b1cdbd2cSJim Jagielski 									   FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
421*b1cdbd2cSJim Jagielski 									   OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
422*b1cdbd2cSJim Jagielski 
423*b1cdbd2cSJim Jagielski 			if ( hFile != INVALID_HANDLE_VALUE )
424*b1cdbd2cSJim Jagielski 			{
425*b1cdbd2cSJim Jagielski 				SetFileTime( hFile, NULL, NULL, &fdSource.ftLastWriteTime );
426*b1cdbd2cSJim Jagielski 				CloseHandle( hFile );
427*b1cdbd2cSJim Jagielski 			}
428*b1cdbd2cSJim Jagielski 
429*b1cdbd2cSJim Jagielski 			SetFileAttributes( aFullTarget.GetBuffer(), fdSource.dwFileAttributes );
430*b1cdbd2cSJim Jagielski 		}
431*b1cdbd2cSJim Jagielski 	}
432*b1cdbd2cSJim Jagielski #endif
433*b1cdbd2cSJim Jagielski 	// bei Move ggf. das File/Dir loeschen
434*b1cdbd2cSJim Jagielski 	if ( FSYS_ERR_OK == ERRCODE_TOERROR(eRet) && ( pImp->nActions  & FSYS_ACTION_MOVE ) )
435*b1cdbd2cSJim Jagielski 	{
436*b1cdbd2cSJim Jagielski 		ErrCode eKillErr = Error( rSource.Kill() | ERRCODE_WARNING_MASK, &rSource, 0 );
437*b1cdbd2cSJim Jagielski 		if ( eKillErr != ERRCODE_WARNING_MASK )
438*b1cdbd2cSJim Jagielski 		{
439*b1cdbd2cSJim Jagielski 			if ( rSource.Exists() )
440*b1cdbd2cSJim Jagielski 				// loeschen ging nicht => dann die Kopie wieder loeschen
441*b1cdbd2cSJim Jagielski 				aTgt.Kill( pImp->nActions );
442*b1cdbd2cSJim Jagielski 			if ( !eWarn )
443*b1cdbd2cSJim Jagielski 				eWarn = eKillErr;
444*b1cdbd2cSJim Jagielski 		}
445*b1cdbd2cSJim Jagielski 	}
446*b1cdbd2cSJim Jagielski 
447*b1cdbd2cSJim Jagielski 	return !eRet ? eWarn : eRet;
448*b1cdbd2cSJim Jagielski }
449*b1cdbd2cSJim Jagielski 
450*b1cdbd2cSJim Jagielski // -----------------------------------------------------------------------
451*b1cdbd2cSJim Jagielski 
Execute(FSysAction nActions)452*b1cdbd2cSJim Jagielski FSysError FileCopier::Execute( FSysAction nActions )
453*b1cdbd2cSJim Jagielski {
454*b1cdbd2cSJim Jagielski 	return ExecuteExact( nActions );
455*b1cdbd2cSJim Jagielski }
456*b1cdbd2cSJim Jagielski 
457*b1cdbd2cSJim Jagielski // -----------------------------------------------------------------------
458*b1cdbd2cSJim Jagielski 
ExecuteExact(FSysAction nActions,FSysExact eExact)459*b1cdbd2cSJim Jagielski FSysError FileCopier::ExecuteExact( FSysAction nActions, FSysExact eExact )
460*b1cdbd2cSJim Jagielski {
461*b1cdbd2cSJim Jagielski 	DirEntry aAbsSource = DirEntry( aSource);
462*b1cdbd2cSJim Jagielski 	DirEntry aAbsTarget = DirEntry( aTarget );
463*b1cdbd2cSJim Jagielski 	pImp->nActions = nActions;
464*b1cdbd2cSJim Jagielski 
465*b1cdbd2cSJim Jagielski 	// check if both pathes are accessible and source and target are different
466*b1cdbd2cSJim Jagielski 	if ( !aAbsTarget.ToAbs() || !aAbsSource.ToAbs() || aAbsTarget == aAbsSource )
467*b1cdbd2cSJim Jagielski 		return FSYS_ERR_ACCESSDENIED;
468*b1cdbd2cSJim Jagielski 
469*b1cdbd2cSJim Jagielski 	// check if copy would be endless recursive into itself
470*b1cdbd2cSJim Jagielski 	if ( FSYS_ACTION_RECURSIVE == ( nActions & FSYS_ACTION_RECURSIVE ) &&
471*b1cdbd2cSJim Jagielski 		 aAbsSource.Contains( aAbsTarget ) )
472*b1cdbd2cSJim Jagielski 		return ERRCODE_IO_RECURSIVE;
473*b1cdbd2cSJim Jagielski 
474*b1cdbd2cSJim Jagielski 	// target is directory?
475*b1cdbd2cSJim Jagielski 	if ( eExact == FSYS_NOTEXACT &&
476*b1cdbd2cSJim Jagielski 		 FileStat( aAbsTarget ).IsKind(FSYS_KIND_DIR) && FileStat( aAbsSource ).IsKind(FSYS_KIND_FILE) )
477*b1cdbd2cSJim Jagielski 		// append name of source
478*b1cdbd2cSJim Jagielski 		aAbsTarget += aSource.GetName();
479*b1cdbd2cSJim Jagielski 
480*b1cdbd2cSJim Jagielski 	// recursive copy
481*b1cdbd2cSJim Jagielski 	return DoCopy_Impl( aAbsSource, aAbsTarget );
482*b1cdbd2cSJim Jagielski }
483