xref: /trunk/main/sal/osl/unx/file_misc.cxx (revision 9204bf8294f34a46abdfeceee27c48e7b7b2ef12)
187d2adbcSAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
387d2adbcSAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
487d2adbcSAndrew Rist  * or more contributor license agreements.  See the NOTICE file
587d2adbcSAndrew Rist  * distributed with this work for additional information
687d2adbcSAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
787d2adbcSAndrew Rist  * to you under the Apache License, Version 2.0 (the
887d2adbcSAndrew Rist  * "License"); you may not use this file except in compliance
987d2adbcSAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
1187d2adbcSAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
1387d2adbcSAndrew Rist  * Unless required by applicable law or agreed to in writing,
1487d2adbcSAndrew Rist  * software distributed under the License is distributed on an
1587d2adbcSAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
1687d2adbcSAndrew Rist  * KIND, either express or implied.  See the License for the
1787d2adbcSAndrew Rist  * specific language governing permissions and limitations
1887d2adbcSAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
2087d2adbcSAndrew Rist  *************************************************************/
2187d2adbcSAndrew Rist 
2287d2adbcSAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir #include "osl/file.hxx"
25cdf0e10cSrcweir 
26cdf0e10cSrcweir #include "osl/diagnose.h"
27cdf0e10cSrcweir #include "osl/thread.h"
28cdf0e10cSrcweir #include <osl/signal.h>
29cdf0e10cSrcweir #include "rtl/alloc.h"
30cdf0e10cSrcweir 
31cdf0e10cSrcweir #include "system.h"
32cdf0e10cSrcweir #include "file_impl.hxx"
33cdf0e10cSrcweir #include "file_error_transl.h"
34cdf0e10cSrcweir #include "file_path_helper.hxx"
35cdf0e10cSrcweir #include "file_url.h"
36cdf0e10cSrcweir #include "uunxapi.hxx"
37cdf0e10cSrcweir 
38cdf0e10cSrcweir #include <sys/types.h>
39cdf0e10cSrcweir #include <errno.h>
40cdf0e10cSrcweir #include <dirent.h>
41cdf0e10cSrcweir #include <limits.h>
42cdf0e10cSrcweir #include <stdio.h>
43cdf0e10cSrcweir #include <string.h>
44cdf0e10cSrcweir #include <unistd.h>
45cdf0e10cSrcweir #include <sys/stat.h>
46cdf0e10cSrcweir #include <sys/mman.h>
47cdf0e10cSrcweir 
48cdf0e10cSrcweir #include <algorithm>
49cdf0e10cSrcweir 
50cdf0e10cSrcweir /************************************************************************
51cdf0e10cSrcweir  *   ToDo
52cdf0e10cSrcweir  *
53cdf0e10cSrcweir  *   - Fix: check for corresponding struct sizes in exported functions
54cdf0e10cSrcweir  *   - check size/use of oslDirectory
55cdf0e10cSrcweir  *   - check size/use of oslDirectoryItem
56cdf0e10cSrcweir  ***********************************************************************/
57cdf0e10cSrcweir /******************************************************************************
58cdf0e10cSrcweir  * Data Type Definition
59cdf0e10cSrcweir  ******************************************************************************/
60cdf0e10cSrcweir 
61cdf0e10cSrcweir typedef struct
62cdf0e10cSrcweir {
63cdf0e10cSrcweir     rtl_uString* ustrPath; /* holds native directory path */
64cdf0e10cSrcweir     DIR*         pDirStruct;
65cdf0e10cSrcweir } oslDirectoryImpl;
66cdf0e10cSrcweir 
67cdf0e10cSrcweir #if 0
68cdf0e10cSrcweir /* FIXME: reintroducing this may save some extra bytes per Item */
69cdf0e10cSrcweir typedef struct
70cdf0e10cSrcweir {
71cdf0e10cSrcweir     rtl_uString* ustrFileName; /* holds native file name */
72cdf0e10cSrcweir     rtl_uString* ustrDirPath; /* holds native dir path */
73cdf0e10cSrcweir     sal_uInt32   RefCount;
74cdf0e10cSrcweir } oslDirectoryItemImpl;
75cdf0e10cSrcweir #endif
76cdf0e10cSrcweir 
DirectoryItem_Impl(rtl_uString * ustrFilePath,unsigned char DType)77cdf0e10cSrcweir DirectoryItem_Impl::DirectoryItem_Impl(
78cdf0e10cSrcweir     rtl_uString * ustrFilePath, unsigned char DType)
79cdf0e10cSrcweir     : m_RefCount     (1),
80cdf0e10cSrcweir       m_ustrFilePath (ustrFilePath),
81cdf0e10cSrcweir       m_DType        (DType)
82cdf0e10cSrcweir {
83cdf0e10cSrcweir     if (m_ustrFilePath != 0)
84cdf0e10cSrcweir         rtl_uString_acquire(m_ustrFilePath);
85cdf0e10cSrcweir }
~DirectoryItem_Impl()86cdf0e10cSrcweir DirectoryItem_Impl::~DirectoryItem_Impl()
87cdf0e10cSrcweir {
88cdf0e10cSrcweir     if (m_ustrFilePath != 0)
89cdf0e10cSrcweir         rtl_uString_release(m_ustrFilePath);
90cdf0e10cSrcweir }
91cdf0e10cSrcweir 
operator new(size_t n)92cdf0e10cSrcweir void * DirectoryItem_Impl::operator new(size_t n)
93cdf0e10cSrcweir {
94cdf0e10cSrcweir     return rtl_allocateMemory(n);
95cdf0e10cSrcweir }
operator delete(void * p,size_t)96cdf0e10cSrcweir void DirectoryItem_Impl::operator delete(void * p, size_t)
97cdf0e10cSrcweir {
98cdf0e10cSrcweir     rtl_freeMemory(p);
99cdf0e10cSrcweir }
100cdf0e10cSrcweir 
acquire()101cdf0e10cSrcweir void DirectoryItem_Impl::acquire()
102cdf0e10cSrcweir {
103cdf0e10cSrcweir     ++m_RefCount;
104cdf0e10cSrcweir }
release()105cdf0e10cSrcweir void DirectoryItem_Impl::release()
106cdf0e10cSrcweir {
107cdf0e10cSrcweir     if (0 == --m_RefCount)
108cdf0e10cSrcweir         delete this;
109cdf0e10cSrcweir }
110cdf0e10cSrcweir 
getFileType() const111cdf0e10cSrcweir oslFileType DirectoryItem_Impl::getFileType() const
112cdf0e10cSrcweir {
113cdf0e10cSrcweir     switch (m_DType)
114cdf0e10cSrcweir     {
115cdf0e10cSrcweir #ifdef _DIRENT_HAVE_D_TYPE
116cdf0e10cSrcweir         case DT_LNK:
117cdf0e10cSrcweir             return osl_File_Type_Link;
118cdf0e10cSrcweir         case DT_DIR:
119cdf0e10cSrcweir             return osl_File_Type_Directory;
120cdf0e10cSrcweir         case DT_REG:
121cdf0e10cSrcweir             return osl_File_Type_Regular;
122cdf0e10cSrcweir         case DT_FIFO:
123cdf0e10cSrcweir             return osl_File_Type_Fifo;
124cdf0e10cSrcweir         case DT_SOCK:
125cdf0e10cSrcweir             return osl_File_Type_Socket;
126cdf0e10cSrcweir         case DT_CHR:
127cdf0e10cSrcweir         case DT_BLK:
128cdf0e10cSrcweir             return osl_File_Type_Special;
129cdf0e10cSrcweir #endif /* _DIRENT_HAVE_D_TYPE */
130cdf0e10cSrcweir         default:
131cdf0e10cSrcweir             break;
132cdf0e10cSrcweir     }
133cdf0e10cSrcweir     return osl_File_Type_Unknown;
134cdf0e10cSrcweir }
135cdf0e10cSrcweir 
136cdf0e10cSrcweir /******************************************************************************
137cdf0e10cSrcweir  * C-String Function Declarations
138cdf0e10cSrcweir  *****************************************************************************/
139cdf0e10cSrcweir 
140cdf0e10cSrcweir static oslFileError osl_psz_createDirectory(const sal_Char* pszPath);
141cdf0e10cSrcweir static oslFileError osl_psz_removeDirectory(const sal_Char* pszPath);
142cdf0e10cSrcweir 
143cdf0e10cSrcweir /*******************************************************************
144cdf0e10cSrcweir  * osl_openDirectory
145cdf0e10cSrcweir  ******************************************************************/
146cdf0e10cSrcweir 
osl_openDirectory(rtl_uString * ustrDirectoryURL,oslDirectory * pDirectory)147cdf0e10cSrcweir oslFileError SAL_CALL osl_openDirectory(rtl_uString* ustrDirectoryURL, oslDirectory* pDirectory)
148cdf0e10cSrcweir {
149cdf0e10cSrcweir     rtl_uString* ustrSystemPath = NULL;
150cdf0e10cSrcweir     oslFileError eRet;
151cdf0e10cSrcweir 
152cdf0e10cSrcweir     char path[PATH_MAX];
153cdf0e10cSrcweir 
154cdf0e10cSrcweir     if ((0 == ustrDirectoryURL) || (0 == ustrDirectoryURL->length) || (0 == pDirectory))
155cdf0e10cSrcweir         return osl_File_E_INVAL;
156cdf0e10cSrcweir 
157cdf0e10cSrcweir     /* convert file URL to system path */
158cdf0e10cSrcweir     eRet = osl_getSystemPathFromFileURL_Ex(ustrDirectoryURL, &ustrSystemPath, sal_False);
159cdf0e10cSrcweir 
160cdf0e10cSrcweir     if( osl_File_E_None != eRet )
161cdf0e10cSrcweir         return eRet;
162cdf0e10cSrcweir 
163cdf0e10cSrcweir     osl_systemPathRemoveSeparator(ustrSystemPath);
164cdf0e10cSrcweir 
165cdf0e10cSrcweir     /* convert unicode path to text */
166cdf0e10cSrcweir     if ( UnicodeToText( path, PATH_MAX, ustrSystemPath->buffer, ustrSystemPath->length )
167cdf0e10cSrcweir #ifdef MACOSX
168cdf0e10cSrcweir      && macxp_resolveAlias( path, PATH_MAX ) == 0
169cdf0e10cSrcweir #endif /* MACOSX */
170cdf0e10cSrcweir      )
171cdf0e10cSrcweir     {
172cdf0e10cSrcweir         /* open directory */
173cdf0e10cSrcweir         DIR *pdir = opendir( path );
174cdf0e10cSrcweir 
175cdf0e10cSrcweir         if( pdir )
176cdf0e10cSrcweir         {
177cdf0e10cSrcweir             /* create and initialize impl structure */
178cdf0e10cSrcweir             oslDirectoryImpl* pDirImpl = (oslDirectoryImpl*) rtl_allocateMemory( sizeof(oslDirectoryImpl) );
179cdf0e10cSrcweir 
180cdf0e10cSrcweir             if( pDirImpl )
181cdf0e10cSrcweir             {
182cdf0e10cSrcweir                 pDirImpl->pDirStruct = pdir;
183cdf0e10cSrcweir                 pDirImpl->ustrPath = ustrSystemPath;
184cdf0e10cSrcweir 
185cdf0e10cSrcweir                 *pDirectory = (oslDirectory) pDirImpl;
186cdf0e10cSrcweir                 return osl_File_E_None;
187cdf0e10cSrcweir             }
188cdf0e10cSrcweir             else
189cdf0e10cSrcweir             {
190cdf0e10cSrcweir                 errno = ENOMEM;
191cdf0e10cSrcweir                 closedir( pdir );
192cdf0e10cSrcweir             }
193cdf0e10cSrcweir         }
194cdf0e10cSrcweir         else
195cdf0e10cSrcweir         {
196cdf0e10cSrcweir #ifdef DEBUG_OSL_FILE
197cdf0e10cSrcweir             perror ("osl_openDirectory"); fprintf (stderr, path);
198cdf0e10cSrcweir #endif
199cdf0e10cSrcweir         }
200cdf0e10cSrcweir     }
201cdf0e10cSrcweir 
202cdf0e10cSrcweir     rtl_uString_release( ustrSystemPath );
203cdf0e10cSrcweir 
204cdf0e10cSrcweir     return oslTranslateFileError(OSL_FET_ERROR, errno);
205cdf0e10cSrcweir }
206cdf0e10cSrcweir 
207cdf0e10cSrcweir /****************************************************************************/
208cdf0e10cSrcweir /* osl_closeDirectory */
209cdf0e10cSrcweir /****************************************************************************/
210cdf0e10cSrcweir 
osl_closeDirectory(oslDirectory Directory)211cdf0e10cSrcweir oslFileError SAL_CALL osl_closeDirectory( oslDirectory Directory )
212cdf0e10cSrcweir {
213cdf0e10cSrcweir     oslDirectoryImpl* pDirImpl = (oslDirectoryImpl*) Directory;
214cdf0e10cSrcweir     oslFileError err = osl_File_E_None;
215cdf0e10cSrcweir 
216cdf0e10cSrcweir     OSL_ASSERT( Directory );
217cdf0e10cSrcweir 
218cdf0e10cSrcweir     if( NULL == pDirImpl )
219cdf0e10cSrcweir         return osl_File_E_INVAL;
220cdf0e10cSrcweir 
221cdf0e10cSrcweir     /* close directory */
222cdf0e10cSrcweir     if( closedir( pDirImpl->pDirStruct ) )
223cdf0e10cSrcweir     {
224cdf0e10cSrcweir         err = oslTranslateFileError(OSL_FET_ERROR, errno);
225cdf0e10cSrcweir     }
226cdf0e10cSrcweir 
227cdf0e10cSrcweir     /* cleanup members */
228cdf0e10cSrcweir     rtl_uString_release( pDirImpl->ustrPath );
229cdf0e10cSrcweir 
230cdf0e10cSrcweir     rtl_freeMemory( pDirImpl );
231cdf0e10cSrcweir 
232cdf0e10cSrcweir     return err;
233cdf0e10cSrcweir }
234cdf0e10cSrcweir 
235cdf0e10cSrcweir /**********************************************
236cdf0e10cSrcweir  * osl_readdir_impl_
237cdf0e10cSrcweir  *
238cdf0e10cSrcweir  * readdir wrapper, filters out "." and ".."
239cdf0e10cSrcweir  * on request
240cdf0e10cSrcweir  *********************************************/
241cdf0e10cSrcweir 
osl_readdir_impl_(DIR * pdir,sal_Bool bFilterLocalAndParentDir)242cdf0e10cSrcweir static struct dirent* osl_readdir_impl_(DIR* pdir, sal_Bool bFilterLocalAndParentDir)
243cdf0e10cSrcweir {
244cdf0e10cSrcweir     struct dirent* pdirent;
245cdf0e10cSrcweir 
246cdf0e10cSrcweir     while ((pdirent = readdir(pdir)) != NULL)
247cdf0e10cSrcweir     {
248cdf0e10cSrcweir         if (bFilterLocalAndParentDir &&
249cdf0e10cSrcweir             ((0 == strcmp(pdirent->d_name, ".")) || (0 == strcmp(pdirent->d_name, ".."))))
250cdf0e10cSrcweir             continue;
251cdf0e10cSrcweir         else
252cdf0e10cSrcweir             break;
253cdf0e10cSrcweir     }
254cdf0e10cSrcweir 
255cdf0e10cSrcweir     return pdirent;
256cdf0e10cSrcweir }
257cdf0e10cSrcweir 
258cdf0e10cSrcweir /****************************************************************************
259cdf0e10cSrcweir  * osl_getNextDirectoryItem
260cdf0e10cSrcweir  ***************************************************************************/
261cdf0e10cSrcweir 
osl_getNextDirectoryItem(oslDirectory Directory,oslDirectoryItem * pItem,sal_uInt32)262cdf0e10cSrcweir oslFileError SAL_CALL osl_getNextDirectoryItem(oslDirectory Directory, oslDirectoryItem* pItem, sal_uInt32 /*uHint*/)
263cdf0e10cSrcweir {
264cdf0e10cSrcweir     oslDirectoryImpl* pDirImpl     = (oslDirectoryImpl*)Directory;
265cdf0e10cSrcweir     rtl_uString*      ustrFileName = NULL;
266cdf0e10cSrcweir     rtl_uString*      ustrFilePath = NULL;
267cdf0e10cSrcweir     struct dirent*    pEntry;
268cdf0e10cSrcweir 
269cdf0e10cSrcweir     OSL_ASSERT(Directory);
270cdf0e10cSrcweir     OSL_ASSERT(pItem);
271cdf0e10cSrcweir 
272cdf0e10cSrcweir     if ((NULL == Directory) || (NULL == pItem))
273cdf0e10cSrcweir         return osl_File_E_INVAL;
274cdf0e10cSrcweir 
275cdf0e10cSrcweir     pEntry = osl_readdir_impl_(pDirImpl->pDirStruct, sal_True);
276cdf0e10cSrcweir 
277cdf0e10cSrcweir     if (NULL == pEntry)
278cdf0e10cSrcweir         return osl_File_E_NOENT;
279cdf0e10cSrcweir 
280cdf0e10cSrcweir 
281cdf0e10cSrcweir #if defined(MACOSX)
282cdf0e10cSrcweir 
283cdf0e10cSrcweir     // convert decomposed filename to precomposed unicode
284cdf0e10cSrcweir     char composed_name[BUFSIZ];
285cdf0e10cSrcweir     CFMutableStringRef strRef = CFStringCreateMutable (NULL, 0 );
286cdf0e10cSrcweir     CFStringAppendCString( strRef, pEntry->d_name, kCFStringEncodingUTF8 ); //UTF8 is default on Mac OSX
287cdf0e10cSrcweir     CFStringNormalize( strRef, kCFStringNormalizationFormC );
288cdf0e10cSrcweir     CFStringGetCString( strRef, composed_name, BUFSIZ, kCFStringEncodingUTF8 );
289cdf0e10cSrcweir     CFRelease( strRef );
290cdf0e10cSrcweir     rtl_string2UString( &ustrFileName, composed_name, strlen( composed_name),
291cdf0e10cSrcweir     osl_getThreadTextEncoding(), OSTRING_TO_OUSTRING_CVTFLAGS );
292cdf0e10cSrcweir 
293cdf0e10cSrcweir #else // not MACOSX
294cdf0e10cSrcweir     /* convert file name to unicode */
295cdf0e10cSrcweir     rtl_string2UString( &ustrFileName, pEntry->d_name, strlen( pEntry->d_name ),
296cdf0e10cSrcweir         osl_getThreadTextEncoding(), OSTRING_TO_OUSTRING_CVTFLAGS );
297cdf0e10cSrcweir     OSL_ASSERT(ustrFileName != 0);
298cdf0e10cSrcweir 
299cdf0e10cSrcweir #endif
300cdf0e10cSrcweir 
301cdf0e10cSrcweir     osl_systemPathMakeAbsolutePath(pDirImpl->ustrPath, ustrFileName, &ustrFilePath);
302cdf0e10cSrcweir     rtl_uString_release( ustrFileName );
303cdf0e10cSrcweir 
304cdf0e10cSrcweir     DirectoryItem_Impl * pImpl = static_cast< DirectoryItem_Impl* >(*pItem);
305cdf0e10cSrcweir     if (0 != pImpl)
306cdf0e10cSrcweir     {
307cdf0e10cSrcweir         pImpl->release(), pImpl = 0;
308cdf0e10cSrcweir     }
309cdf0e10cSrcweir #ifdef _DIRENT_HAVE_D_TYPE
310cdf0e10cSrcweir     pImpl = new DirectoryItem_Impl(ustrFilePath, pEntry->d_type);
311cdf0e10cSrcweir #else
312cdf0e10cSrcweir     pImpl = new DirectoryItem_Impl(ustrFilePath);
313cdf0e10cSrcweir #endif /* _DIRENT_HAVE_D_TYPE */
314cdf0e10cSrcweir     *pItem = pImpl;
315cdf0e10cSrcweir     rtl_uString_release( ustrFilePath );
316cdf0e10cSrcweir 
317cdf0e10cSrcweir     return osl_File_E_None;
318cdf0e10cSrcweir }
319cdf0e10cSrcweir 
320cdf0e10cSrcweir /****************************************************************************/
321cdf0e10cSrcweir /* osl_getDirectoryItem */
322cdf0e10cSrcweir /****************************************************************************/
323cdf0e10cSrcweir 
osl_getDirectoryItem(rtl_uString * ustrFileURL,oslDirectoryItem * pItem)324cdf0e10cSrcweir oslFileError SAL_CALL osl_getDirectoryItem( rtl_uString* ustrFileURL, oslDirectoryItem* pItem )
325cdf0e10cSrcweir {
326cdf0e10cSrcweir     rtl_uString* ustrSystemPath = NULL;
327cdf0e10cSrcweir     oslFileError osl_error      = osl_File_E_INVAL;
328cdf0e10cSrcweir 
329cdf0e10cSrcweir     OSL_ASSERT((0 != ustrFileURL) && (0 != pItem));
330cdf0e10cSrcweir     if ((0 == ustrFileURL) || (0 == ustrFileURL->length) || (0 == pItem))
331cdf0e10cSrcweir         return osl_File_E_INVAL;
332cdf0e10cSrcweir 
333cdf0e10cSrcweir     osl_error = osl_getSystemPathFromFileURL_Ex(ustrFileURL, &ustrSystemPath, sal_False);
334cdf0e10cSrcweir     if (osl_File_E_None != osl_error)
335cdf0e10cSrcweir         return osl_error;
336cdf0e10cSrcweir 
337cdf0e10cSrcweir     osl_systemPathRemoveSeparator(ustrSystemPath);
338cdf0e10cSrcweir 
339cdf0e10cSrcweir     if (-1 == access_u(ustrSystemPath, F_OK))
340cdf0e10cSrcweir     {
341cdf0e10cSrcweir         osl_error = oslTranslateFileError(OSL_FET_ERROR, errno);
342cdf0e10cSrcweir     }
343cdf0e10cSrcweir     else
344cdf0e10cSrcweir     {
345cdf0e10cSrcweir         *pItem = new DirectoryItem_Impl(ustrSystemPath);
346cdf0e10cSrcweir     }
347cdf0e10cSrcweir     rtl_uString_release(ustrSystemPath);
348cdf0e10cSrcweir 
349cdf0e10cSrcweir     return osl_error;
350cdf0e10cSrcweir }
351cdf0e10cSrcweir 
352cdf0e10cSrcweir 
353cdf0e10cSrcweir /****************************************************************************/
354cdf0e10cSrcweir /* osl_acquireDirectoryItem */
355cdf0e10cSrcweir /****************************************************************************/
356cdf0e10cSrcweir 
osl_acquireDirectoryItem(oslDirectoryItem Item)357cdf0e10cSrcweir oslFileError SAL_CALL osl_acquireDirectoryItem( oslDirectoryItem Item )
358cdf0e10cSrcweir {
359cdf0e10cSrcweir     DirectoryItem_Impl * pImpl = static_cast< DirectoryItem_Impl* >(Item);
360cdf0e10cSrcweir     if (0 == pImpl)
361cdf0e10cSrcweir         return osl_File_E_INVAL;
362cdf0e10cSrcweir 
363cdf0e10cSrcweir     pImpl->acquire();
364cdf0e10cSrcweir     return osl_File_E_None;
365cdf0e10cSrcweir }
366cdf0e10cSrcweir 
367cdf0e10cSrcweir /****************************************************************************/
368cdf0e10cSrcweir /* osl_releaseDirectoryItem */
369cdf0e10cSrcweir /****************************************************************************/
370cdf0e10cSrcweir 
osl_releaseDirectoryItem(oslDirectoryItem Item)371cdf0e10cSrcweir oslFileError SAL_CALL osl_releaseDirectoryItem( oslDirectoryItem Item )
372cdf0e10cSrcweir {
373cdf0e10cSrcweir     DirectoryItem_Impl * pImpl = static_cast< DirectoryItem_Impl* >(Item);
374cdf0e10cSrcweir     if (0 == pImpl)
375cdf0e10cSrcweir         return osl_File_E_INVAL;
376cdf0e10cSrcweir 
377cdf0e10cSrcweir     pImpl->release();
378cdf0e10cSrcweir     return osl_File_E_None;
379cdf0e10cSrcweir }
380cdf0e10cSrcweir 
381cdf0e10cSrcweir /****************************************************************************/
382cdf0e10cSrcweir /* osl_createDirectory */
383cdf0e10cSrcweir /****************************************************************************/
384cdf0e10cSrcweir 
osl_createDirectory(rtl_uString * ustrDirectoryURL)385cdf0e10cSrcweir oslFileError SAL_CALL osl_createDirectory( rtl_uString* ustrDirectoryURL )
386cdf0e10cSrcweir {
387cdf0e10cSrcweir     char path[PATH_MAX];
388cdf0e10cSrcweir     oslFileError eRet;
389cdf0e10cSrcweir 
390cdf0e10cSrcweir     OSL_ASSERT( ustrDirectoryURL );
391cdf0e10cSrcweir 
392cdf0e10cSrcweir     /* convert directory url to system path */
393cdf0e10cSrcweir     eRet = FileURLToPath( path, PATH_MAX, ustrDirectoryURL );
394cdf0e10cSrcweir     if( eRet != osl_File_E_None )
395cdf0e10cSrcweir         return eRet;
396cdf0e10cSrcweir 
397cdf0e10cSrcweir #ifdef MACOSX
398cdf0e10cSrcweir     if ( macxp_resolveAlias( path, PATH_MAX ) != 0 )
399cdf0e10cSrcweir         return oslTranslateFileError( OSL_FET_ERROR, errno );
400cdf0e10cSrcweir #endif/* MACOSX */
401cdf0e10cSrcweir 
402cdf0e10cSrcweir     return osl_psz_createDirectory( path );
403cdf0e10cSrcweir }
404cdf0e10cSrcweir 
405cdf0e10cSrcweir /****************************************************************************/
406cdf0e10cSrcweir /* osl_removeDirectory */
407cdf0e10cSrcweir /****************************************************************************/
408cdf0e10cSrcweir 
osl_removeDirectory(rtl_uString * ustrDirectoryURL)409cdf0e10cSrcweir oslFileError SAL_CALL osl_removeDirectory( rtl_uString* ustrDirectoryURL )
410cdf0e10cSrcweir {
411cdf0e10cSrcweir     char path[PATH_MAX];
412cdf0e10cSrcweir     oslFileError eRet;
413cdf0e10cSrcweir 
414cdf0e10cSrcweir     OSL_ASSERT( ustrDirectoryURL );
415cdf0e10cSrcweir 
416cdf0e10cSrcweir     /* convert directory url to system path */
417cdf0e10cSrcweir     eRet = FileURLToPath( path, PATH_MAX, ustrDirectoryURL );
418cdf0e10cSrcweir     if( eRet != osl_File_E_None )
419cdf0e10cSrcweir         return eRet;
420cdf0e10cSrcweir 
421cdf0e10cSrcweir #ifdef MACOSX
422cdf0e10cSrcweir     if ( macxp_resolveAlias( path, PATH_MAX ) != 0 )
423cdf0e10cSrcweir         return oslTranslateFileError( OSL_FET_ERROR, errno );
424cdf0e10cSrcweir #endif/* MACOSX */
425cdf0e10cSrcweir 
426cdf0e10cSrcweir     return osl_psz_removeDirectory( path );
427cdf0e10cSrcweir }
428cdf0e10cSrcweir 
429cdf0e10cSrcweir /*****************************************
430cdf0e10cSrcweir  * osl_psz_createDirectory
431cdf0e10cSrcweir  ****************************************/
432cdf0e10cSrcweir 
osl_psz_createDirectory(const sal_Char * pszPath)433cdf0e10cSrcweir static oslFileError osl_psz_createDirectory( const sal_Char* pszPath )
434cdf0e10cSrcweir {
435cdf0e10cSrcweir     int nRet=0;
436cdf0e10cSrcweir     int mode = S_IRWXU | S_IRWXG | S_IRWXO;
437cdf0e10cSrcweir 
438cdf0e10cSrcweir     nRet = mkdir(pszPath,mode);
439cdf0e10cSrcweir 
440cdf0e10cSrcweir     if ( nRet < 0 )
441cdf0e10cSrcweir     {
442cdf0e10cSrcweir         nRet=errno;
443cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
444cdf0e10cSrcweir     }
445cdf0e10cSrcweir 
446cdf0e10cSrcweir     return osl_File_E_None;
447cdf0e10cSrcweir }
448cdf0e10cSrcweir 
449cdf0e10cSrcweir /*****************************************
450cdf0e10cSrcweir  * osl_psz_removeDirectory
451cdf0e10cSrcweir  ****************************************/
452cdf0e10cSrcweir 
osl_psz_removeDirectory(const sal_Char * pszPath)453cdf0e10cSrcweir static oslFileError osl_psz_removeDirectory( const sal_Char* pszPath )
454cdf0e10cSrcweir {
455cdf0e10cSrcweir     int nRet=0;
456cdf0e10cSrcweir 
457cdf0e10cSrcweir     nRet = rmdir(pszPath);
458cdf0e10cSrcweir 
459cdf0e10cSrcweir     if ( nRet < 0 )
460cdf0e10cSrcweir     {
461cdf0e10cSrcweir         nRet=errno;
462cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
463cdf0e10cSrcweir     }
464cdf0e10cSrcweir 
465cdf0e10cSrcweir     return osl_File_E_None;
466cdf0e10cSrcweir }
467cdf0e10cSrcweir 
468cdf0e10cSrcweir /****************************************************************************/
469cdf0e10cSrcweir /* osl_createDirectoryPath */
470cdf0e10cSrcweir /****************************************************************************/
471cdf0e10cSrcweir 
path_make_parent(sal_Unicode * path)472cdf0e10cSrcweir static int path_make_parent(sal_Unicode* path)
473cdf0e10cSrcweir {
474cdf0e10cSrcweir     int i = rtl_ustr_lastIndexOfChar(path, '/');
475cdf0e10cSrcweir 
476cdf0e10cSrcweir     if (i > 0)
477cdf0e10cSrcweir     {
478cdf0e10cSrcweir         *(path + i) = 0;
479cdf0e10cSrcweir         return i;
480cdf0e10cSrcweir     }
481cdf0e10cSrcweir     else
482cdf0e10cSrcweir         return 0;
483cdf0e10cSrcweir }
484cdf0e10cSrcweir 
create_dir_with_callback(sal_Unicode * directory_path,oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,void * pData)485cdf0e10cSrcweir static int create_dir_with_callback(
486cdf0e10cSrcweir     sal_Unicode* directory_path,
487cdf0e10cSrcweir     oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,
488cdf0e10cSrcweir     void* pData)
489cdf0e10cSrcweir {
490cdf0e10cSrcweir     int mode = S_IRWXU | S_IRWXG | S_IRWXO;
491cdf0e10cSrcweir 
492cdf0e10cSrcweir     if (osl::mkdir(directory_path, mode) == 0)
493cdf0e10cSrcweir     {
494cdf0e10cSrcweir         if (aDirectoryCreationCallbackFunc)
495cdf0e10cSrcweir         {
496cdf0e10cSrcweir             rtl::OUString url;
497cdf0e10cSrcweir             osl::FileBase::getFileURLFromSystemPath(directory_path, url);
498cdf0e10cSrcweir             aDirectoryCreationCallbackFunc(pData, url.pData);
499cdf0e10cSrcweir         }
500cdf0e10cSrcweir         return 0;
501cdf0e10cSrcweir     }
502cdf0e10cSrcweir     return errno;
503cdf0e10cSrcweir }
504cdf0e10cSrcweir 
create_dir_recursively_(sal_Unicode * dir_path,oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,void * pData)505cdf0e10cSrcweir static oslFileError create_dir_recursively_(
506cdf0e10cSrcweir     sal_Unicode* dir_path,
507cdf0e10cSrcweir     oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,
508cdf0e10cSrcweir     void* pData)
509cdf0e10cSrcweir {
510cdf0e10cSrcweir     OSL_PRECOND((rtl_ustr_getLength(dir_path) > 0) && ((dir_path + (rtl_ustr_getLength(dir_path) - 1)) != (dir_path + rtl_ustr_lastIndexOfChar(dir_path, '/'))), \
511cdf0e10cSrcweir     "Path must not end with a slash");
512cdf0e10cSrcweir 
513cdf0e10cSrcweir     int native_err = create_dir_with_callback(
514cdf0e10cSrcweir         dir_path, aDirectoryCreationCallbackFunc, pData);
515cdf0e10cSrcweir 
516cdf0e10cSrcweir     if (native_err == 0)
517cdf0e10cSrcweir         return osl_File_E_None;
518cdf0e10cSrcweir 
519cdf0e10cSrcweir     if (native_err != ENOENT)
520cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, native_err);
521cdf0e10cSrcweir 
522cdf0e10cSrcweir     // we step back until '/a_dir' at maximum because
523cdf0e10cSrcweir     // we should get an error unequal ENOENT when
524cdf0e10cSrcweir     // we try to create 'a_dir' at '/' and would so
525cdf0e10cSrcweir     // return before
526cdf0e10cSrcweir     int pos = path_make_parent(dir_path);
527cdf0e10cSrcweir 
528cdf0e10cSrcweir     oslFileError osl_error = create_dir_recursively_(
529cdf0e10cSrcweir         dir_path, aDirectoryCreationCallbackFunc, pData);
530cdf0e10cSrcweir 
531cdf0e10cSrcweir     if (osl_File_E_None != osl_error)
532cdf0e10cSrcweir         return osl_error;
533cdf0e10cSrcweir 
534cdf0e10cSrcweir     dir_path[pos] = '/';
535cdf0e10cSrcweir 
536cdf0e10cSrcweir     return create_dir_recursively_(dir_path, aDirectoryCreationCallbackFunc, pData);
537cdf0e10cSrcweir }
538cdf0e10cSrcweir 
osl_createDirectoryPath(rtl_uString * aDirectoryUrl,oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,void * pData)539cdf0e10cSrcweir oslFileError SAL_CALL osl_createDirectoryPath(
540cdf0e10cSrcweir     rtl_uString* aDirectoryUrl,
541cdf0e10cSrcweir     oslDirectoryCreationCallbackFunc aDirectoryCreationCallbackFunc,
542cdf0e10cSrcweir     void* pData)
543cdf0e10cSrcweir {
544cdf0e10cSrcweir     if (aDirectoryUrl == NULL)
545cdf0e10cSrcweir         return osl_File_E_INVAL;
546cdf0e10cSrcweir 
547cdf0e10cSrcweir     rtl::OUString sys_path;
548cdf0e10cSrcweir     oslFileError osl_error = osl_getSystemPathFromFileURL_Ex(
549cdf0e10cSrcweir         aDirectoryUrl, &sys_path.pData, sal_False);
550cdf0e10cSrcweir 
551cdf0e10cSrcweir     if (osl_error != osl_File_E_None)
552cdf0e10cSrcweir         return osl_error;
553cdf0e10cSrcweir 
554cdf0e10cSrcweir     osl::systemPathRemoveSeparator(sys_path);
555cdf0e10cSrcweir 
556cdf0e10cSrcweir     // const_cast because sys_path is a local copy which we want to modify inplace instead of
557*0170220dSmseidel     // copy it into another buffer on the heap again
558cdf0e10cSrcweir     return create_dir_recursively_(sys_path.pData->buffer, aDirectoryCreationCallbackFunc, pData);
559cdf0e10cSrcweir }
560cdf0e10cSrcweir 
561cdf0e10cSrcweir /******************************************************************************
562cdf0e10cSrcweir  * C-String Function Declarations
563cdf0e10cSrcweir  *****************************************************************************/
564cdf0e10cSrcweir 
565cdf0e10cSrcweir static oslFileError osl_psz_removeFile(const sal_Char* pszPath);
566cdf0e10cSrcweir static oslFileError osl_psz_copyFile(const sal_Char* pszPath, const sal_Char* pszDestPath);
567cdf0e10cSrcweir static oslFileError osl_psz_moveFile(const sal_Char* pszPath, const sal_Char* pszDestPath);
568cdf0e10cSrcweir 
569cdf0e10cSrcweir 
570cdf0e10cSrcweir /******************************************************************************
571cdf0e10cSrcweir  * Static Module Utility Function Declarations
572cdf0e10cSrcweir  *****************************************************************************/
573cdf0e10cSrcweir 
574cdf0e10cSrcweir static oslFileError  oslDoCopy(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName, mode_t nMode, size_t nSourceSize, int DestFileExists);
575cdf0e10cSrcweir static oslFileError  oslChangeFileModes(const sal_Char* pszFileName, mode_t nMode, time_t nAcTime, time_t nModTime, uid_t nUID, gid_t nGID);
576cdf0e10cSrcweir static int           oslDoCopyLink(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName);
577cdf0e10cSrcweir static int           oslDoCopyFile(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName, size_t nSourceSize, mode_t mode);
578cdf0e10cSrcweir static oslFileError  oslDoMoveFile(const sal_Char* pszPath, const sal_Char* pszDestPath);
579cdf0e10cSrcweir 
580cdf0e10cSrcweir /****************************************************************************/
581cdf0e10cSrcweir /* osl_moveFile */
582cdf0e10cSrcweir /****************************************************************************/
583cdf0e10cSrcweir 
osl_moveFile(rtl_uString * ustrFileURL,rtl_uString * ustrDestURL)584cdf0e10cSrcweir oslFileError SAL_CALL osl_moveFile( rtl_uString* ustrFileURL, rtl_uString* ustrDestURL )
585cdf0e10cSrcweir {
586cdf0e10cSrcweir     char srcPath[PATH_MAX];
587cdf0e10cSrcweir     char destPath[PATH_MAX];
588cdf0e10cSrcweir     oslFileError eRet;
589cdf0e10cSrcweir 
590cdf0e10cSrcweir     OSL_ASSERT( ustrFileURL );
591cdf0e10cSrcweir     OSL_ASSERT( ustrDestURL );
592cdf0e10cSrcweir 
593cdf0e10cSrcweir     /* convert source url to system path */
594cdf0e10cSrcweir     eRet = FileURLToPath( srcPath, PATH_MAX, ustrFileURL );
595cdf0e10cSrcweir     if( eRet != osl_File_E_None )
596cdf0e10cSrcweir         return eRet;
597cdf0e10cSrcweir 
598cdf0e10cSrcweir     /* convert destination url to system path */
599cdf0e10cSrcweir     eRet = FileURLToPath( destPath, PATH_MAX, ustrDestURL );
600cdf0e10cSrcweir     if( eRet != osl_File_E_None )
601cdf0e10cSrcweir         return eRet;
602cdf0e10cSrcweir 
603cdf0e10cSrcweir #ifdef MACOSX
604cdf0e10cSrcweir     if ( macxp_resolveAlias( srcPath, PATH_MAX ) != 0 || macxp_resolveAlias( destPath, PATH_MAX ) != 0 )
605cdf0e10cSrcweir         return oslTranslateFileError( OSL_FET_ERROR, errno );
606cdf0e10cSrcweir #endif/* MACOSX */
607cdf0e10cSrcweir 
608cdf0e10cSrcweir     return oslDoMoveFile( srcPath, destPath );
609cdf0e10cSrcweir }
610cdf0e10cSrcweir 
611cdf0e10cSrcweir /****************************************************************************/
612cdf0e10cSrcweir /* osl_copyFile */
613cdf0e10cSrcweir /****************************************************************************/
614cdf0e10cSrcweir 
osl_copyFile(rtl_uString * ustrFileURL,rtl_uString * ustrDestURL)615cdf0e10cSrcweir oslFileError SAL_CALL osl_copyFile( rtl_uString* ustrFileURL, rtl_uString* ustrDestURL )
616cdf0e10cSrcweir {
617cdf0e10cSrcweir     char srcPath[PATH_MAX];
618cdf0e10cSrcweir     char destPath[PATH_MAX];
619cdf0e10cSrcweir     oslFileError eRet;
620cdf0e10cSrcweir 
621cdf0e10cSrcweir     OSL_ASSERT( ustrFileURL );
622cdf0e10cSrcweir     OSL_ASSERT( ustrDestURL );
623cdf0e10cSrcweir 
624cdf0e10cSrcweir     /* convert source url to system path */
625cdf0e10cSrcweir     eRet = FileURLToPath( srcPath, PATH_MAX, ustrFileURL );
626cdf0e10cSrcweir     if( eRet != osl_File_E_None )
627cdf0e10cSrcweir         return eRet;
628cdf0e10cSrcweir 
629cdf0e10cSrcweir     /* convert destination url to system path */
630cdf0e10cSrcweir     eRet = FileURLToPath( destPath, PATH_MAX, ustrDestURL );
631cdf0e10cSrcweir     if( eRet != osl_File_E_None )
632cdf0e10cSrcweir         return eRet;
633cdf0e10cSrcweir 
634cdf0e10cSrcweir #ifdef MACOSX
635cdf0e10cSrcweir     if ( macxp_resolveAlias( srcPath, PATH_MAX ) != 0 || macxp_resolveAlias( destPath, PATH_MAX ) != 0 )
636cdf0e10cSrcweir         return oslTranslateFileError( OSL_FET_ERROR, errno );
637cdf0e10cSrcweir #endif/* MACOSX */
638cdf0e10cSrcweir 
639cdf0e10cSrcweir     return osl_psz_copyFile( srcPath, destPath );
640cdf0e10cSrcweir }
641cdf0e10cSrcweir 
642cdf0e10cSrcweir /****************************************************************************/
643cdf0e10cSrcweir /* osl_removeFile */
644cdf0e10cSrcweir /****************************************************************************/
645cdf0e10cSrcweir 
osl_removeFile(rtl_uString * ustrFileURL)646cdf0e10cSrcweir oslFileError SAL_CALL osl_removeFile( rtl_uString* ustrFileURL )
647cdf0e10cSrcweir {
648cdf0e10cSrcweir     char path[PATH_MAX];
649cdf0e10cSrcweir     oslFileError eRet;
650cdf0e10cSrcweir 
651cdf0e10cSrcweir     OSL_ASSERT( ustrFileURL );
652cdf0e10cSrcweir 
653cdf0e10cSrcweir     /* convert file url to system path */
654cdf0e10cSrcweir     eRet = FileURLToPath( path, PATH_MAX, ustrFileURL );
655cdf0e10cSrcweir     if( eRet != osl_File_E_None )
656cdf0e10cSrcweir         return eRet;
657cdf0e10cSrcweir 
658cdf0e10cSrcweir #ifdef MACOSX
659cdf0e10cSrcweir     if ( macxp_resolveAlias( path, PATH_MAX ) != 0 )
660cdf0e10cSrcweir         return oslTranslateFileError( OSL_FET_ERROR, errno );
661cdf0e10cSrcweir #endif/* MACOSX */
662cdf0e10cSrcweir 
663cdf0e10cSrcweir     return osl_psz_removeFile( path );
664cdf0e10cSrcweir }
665cdf0e10cSrcweir 
666cdf0e10cSrcweir /******************************************************************************
667cdf0e10cSrcweir  * Utility Functions
668cdf0e10cSrcweir  *****************************************************************************/
669cdf0e10cSrcweir 
670cdf0e10cSrcweir /*****************************************
671cdf0e10cSrcweir  * oslDoMoveFile
672cdf0e10cSrcweir  ****************************************/
673cdf0e10cSrcweir 
oslDoMoveFile(const sal_Char * pszPath,const sal_Char * pszDestPath)674cdf0e10cSrcweir static oslFileError oslDoMoveFile( const sal_Char* pszPath, const sal_Char* pszDestPath)
675cdf0e10cSrcweir {
676cdf0e10cSrcweir     oslFileError tErr=osl_File_E_invalidError;
677cdf0e10cSrcweir 
678cdf0e10cSrcweir     tErr = osl_psz_moveFile(pszPath,pszDestPath);
679cdf0e10cSrcweir     if ( tErr == osl_File_E_None )
680cdf0e10cSrcweir     {
681cdf0e10cSrcweir         return tErr;
682cdf0e10cSrcweir     }
683cdf0e10cSrcweir 
684cdf0e10cSrcweir     if ( tErr != osl_File_E_XDEV )
685cdf0e10cSrcweir     {
686cdf0e10cSrcweir         return tErr;
687cdf0e10cSrcweir     }
688cdf0e10cSrcweir 
689cdf0e10cSrcweir     tErr=osl_psz_copyFile(pszPath,pszDestPath);
690cdf0e10cSrcweir 
691cdf0e10cSrcweir     if ( tErr != osl_File_E_None )
692cdf0e10cSrcweir     {
693cdf0e10cSrcweir         oslFileError tErrRemove;
694cdf0e10cSrcweir         tErrRemove=osl_psz_removeFile(pszDestPath);
695cdf0e10cSrcweir         return tErr;
696cdf0e10cSrcweir     }
697cdf0e10cSrcweir 
698cdf0e10cSrcweir     tErr=osl_psz_removeFile(pszPath);
699cdf0e10cSrcweir 
700cdf0e10cSrcweir     return tErr;
701cdf0e10cSrcweir }
702cdf0e10cSrcweir 
703cdf0e10cSrcweir /*****************************************
704cdf0e10cSrcweir  * osl_psz_removeFile
705cdf0e10cSrcweir  ****************************************/
osl_psz_removeFile(const sal_Char * pszPath)706cdf0e10cSrcweir static oslFileError osl_psz_removeFile( const sal_Char* pszPath )
707cdf0e10cSrcweir {
708cdf0e10cSrcweir     int nRet=0;
709cdf0e10cSrcweir     struct stat aStat;
710cdf0e10cSrcweir 
711cdf0e10cSrcweir     nRet = lstat(pszPath,&aStat);
712cdf0e10cSrcweir     if ( nRet < 0 )
713cdf0e10cSrcweir     {
714cdf0e10cSrcweir         nRet=errno;
715cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
716cdf0e10cSrcweir     }
717cdf0e10cSrcweir 
718cdf0e10cSrcweir     if ( S_ISDIR(aStat.st_mode) )
719cdf0e10cSrcweir     {
720cdf0e10cSrcweir         return osl_File_E_ISDIR;
721cdf0e10cSrcweir     }
722cdf0e10cSrcweir 
723cdf0e10cSrcweir     nRet = unlink(pszPath);
724cdf0e10cSrcweir     if ( nRet < 0 )
725cdf0e10cSrcweir     {
726cdf0e10cSrcweir         nRet=errno;
727cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
728cdf0e10cSrcweir     }
729cdf0e10cSrcweir 
730cdf0e10cSrcweir     return osl_File_E_None;
731cdf0e10cSrcweir }
732cdf0e10cSrcweir 
733cdf0e10cSrcweir /*****************************************
734cdf0e10cSrcweir  * osl_psz_moveFile
735cdf0e10cSrcweir  ****************************************/
736cdf0e10cSrcweir 
osl_psz_moveFile(const sal_Char * pszPath,const sal_Char * pszDestPath)737cdf0e10cSrcweir static oslFileError osl_psz_moveFile(const sal_Char* pszPath, const sal_Char* pszDestPath)
738cdf0e10cSrcweir {
739cdf0e10cSrcweir 
740cdf0e10cSrcweir     int nRet = 0;
741cdf0e10cSrcweir 
742cdf0e10cSrcweir     nRet = rename(pszPath,pszDestPath);
743cdf0e10cSrcweir 
744cdf0e10cSrcweir     if ( nRet < 0 )
745cdf0e10cSrcweir     {
746cdf0e10cSrcweir         nRet=errno;
747cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
748cdf0e10cSrcweir     }
749cdf0e10cSrcweir 
750cdf0e10cSrcweir     return osl_File_E_None;
751cdf0e10cSrcweir }
752cdf0e10cSrcweir 
753cdf0e10cSrcweir /*****************************************
754cdf0e10cSrcweir  * osl_psz_copyFile
755cdf0e10cSrcweir  ****************************************/
756cdf0e10cSrcweir 
osl_psz_copyFile(const sal_Char * pszPath,const sal_Char * pszDestPath)757cdf0e10cSrcweir static oslFileError osl_psz_copyFile( const sal_Char* pszPath, const sal_Char* pszDestPath )
758cdf0e10cSrcweir {
759cdf0e10cSrcweir     time_t nAcTime=0;
760cdf0e10cSrcweir     time_t nModTime=0;
761cdf0e10cSrcweir     uid_t nUID=0;
762cdf0e10cSrcweir     gid_t nGID=0;
763cdf0e10cSrcweir     int nRet=0;
764cdf0e10cSrcweir     mode_t nMode=0;
765cdf0e10cSrcweir     struct stat aFileStat;
766cdf0e10cSrcweir     oslFileError tErr=osl_File_E_invalidError;
767cdf0e10cSrcweir     size_t nSourceSize=0;
768cdf0e10cSrcweir     int DestFileExists=1;
769cdf0e10cSrcweir 
770cdf0e10cSrcweir     /* mfe: does the source file really exists? */
771cdf0e10cSrcweir     nRet = lstat(pszPath,&aFileStat);
772cdf0e10cSrcweir 
773cdf0e10cSrcweir     if ( nRet < 0 )
774cdf0e10cSrcweir     {
775cdf0e10cSrcweir         nRet=errno;
776cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
777cdf0e10cSrcweir     }
778cdf0e10cSrcweir 
779cdf0e10cSrcweir     /* mfe: we do only copy files here! */
780cdf0e10cSrcweir     if ( S_ISDIR(aFileStat.st_mode) )
781cdf0e10cSrcweir     {
782cdf0e10cSrcweir         return osl_File_E_ISDIR;
783cdf0e10cSrcweir     }
784cdf0e10cSrcweir 
785cdf0e10cSrcweir     nSourceSize=(size_t)aFileStat.st_size;
786cdf0e10cSrcweir     nMode=aFileStat.st_mode;
787cdf0e10cSrcweir     nAcTime=aFileStat.st_atime;
788cdf0e10cSrcweir     nModTime=aFileStat.st_mtime;
789cdf0e10cSrcweir     nUID=aFileStat.st_uid;
790cdf0e10cSrcweir     nGID=aFileStat.st_gid;
791cdf0e10cSrcweir 
792cdf0e10cSrcweir     nRet = stat(pszDestPath,&aFileStat);
793cdf0e10cSrcweir     if ( nRet < 0 )
794cdf0e10cSrcweir     {
795cdf0e10cSrcweir         nRet=errno;
796cdf0e10cSrcweir 
797cdf0e10cSrcweir         if ( nRet == ENOENT )
798cdf0e10cSrcweir         {
799cdf0e10cSrcweir             DestFileExists=0;
800cdf0e10cSrcweir         }
801cdf0e10cSrcweir /*      return oslTranslateFileError(nRet);*/
802cdf0e10cSrcweir     }
803cdf0e10cSrcweir 
804cdf0e10cSrcweir     /* mfe: the destination file must not be a directory! */
805cdf0e10cSrcweir     if ( nRet == 0 && S_ISDIR(aFileStat.st_mode) )
806cdf0e10cSrcweir     {
807cdf0e10cSrcweir         return osl_File_E_ISDIR;
808cdf0e10cSrcweir     }
809cdf0e10cSrcweir     else
810cdf0e10cSrcweir     {
811cdf0e10cSrcweir         /* mfe: file does not exists or is no dir */
812cdf0e10cSrcweir     }
813cdf0e10cSrcweir 
814cdf0e10cSrcweir     tErr = oslDoCopy(pszPath,pszDestPath,nMode,nSourceSize,DestFileExists);
815cdf0e10cSrcweir 
816cdf0e10cSrcweir     if ( tErr != osl_File_E_None )
817cdf0e10cSrcweir     {
818cdf0e10cSrcweir         return tErr;
819cdf0e10cSrcweir     }
820cdf0e10cSrcweir 
821cdf0e10cSrcweir     /*
822cdf0e10cSrcweir      *   mfe: ignore return code
823cdf0e10cSrcweir      *        since only the success of the copy is
824cdf0e10cSrcweir      *        important
825cdf0e10cSrcweir      */
826cdf0e10cSrcweir     oslChangeFileModes(pszDestPath,nMode,nAcTime,nModTime,nUID,nGID);
827cdf0e10cSrcweir 
828cdf0e10cSrcweir     return tErr;
829cdf0e10cSrcweir }
830cdf0e10cSrcweir 
831cdf0e10cSrcweir 
832cdf0e10cSrcweir /******************************************************************************
833cdf0e10cSrcweir  * Utility Functions
834cdf0e10cSrcweir  *****************************************************************************/
835cdf0e10cSrcweir 
836cdf0e10cSrcweir /*****************************************
837cdf0e10cSrcweir  * oslDoCopy
838cdf0e10cSrcweir  ****************************************/
839cdf0e10cSrcweir 
840cdf0e10cSrcweir #define TMP_DEST_FILE_EXTENSION ".osl-tmp"
841cdf0e10cSrcweir 
oslDoCopy(const sal_Char * pszSourceFileName,const sal_Char * pszDestFileName,mode_t nMode,size_t nSourceSize,int DestFileExists)842cdf0e10cSrcweir static oslFileError oslDoCopy(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName, mode_t nMode, size_t nSourceSize, int DestFileExists)
843cdf0e10cSrcweir {
844cdf0e10cSrcweir     int      nRet=0;
845cdf0e10cSrcweir     sal_Char pszTmpDestFile[PATH_MAX];
846cdf0e10cSrcweir     size_t   size_tmp_dest_buff = sizeof(pszTmpDestFile);
847cdf0e10cSrcweir 
848cdf0e10cSrcweir     /* Quick fix for #106048, the whole copy file function seems
849cdf0e10cSrcweir        to be erroneous anyway and needs to be rewritten.
850*0170220dSmseidel        Besides osl_copyFile is currently not used from AOO code.
851cdf0e10cSrcweir     */
852cdf0e10cSrcweir     memset(pszTmpDestFile, 0, size_tmp_dest_buff);
853cdf0e10cSrcweir 
854cdf0e10cSrcweir     if ( DestFileExists )
855cdf0e10cSrcweir     {
856cdf0e10cSrcweir         strncpy(pszTmpDestFile, pszDestFileName, size_tmp_dest_buff - 1);
857cdf0e10cSrcweir 
858cdf0e10cSrcweir         if ((strlen(pszTmpDestFile) + strlen(TMP_DEST_FILE_EXTENSION)) >= size_tmp_dest_buff)
859cdf0e10cSrcweir             return osl_File_E_NAMETOOLONG;
860cdf0e10cSrcweir 
861cdf0e10cSrcweir         strncat(pszTmpDestFile, TMP_DEST_FILE_EXTENSION, strlen(TMP_DEST_FILE_EXTENSION));
862cdf0e10cSrcweir 
863cdf0e10cSrcweir         /* FIXME: what if pszTmpDestFile already exists? */
864cdf0e10cSrcweir         /*        with getcanonical??? */
865cdf0e10cSrcweir         nRet=rename(pszDestFileName,pszTmpDestFile);
866cdf0e10cSrcweir     }
867cdf0e10cSrcweir 
868cdf0e10cSrcweir     /* mfe: should be S_ISREG */
869cdf0e10cSrcweir     if ( !S_ISLNK(nMode) )
870cdf0e10cSrcweir     {
871cdf0e10cSrcweir         /* copy SourceFile to DestFile */
872cdf0e10cSrcweir         nRet = oslDoCopyFile(pszSourceFileName,pszDestFileName,nSourceSize, nMode);
873cdf0e10cSrcweir     }
874cdf0e10cSrcweir     /* mfe: OK redundant at the moment */
875cdf0e10cSrcweir     else if ( S_ISLNK(nMode) )
876cdf0e10cSrcweir     {
877cdf0e10cSrcweir         nRet = oslDoCopyLink(pszSourceFileName,pszDestFileName);
878cdf0e10cSrcweir     }
879cdf0e10cSrcweir     else
880cdf0e10cSrcweir     {
881cdf0e10cSrcweir         /* mfe: what to do here? */
882cdf0e10cSrcweir         nRet=ENOSYS;
883cdf0e10cSrcweir     }
884cdf0e10cSrcweir 
885cdf0e10cSrcweir     if ( nRet > 0 && DestFileExists == 1 )
886cdf0e10cSrcweir     {
887cdf0e10cSrcweir         unlink(pszDestFileName);
888cdf0e10cSrcweir         rename(pszTmpDestFile,pszDestFileName);
889cdf0e10cSrcweir     }
890cdf0e10cSrcweir 
891cdf0e10cSrcweir     if ( nRet > 0 )
892cdf0e10cSrcweir     {
893cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
894cdf0e10cSrcweir     }
895cdf0e10cSrcweir 
896cdf0e10cSrcweir     if ( DestFileExists == 1 )
897cdf0e10cSrcweir     {
898cdf0e10cSrcweir         unlink(pszTmpDestFile);
899cdf0e10cSrcweir     }
900cdf0e10cSrcweir 
901cdf0e10cSrcweir     return osl_File_E_None;
902cdf0e10cSrcweir }
903cdf0e10cSrcweir 
904cdf0e10cSrcweir /*****************************************
905cdf0e10cSrcweir  * oslChangeFileModes
906cdf0e10cSrcweir  ****************************************/
907cdf0e10cSrcweir 
oslChangeFileModes(const sal_Char * pszFileName,mode_t nMode,time_t nAcTime,time_t nModTime,uid_t nUID,gid_t nGID)908cdf0e10cSrcweir static oslFileError oslChangeFileModes( const sal_Char* pszFileName, mode_t nMode, time_t nAcTime, time_t nModTime, uid_t nUID, gid_t nGID)
909cdf0e10cSrcweir {
910cdf0e10cSrcweir     int nRet=0;
911cdf0e10cSrcweir     struct utimbuf aTimeBuffer;
912cdf0e10cSrcweir 
913cdf0e10cSrcweir     nRet = chmod(pszFileName,nMode);
914cdf0e10cSrcweir     if ( nRet < 0 )
915cdf0e10cSrcweir     {
916cdf0e10cSrcweir         nRet=errno;
917cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
918cdf0e10cSrcweir     }
919cdf0e10cSrcweir 
920cdf0e10cSrcweir     aTimeBuffer.actime=nAcTime;
921cdf0e10cSrcweir     aTimeBuffer.modtime=nModTime;
922cdf0e10cSrcweir     nRet=utime(pszFileName,&aTimeBuffer);
923cdf0e10cSrcweir     if ( nRet < 0 )
924cdf0e10cSrcweir     {
925cdf0e10cSrcweir         nRet=errno;
926cdf0e10cSrcweir         return oslTranslateFileError(OSL_FET_ERROR, nRet);
927cdf0e10cSrcweir     }
928cdf0e10cSrcweir 
929cdf0e10cSrcweir     if ( nUID != getuid() )
930cdf0e10cSrcweir     {
931cdf0e10cSrcweir         nUID=getuid();
932cdf0e10cSrcweir     }
933cdf0e10cSrcweir 
934cdf0e10cSrcweir     nRet=chown(pszFileName,nUID,nGID);
935cdf0e10cSrcweir     if ( nRet < 0 )
936cdf0e10cSrcweir     {
937cdf0e10cSrcweir         nRet=errno;
938cdf0e10cSrcweir 
939cdf0e10cSrcweir         /* mfe: do not return an error here! */
940cdf0e10cSrcweir         /* return oslTranslateFileError(nRet);*/
941cdf0e10cSrcweir     }
942cdf0e10cSrcweir 
943cdf0e10cSrcweir     return osl_File_E_None;
944cdf0e10cSrcweir }
945cdf0e10cSrcweir 
946cdf0e10cSrcweir /*****************************************
947cdf0e10cSrcweir  * oslDoCopyLink
948cdf0e10cSrcweir  ****************************************/
949cdf0e10cSrcweir 
oslDoCopyLink(const sal_Char * pszSourceFileName,const sal_Char * pszDestFileName)950cdf0e10cSrcweir static int oslDoCopyLink(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName)
951cdf0e10cSrcweir {
952cdf0e10cSrcweir     int nRet=0;
953cdf0e10cSrcweir 
954cdf0e10cSrcweir     /* mfe: if dest file is symbolic link remove the link and place the file instead (hro says so) */
955cdf0e10cSrcweir     /* mfe: if source is a link copy the link and not the file it points to (hro says so) */
956cdf0e10cSrcweir     sal_Char pszLinkContent[PATH_MAX];
957cdf0e10cSrcweir 
958cdf0e10cSrcweir     pszLinkContent[0] = '\0';
959cdf0e10cSrcweir 
9603f12a67eSPedro Giffuni     nRet = readlink(pszSourceFileName,pszLinkContent,PATH_MAX-1);
961cdf0e10cSrcweir 
962cdf0e10cSrcweir     if ( nRet < 0 )
963cdf0e10cSrcweir     {
964cdf0e10cSrcweir         nRet=errno;
965cdf0e10cSrcweir         return nRet;
966cdf0e10cSrcweir     }
967cdf0e10cSrcweir     else
968cdf0e10cSrcweir         pszLinkContent[ nRet ] = 0;
969cdf0e10cSrcweir 
970cdf0e10cSrcweir     nRet = symlink(pszLinkContent,pszDestFileName);
971cdf0e10cSrcweir 
972cdf0e10cSrcweir     if ( nRet < 0 )
973cdf0e10cSrcweir     {
974cdf0e10cSrcweir         nRet=errno;
975cdf0e10cSrcweir         return nRet;
976cdf0e10cSrcweir     }
977cdf0e10cSrcweir 
978cdf0e10cSrcweir     return 0;
979cdf0e10cSrcweir }
980cdf0e10cSrcweir 
981cdf0e10cSrcweir /*****************************************
982cdf0e10cSrcweir  * oslDoCopyFile
983cdf0e10cSrcweir  ****************************************/
984cdf0e10cSrcweir 
oslDoCopyFile(const sal_Char * pszSourceFileName,const sal_Char * pszDestFileName,size_t nSourceSize,mode_t mode)985cdf0e10cSrcweir static int oslDoCopyFile(const sal_Char* pszSourceFileName, const sal_Char* pszDestFileName, size_t nSourceSize, mode_t mode)
986cdf0e10cSrcweir {
987cdf0e10cSrcweir     int SourceFileFD=0;
988cdf0e10cSrcweir     int DestFileFD=0;
989cdf0e10cSrcweir     int nRet=0;
990cdf0e10cSrcweir 
991cdf0e10cSrcweir     SourceFileFD=open(pszSourceFileName,O_RDONLY);
992cdf0e10cSrcweir     if ( SourceFileFD < 0 )
993cdf0e10cSrcweir     {
994cdf0e10cSrcweir         nRet=errno;
995cdf0e10cSrcweir         return nRet;
996cdf0e10cSrcweir     }
997cdf0e10cSrcweir 
998cdf0e10cSrcweir     DestFileFD=open(pszDestFileName, O_WRONLY | O_CREAT, mode);
999cdf0e10cSrcweir 
1000cdf0e10cSrcweir     if ( DestFileFD < 0 )
1001cdf0e10cSrcweir     {
1002cdf0e10cSrcweir         nRet=errno;
1003cdf0e10cSrcweir         close(SourceFileFD);
1004cdf0e10cSrcweir         return nRet;
1005cdf0e10cSrcweir     }
1006cdf0e10cSrcweir 
1007cdf0e10cSrcweir     size_t nWritten = 0;
1008cdf0e10cSrcweir     size_t nRemains = nSourceSize;
1009cdf0e10cSrcweir 
1010cdf0e10cSrcweir     if ( nRemains )
1011cdf0e10cSrcweir     {
1012cdf0e10cSrcweir         /* mmap has problems, try the direct streaming */
1013cdf0e10cSrcweir         char pBuffer[0x8000];
1014cdf0e10cSrcweir         size_t nRead = 0;
1015cdf0e10cSrcweir 
1016cdf0e10cSrcweir         nRemains = nSourceSize;
1017cdf0e10cSrcweir 
1018cdf0e10cSrcweir         do
1019cdf0e10cSrcweir         {
1020cdf0e10cSrcweir             nRead = 0;
1021cdf0e10cSrcweir             nWritten = 0;
1022cdf0e10cSrcweir 
1023cdf0e10cSrcweir             size_t nToRead = std::min( (size_t)0x8000, nRemains );
1024cdf0e10cSrcweir             nRead = read( SourceFileFD, pBuffer, nToRead );
1025cdf0e10cSrcweir             if ( (size_t)-1 != nRead )
1026cdf0e10cSrcweir                 nWritten = write( DestFileFD, pBuffer, nRead );
1027cdf0e10cSrcweir 
1028cdf0e10cSrcweir             if ( (size_t)-1 != nWritten )
1029cdf0e10cSrcweir                 nRemains -= nWritten;
1030cdf0e10cSrcweir         }
1031cdf0e10cSrcweir         while( nRemains && (size_t)-1 != nRead && nRead == nWritten );
1032cdf0e10cSrcweir     }
1033cdf0e10cSrcweir 
1034cdf0e10cSrcweir     if ( nRemains )
1035cdf0e10cSrcweir     {
1036cdf0e10cSrcweir         if ( errno )
1037cdf0e10cSrcweir             nRet = errno;
1038cdf0e10cSrcweir         else
1039cdf0e10cSrcweir             nRet = ENOSPC;
1040cdf0e10cSrcweir     }
1041cdf0e10cSrcweir 
1042cdf0e10cSrcweir     close( SourceFileFD );
1043cdf0e10cSrcweir     if ( close( DestFileFD ) == -1 && nRet == 0 )
1044cdf0e10cSrcweir         nRet = errno;
1045cdf0e10cSrcweir 
1046cdf0e10cSrcweir     return nRet;
1047cdf0e10cSrcweir }
1048