1 /**************************************************************
2 *
3 * Licensed to the Apache Software Foundation (ASF) under one
4 * or more contributor license agreements. See the NOTICE file
5 * distributed with this work for additional information
6 * regarding copyright ownership. The ASF licenses this file
7 * to you under the Apache License, Version 2.0 (the
8 * "License"); you may not use this file except in compliance
9 * with the License. You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing,
14 * software distributed under the License is distributed on an
15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16 * KIND, either express or implied. See the License for the
17 * specific language governing permissions and limitations
18 * under the License.
19 *
20 *************************************************************/
21
22
23
24 // no include "precompiled_tools.hxx" because this file is included in strmsys.cxx
25
26 #include <stdio.h>
27 #include <string.h>
28 #include <sys/stat.h>
29 #include <fcntl.h>
30 #include <errno.h>
31 #include <unistd.h>
32 #include <limits.h>
33 #include <stdlib.h> // fuer getenv()
34
35 #include <tools/debug.hxx>
36 #include <tools/fsys.hxx>
37 #include <tools/stream.hxx>
38
39 #include <vos/mutex.hxx>
40 #include <osl/thread.h> // osl_getThreadTextEncoding
41
42 // class FileBase
43 #include <osl/file.hxx>
44 #include <rtl/instance.hxx>
45
46 using namespace osl;
47
48 // -----------------------------------------------------------------------
49
50 // ----------------
51 // - InternalLock -
52 // ----------------
53
54 class InternalStreamLock;
55 DECLARE_LIST( InternalStreamLockList, InternalStreamLock* )
56 namespace { struct LockList : public rtl::Static< InternalStreamLockList, LockList > {}; }
57
58 #ifndef BOOTSTRAP
59 namespace { struct LockMutex : public rtl::Static< vos::OMutex, LockMutex > {}; }
60 #endif
61
62 class InternalStreamLock
63 {
64 sal_Size m_nStartPos;
65 sal_Size m_nEndPos;
66 SvFileStream* m_pStream;
67 struct stat m_aStat;
68
69 InternalStreamLock( sal_Size, sal_Size, SvFileStream* );
70 ~InternalStreamLock();
71 public:
72 static sal_Bool LockFile( sal_Size nStart, sal_Size nEnd, SvFileStream* );
73 static void UnlockFile( sal_Size nStart, sal_Size nEnd, SvFileStream* );
74 };
75
InternalStreamLock(sal_Size nStart,sal_Size nEnd,SvFileStream * pStream)76 InternalStreamLock::InternalStreamLock(
77 sal_Size nStart,
78 sal_Size nEnd,
79 SvFileStream* pStream ) :
80 m_nStartPos( nStart ),
81 m_nEndPos( nEnd ),
82 m_pStream( pStream )
83 {
84 ByteString aFileName(m_pStream->GetFileName(), osl_getThreadTextEncoding());
85 stat( aFileName.GetBuffer(), &m_aStat );
86 LockList::get().Insert( this, LIST_APPEND );
87 #if OSL_DEBUG_LEVEL > 1
88 fprintf( stderr, "locked %s", aFileName.GetBuffer() );
89 if( m_nStartPos || m_nEndPos )
90 fprintf(stderr, " [ %ld ... %ld ]", m_nStartPos, m_nEndPos );
91 fprintf( stderr, "\n" );
92 #endif
93 }
94
~InternalStreamLock()95 InternalStreamLock::~InternalStreamLock()
96 {
97 LockList::get().Remove( this );
98 #if OSL_DEBUG_LEVEL > 1
99 ByteString aFileName(m_pStream->GetFileName(), osl_getThreadTextEncoding());
100 fprintf( stderr, "unlocked %s", aFileName.GetBuffer() );
101 if( m_nStartPos || m_nEndPos )
102 fprintf(stderr, " [ %ld ... %ld ]", m_nStartPos, m_nEndPos );
103 fprintf( stderr, "\n" );
104 #endif
105 }
106
LockFile(sal_Size nStart,sal_Size nEnd,SvFileStream * pStream)107 sal_Bool InternalStreamLock::LockFile( sal_Size nStart, sal_Size nEnd, SvFileStream* pStream )
108 {
109 #ifndef BOOTSTRAP
110 vos:: OGuard aGuard( LockMutex::get() );
111 #endif
112 ByteString aFileName(pStream->GetFileName(), osl_getThreadTextEncoding());
113 struct stat aStat;
114 if( stat( aFileName.GetBuffer(), &aStat ) )
115 return sal_False;
116
117 if( S_ISDIR( aStat.st_mode ) )
118 return sal_True;
119
120 InternalStreamLock* pLock = NULL;
121 InternalStreamLockList &rLockList = LockList::get();
122 for( sal_uIntPtr i = 0; i < rLockList.Count(); ++i )
123 {
124 pLock = rLockList.GetObject( i );
125 if( aStat.st_ino == pLock->m_aStat.st_ino )
126 {
127 sal_Bool bDenyByOptions = sal_False;
128 StreamMode nLockMode = pLock->m_pStream->GetStreamMode();
129 StreamMode nNewMode = pStream->GetStreamMode();
130
131 if( nLockMode & STREAM_SHARE_DENYALL )
132 bDenyByOptions = sal_True;
133 else if( ( nLockMode & STREAM_SHARE_DENYWRITE ) &&
134 ( nNewMode & STREAM_WRITE ) )
135 bDenyByOptions = sal_True;
136 else if( ( nLockMode & STREAM_SHARE_DENYREAD ) &&
137 ( nNewMode & STREAM_READ ) )
138 bDenyByOptions = sal_True;
139
140 if( bDenyByOptions )
141 {
142 if( pLock->m_nStartPos == 0 && pLock->m_nEndPos == 0 ) // whole file is already locked
143 return sal_False;
144 if( nStart == 0 && nEnd == 0) // cannot lock whole file
145 return sal_False;
146
147 if( ( nStart < pLock->m_nStartPos && nEnd > pLock->m_nStartPos ) ||
148 ( nStart < pLock->m_nEndPos && nEnd > pLock->m_nEndPos ) )
149 return sal_False;
150 }
151 }
152 }
153 pLock = new InternalStreamLock( nStart, nEnd, pStream );
154 return sal_True;
155 }
156
UnlockFile(sal_Size nStart,sal_Size nEnd,SvFileStream * pStream)157 void InternalStreamLock::UnlockFile( sal_Size nStart, sal_Size nEnd, SvFileStream* pStream )
158 {
159 #ifndef BOOTSTRAP
160 vos:: OGuard aGuard( LockMutex::get() );
161 #endif
162 InternalStreamLock* pLock = NULL;
163 InternalStreamLockList &rLockList = LockList::get();
164 if( nStart == 0 && nEnd == 0 )
165 {
166 for( sal_uIntPtr i = 0; i < rLockList.Count(); ++i )
167 {
168 if( ( pLock = rLockList.GetObject( i ) )->m_pStream == pStream )
169 {
170 delete pLock;
171 i--;
172 }
173 }
174 return;
175 }
176 for( sal_uIntPtr i = 0; i < rLockList.Count(); ++i )
177 {
178 if( ( pLock = rLockList.GetObject( i ) )->m_pStream == pStream &&
179 nStart == pLock->m_nStartPos && nEnd == pLock->m_nEndPos )
180 {
181 delete pLock;
182 return;
183 }
184 }
185 }
186
187 // --------------
188 // - StreamData -
189 // --------------
190
191 class StreamData
192 {
193 public:
194 int nHandle;
195
StreamData()196 StreamData() { nHandle = 0; }
197 };
198
199 // -----------------------------------------------------------------------
200
GetSvError(int nErrno)201 static sal_uInt32 GetSvError( int nErrno )
202 {
203 static struct { int nErr; sal_uInt32 sv; } errArr[] =
204 {
205 { 0, SVSTREAM_OK },
206 { EACCES, SVSTREAM_ACCESS_DENIED },
207 { EBADF, SVSTREAM_INVALID_HANDLE },
208 #if defined( RS6000 ) || defined( ALPHA ) || defined( HP9000 ) || defined( NETBSD ) || defined(FREEBSD) || defined(MACOSX) || defined(__FreeBSD_kernel__)
209 { EDEADLK, SVSTREAM_LOCKING_VIOLATION },
210 #else
211 { EDEADLOCK, SVSTREAM_LOCKING_VIOLATION },
212 #endif
213 { EINVAL, SVSTREAM_INVALID_PARAMETER },
214 { EMFILE, SVSTREAM_TOO_MANY_OPEN_FILES },
215 { ENFILE, SVSTREAM_TOO_MANY_OPEN_FILES },
216 { ENOENT, SVSTREAM_FILE_NOT_FOUND },
217 { EPERM, SVSTREAM_ACCESS_DENIED },
218 { EROFS, SVSTREAM_ACCESS_DENIED },
219 { EAGAIN, SVSTREAM_LOCKING_VIOLATION },
220 { EISDIR, SVSTREAM_PATH_NOT_FOUND },
221 { ELOOP, SVSTREAM_PATH_NOT_FOUND },
222 #if ! defined( RS6000 ) && ! defined( ALPHA ) && ! defined( NETBSD ) && ! defined (FREEBSD) && ! defined (MACOSX) && ! defined(__FreeBSD_kernel__)
223 { EMULTIHOP, SVSTREAM_PATH_NOT_FOUND },
224 { ENOLINK, SVSTREAM_PATH_NOT_FOUND },
225 #endif
226 { ENOTDIR, SVSTREAM_PATH_NOT_FOUND },
227 { ETXTBSY, SVSTREAM_ACCESS_DENIED },
228 { EEXIST, SVSTREAM_CANNOT_MAKE },
229 { ENOSPC, SVSTREAM_DISK_FULL },
230 { (int)0xFFFF, SVSTREAM_GENERALERROR }
231 };
232
233 sal_uInt32 nRetVal = SVSTREAM_GENERALERROR; // Standardfehler
234 int i=0;
235 do
236 {
237 if ( errArr[i].nErr == nErrno )
238 {
239 nRetVal = errArr[i].sv;
240 break;
241 }
242 ++i;
243 }
244 while( errArr[i].nErr != 0xFFFF );
245 return nRetVal;
246 }
247
248 /*************************************************************************
249 |*
250 |* SvFileStream::SvFileStream()
251 |*
252 |* Beschreibung STREAM.SDW
253 |* Ersterstellung OV 08.06.94
254 |* Letzte Aenderung OV 08.06.94
255 |*
256 *************************************************************************/
257
SvFileStream(const String & rFileName,StreamMode nOpenMode)258 SvFileStream::SvFileStream( const String& rFileName, StreamMode nOpenMode )
259 {
260 bIsOpen = sal_False;
261 nLockCounter = 0;
262 bIsWritable = sal_False;
263 pInstanceData = new StreamData;
264
265 SetBufferSize( 1024 );
266 // convert URL to SystemPath, if necessary
267 ::rtl::OUString aSystemFileName;
268 if( FileBase::getSystemPathFromFileURL( rFileName , aSystemFileName )
269 != FileBase::E_None )
270 {
271 aSystemFileName = rFileName;
272 }
273 Open( aSystemFileName, nOpenMode );
274 }
275
276 /*************************************************************************
277 |*
278 |* SvFileStream::SvFileStream()
279 |*
280 |* Beschreibung STREAM.SDW
281 |* Ersterstellung OV 22.11.94
282 |* Letzte Aenderung OV 22.11.94
283 |*
284 *************************************************************************/
285
SvFileStream()286 SvFileStream::SvFileStream()
287 {
288 bIsOpen = sal_False;
289 nLockCounter = 0;
290 bIsWritable = sal_False;
291 pInstanceData = new StreamData;
292 SetBufferSize( 1024 );
293 }
294
295 /*************************************************************************
296 |*
297 |* SvFileStream::~SvFileStream()
298 |*
299 |* Beschreibung STREAM.SDW
300 |* Ersterstellung OV 22.11.94
301 |* Letzte Aenderung OV 22.11.94
302 |*
303 *************************************************************************/
304
~SvFileStream()305 SvFileStream::~SvFileStream()
306 {
307 Close();
308
309 InternalStreamLock::UnlockFile( 0, 0, this );
310
311 if (pInstanceData)
312 delete pInstanceData;
313 }
314
315 /*************************************************************************
316 |*
317 |* SvFileStream::GetFileHandle()
318 |*
319 |* Beschreibung STREAM.SDW
320 |* Ersterstellung OV 22.11.94
321 |* Letzte Aenderung OV 22.11.94
322 |*
323 *************************************************************************/
324
GetFileHandle() const325 sal_uIntPtr SvFileStream::GetFileHandle() const
326 {
327 return (sal_uInt32)pInstanceData->nHandle;
328 }
329
330 /*************************************************************************
331 |*
332 |* SvFileStream::IsA()
333 |*
334 |* Beschreibung STREAM.SDW
335 |* Ersterstellung OV 14.06.94
336 |* Letzte Aenderung OV 14.06.94
337 |*
338 *************************************************************************/
339
IsA() const340 sal_uInt16 SvFileStream::IsA() const
341 {
342 return ID_FILESTREAM;
343 }
344
345 /*************************************************************************
346 |*
347 |* SvFileStream::GetData()
348 |*
349 |* Beschreibung STREAM.SDW
350 |* Ersterstellung OV 15.06.94
351 |* Letzte Aenderung OV 15.06.94
352 |*
353 *************************************************************************/
354
GetData(void * pData,sal_Size nSize)355 sal_Size SvFileStream::GetData( void* pData, sal_Size nSize )
356 {
357 #ifdef DBG_UTIL
358 ByteString aTraceStr( "SvFileStream::GetData(): " );
359 aTraceStr += ByteString::CreateFromInt64(nSize);
360 aTraceStr += " Bytes from ";
361 aTraceStr += ByteString(aFilename, osl_getThreadTextEncoding());
362 DBG_TRACE( aTraceStr.GetBuffer() );
363 #endif
364
365 int nRead = 0;
366 if ( IsOpen() )
367 {
368 nRead = read(pInstanceData->nHandle,pData,(unsigned)nSize);
369 if ( nRead == -1 )
370 SetError( ::GetSvError( errno ));
371 }
372 return (sal_Size)nRead;
373 }
374
375 /*************************************************************************
376 |*
377 |* SvFileStream::PutData()
378 |*
379 |* Beschreibung STREAM.SDW
380 |* Ersterstellung OV 15.06.94
381 |* Letzte Aenderung OV 15.06.94
382 |*
383 *************************************************************************/
384
PutData(const void * pData,sal_Size nSize)385 sal_Size SvFileStream::PutData( const void* pData, sal_Size nSize )
386 {
387 #ifdef DBG_UTIL
388 ByteString aTraceStr( "SvFileStrean::PutData: " );
389 aTraceStr += ByteString::CreateFromInt64(nSize);
390 aTraceStr += " Bytes to ";
391 aTraceStr += ByteString(aFilename, osl_getThreadTextEncoding());
392 DBG_TRACE( aTraceStr.GetBuffer() );
393 #endif
394
395 int nWrite = 0;
396 if ( IsOpen() )
397 {
398 nWrite = write(pInstanceData->nHandle,pData,(unsigned)nSize);
399 if ( nWrite == -1 )
400 SetError( ::GetSvError( errno ) );
401 else if( !nWrite )
402 SetError( SVSTREAM_DISK_FULL );
403 }
404 return (sal_Size)nWrite;
405 }
406
407 /*************************************************************************
408 |*
409 |* SvFileStream::SeekPos()
410 |*
411 |* Beschreibung STREAM.SDW
412 |* Ersterstellung OV 15.06.94
413 |* Letzte Aenderung OV 15.06.94
414 |*
415 *************************************************************************/
416
SeekPos(sal_Size nPos)417 sal_Size SvFileStream::SeekPos( sal_Size nPos )
418 {
419 if ( IsOpen() )
420 {
421 long nNewPos;
422 if ( nPos != STREAM_SEEK_TO_END )
423 nNewPos = lseek( pInstanceData->nHandle, (long)nPos, SEEK_SET );
424 else
425 nNewPos = lseek( pInstanceData->nHandle, 0L, SEEK_END );
426
427 if ( nNewPos == -1 )
428 {
429 SetError( SVSTREAM_SEEK_ERROR );
430 return 0L;
431 }
432 // langsam aber sicherer als return nNewPos
433 return lseek(pInstanceData->nHandle,0L,SEEK_CUR);
434 // return nNewPos;
435 }
436 SetError( SVSTREAM_GENERALERROR );
437 return 0L;
438 }
439
440
441 /*************************************************************************
442 |*
443 |* SvFileStream::FlushData()
444 |*
445 |* Beschreibung STREAM.SDW
446 |* Ersterstellung OV 15.06.94
447 |* Letzte Aenderung OV 15.06.94
448 |*
449 *************************************************************************/
450
FlushData()451 void SvFileStream::FlushData()
452 {
453 // lokal gibt es nicht
454 }
455
456 static char *pFileLockEnvVar = (char*)1;
457
458 /*************************************************************************
459 |*
460 |* SvFileStream::LockRange()
461 |*
462 |* Beschreibung STREAM.SDW
463 |* Ersterstellung OV 15.06.94
464 |* Letzte Aenderung OV 15.06.94
465 |*
466 *************************************************************************/
467
LockRange(sal_Size nByteOffset,sal_Size nBytes)468 sal_Bool SvFileStream::LockRange( sal_Size nByteOffset, sal_Size nBytes )
469 {
470 struct flock aflock;
471 aflock.l_start = nByteOffset;
472 aflock.l_whence = SEEK_SET;
473 aflock.l_len = nBytes;
474
475 int nLockMode = 0;
476
477 if ( ! IsOpen() )
478 return sal_False;
479
480 if ( eStreamMode & STREAM_SHARE_DENYALL )
481 {
482 if (bIsWritable)
483 nLockMode = F_WRLCK;
484 else
485 nLockMode = F_RDLCK;
486 }
487
488 if ( eStreamMode & STREAM_SHARE_DENYREAD )
489 {
490 if (bIsWritable)
491 nLockMode = F_WRLCK;
492 else
493 {
494 SetError(SVSTREAM_LOCKING_VIOLATION);
495 return sal_False;
496 }
497 }
498
499 if ( eStreamMode & STREAM_SHARE_DENYWRITE )
500 {
501 if (bIsWritable)
502 nLockMode = F_WRLCK;
503 else
504 nLockMode = F_RDLCK;
505 }
506
507 if (!nLockMode)
508 return sal_True;
509
510 if( ! InternalStreamLock::LockFile( nByteOffset, nByteOffset+nBytes, this ) )
511 {
512 #if OSL_DEBUG_LEVEL > 1
513 fprintf( stderr, "InternalLock on %s [ %ld ... %ld ] failed\n",
514 ByteString(aFilename, osl_getThreadTextEncoding()).GetBuffer(), nByteOffset, nByteOffset+nBytes );
515 #endif
516 return sal_False;
517 }
518
519 // HACK: File-Locking nur via Environmentvariable einschalten
520 // um einen Haenger im Zusammenspiel mit einem Linux
521 // NFS-2-Server (kein Lockdaemon) zu verhindern.
522 // File-Locking ?ber NFS ist generell ein Performancekiller.
523 // HR, 22.10.1997 fuer SOLARIS
524 // CP, 30.11.1997 fuer HPUX
525 // ER, 18.12.1997 fuer IRIX
526 // HR, 18.05.1998 Environmentvariable
527
528 if ( pFileLockEnvVar == (char*)1 )
529 pFileLockEnvVar = getenv("STAR_ENABLE_FILE_LOCKING");
530 if ( ! pFileLockEnvVar )
531 return sal_True;
532
533 aflock.l_type = nLockMode;
534 if (fcntl(pInstanceData->nHandle, F_GETLK, &aflock) == -1)
535 {
536 #if ( defined HPUX && defined BAD_UNION )
537 #ifdef DBG_UTIL
538 fprintf( stderr, "***** FCNTL(lock):errno = %d\n", errno );
539 #endif
540 if ( errno == EINVAL || errno == ENOSYS )
541 return sal_True;
542 #endif
543 #if defined SINIX
544 if (errno == EINVAL)
545 return sal_True;
546 #endif
547 #if defined SOLARIS
548 if (errno == ENOSYS)
549 return sal_True;
550 #endif
551 SetError( ::GetSvError( errno ));
552 return sal_False;
553 }
554 if (aflock.l_type != F_UNLCK)
555 {
556 SetError(SVSTREAM_LOCKING_VIOLATION);
557 return sal_False;
558 }
559
560 aflock.l_type = nLockMode;
561 if (fcntl(pInstanceData->nHandle, F_SETLK, &aflock) == -1)
562 {
563 SetError( ::GetSvError( errno ));
564 return sal_False;
565 }
566 return sal_True;
567 }
568
569 /*************************************************************************
570 |*
571 |* SvFileStream::UnlockRange()
572 |*
573 |* Beschreibung STREAM.SDW
574 |* Ersterstellung OV 15.06.94
575 |* Letzte Aenderung OV 15.06.94
576 |*
577 *************************************************************************/
578
UnlockRange(sal_Size nByteOffset,sal_Size nBytes)579 sal_Bool SvFileStream::UnlockRange( sal_Size nByteOffset, sal_Size nBytes )
580 {
581
582 struct flock aflock;
583 aflock.l_type = F_UNLCK;
584 aflock.l_start = nByteOffset;
585 aflock.l_whence = SEEK_SET;
586 aflock.l_len = nBytes;
587
588 if ( ! IsOpen() )
589 return sal_False;
590
591 InternalStreamLock::UnlockFile( nByteOffset, nByteOffset+nBytes, this );
592
593 if ( ! (eStreamMode &
594 (STREAM_SHARE_DENYALL | STREAM_SHARE_DENYREAD | STREAM_SHARE_DENYWRITE)))
595 return sal_True;
596
597 // wenn File Locking ausgeschaltet, siehe SvFileStream::LockRange
598 if ( ! pFileLockEnvVar )
599 return sal_True;
600
601 if (fcntl(pInstanceData->nHandle, F_SETLK, &aflock) != -1)
602 return sal_True;
603
604 #if ( defined HPUX && defined BAD_UNION )
605 #ifdef DBG_UTIL
606 fprintf( stderr, "***** FCNTL(unlock):errno = %d\n", errno );
607 #endif
608 if ( errno == EINVAL || errno == ENOSYS )
609 return sal_True;
610 #endif
611 #if ( defined SINIX )
612 if (errno == EINVAL)
613 return sal_True;
614 #endif
615
616 SetError( ::GetSvError( errno ));
617 return sal_False;
618 }
619
620 /*************************************************************************
621 |*
622 |* SvFileStream::LockFile()
623 |*
624 |* Beschreibung STREAM.SDW
625 |* Ersterstellung OV 15.06.94
626 |* Letzte Aenderung OV 15.06.94
627 |*
628 *************************************************************************/
629
LockFile()630 sal_Bool SvFileStream::LockFile()
631 {
632 return LockRange( 0UL, 0UL );
633 }
634
635 /*************************************************************************
636 |*
637 |* SvFileStream::UnlockFile()
638 |*
639 |* Beschreibung STREAM.SDW
640 |* Ersterstellung OV 15.06.94
641 |* Letzte Aenderung OV 15.06.94
642 |*
643 *************************************************************************/
644
UnlockFile()645 sal_Bool SvFileStream::UnlockFile()
646 {
647 return UnlockRange( 0UL, 0UL );
648 }
649
650 /*************************************************************************
651 |*
652 |* SvFileStream::Open()
653 |*
654 |* Beschreibung STREAM.SDW
655 |* Ersterstellung OV 15.06.94
656 |* Letzte Aenderung OV 15.06.94
657 |*
658 *************************************************************************/
659
Open(const String & rFilename,StreamMode nOpenMode)660 void SvFileStream::Open( const String& rFilename, StreamMode nOpenMode )
661 {
662 int nAccess, nAccessRW;
663 int nMode;
664 int nHandleTmp;
665 struct stat buf;
666 sal_Bool bStatValid = sal_False;
667
668 Close();
669 errno = 0;
670 eStreamMode = nOpenMode;
671 eStreamMode &= ~STREAM_TRUNC; // beim ReOpen nicht cutten
672
673 // !!! NoOp: Ansonsten ToAbs() verwendern
674 // !!! DirEntry aDirEntry( rFilename );
675 // !!! aFilename = aDirEntry.GetFull();
676 aFilename = rFilename;
677 #ifndef BOOTSTRAP
678 FSysRedirector::DoRedirect( aFilename );
679 #endif
680 ByteString aLocalFilename(aFilename, osl_getThreadTextEncoding());
681
682 #ifdef DBG_UTIL
683 ByteString aTraceStr( "SvFileStream::Open(): " );
684 aTraceStr += aLocalFilename;
685 DBG_TRACE( aTraceStr.GetBuffer() );
686 #endif
687
688 if ( lstat( aLocalFilename.GetBuffer(), &buf ) == 0 )
689 {
690 bStatValid = sal_True;
691 // SvFileStream soll kein Directory oeffnen
692 if( S_ISDIR( buf.st_mode ) )
693 {
694 SetError( ::GetSvError( EISDIR ) );
695 return;
696 }
697 }
698
699
700 if ( !( nOpenMode & STREAM_WRITE ) )
701 nAccessRW = O_RDONLY;
702 else if ( !( nOpenMode & STREAM_READ ) )
703 nAccessRW = O_WRONLY;
704 else
705 nAccessRW = O_RDWR;
706
707 nAccess = 0;
708 // Fix (MDA, 18.01.95): Bei RD_ONLY nicht mit O_CREAT oeffnen
709 // Wichtig auf Read-Only-Dateisystemen (wie CDROM)
710 if ( (!( nOpenMode & STREAM_NOCREATE )) && ( nAccessRW != O_RDONLY ) )
711 nAccess |= O_CREAT;
712 if ( nOpenMode & STREAM_TRUNC )
713 nAccess |= O_TRUNC;
714
715 nMode = S_IREAD | S_IROTH | S_IRGRP;
716 if ( nOpenMode & STREAM_WRITE)
717 {
718 nMode |= (S_IWRITE | S_IWOTH | S_IWGRP);
719
720 if ( nOpenMode & STREAM_COPY_ON_SYMLINK )
721 {
722 if ( bStatValid && S_ISLNK( buf.st_mode ) < 0 )
723 {
724 char *pBuf = new char[ 1024+1 ];
725 if ( readlink( aLocalFilename.GetBuffer(), pBuf, 1024 ) > 0 )
726 {
727 if ( unlink(aLocalFilename.GetBuffer()) == 0 )
728 {
729 #ifdef DBG_UTIL
730 fprintf( stderr,
731 "Copying file on symbolic link (%s).\n",
732 aLocalFilename.GetBuffer() );
733 #endif
734 String aTmpString( pBuf, osl_getThreadTextEncoding() );
735 const DirEntry aSourceEntry( aTmpString );
736 const DirEntry aTargetEntry( aFilename );
737 FileCopier aFileCopier( aSourceEntry, aTargetEntry );
738 aFileCopier.Execute();
739 }
740 }
741 delete [] pBuf;
742 }
743 }
744 }
745
746
747 nHandleTmp = open(aLocalFilename.GetBuffer(),nAccessRW|nAccess, nMode );
748
749 if ( nHandleTmp == -1 )
750 {
751 if ( nAccessRW != O_RDONLY )
752 {
753 // auf Lesen runterschalten
754 nAccessRW = O_RDONLY;
755 nAccess = 0;
756 nMode = S_IREAD | S_IROTH | S_IRGRP;
757 nHandleTmp =open( aLocalFilename.GetBuffer(),
758 nAccessRW|nAccess,
759 nMode );
760 }
761 }
762 if ( nHandleTmp != -1 )
763 {
764 pInstanceData->nHandle = nHandleTmp;
765 bIsOpen = sal_True;
766 if ( nAccessRW != O_RDONLY )
767 bIsWritable = sal_True;
768
769 if ( !LockFile() ) // ganze Datei
770 {
771 close( nHandleTmp );
772 bIsOpen = sal_False;
773 bIsWritable = sal_False;
774 pInstanceData->nHandle = 0;
775 }
776 }
777 else
778 SetError( ::GetSvError( errno ) );
779 }
780
781 /*************************************************************************
782 |*
783 |* SvFileStream::ReOpen()
784 |*
785 |* Beschreibung STREAM.SDW
786 |* Ersterstellung OV 15.06.94
787 |* Letzte Aenderung OV 15.06.94
788 |*
789 *************************************************************************/
790
ReOpen()791 void SvFileStream::ReOpen()
792 {
793 if ( !bIsOpen && aFilename.Len() )
794 Open( aFilename, eStreamMode );
795 }
796
797 /*************************************************************************
798 |*
799 |* SvFileStream::Close()
800 |*
801 |* Beschreibung STREAM.SDW
802 |* Ersterstellung OV 15.06.94
803 |* Letzte Aenderung OV 15.06.94
804 |*
805 *************************************************************************/
806
Close()807 void SvFileStream::Close()
808 {
809 InternalStreamLock::UnlockFile( 0, 0, this );
810
811 if ( IsOpen() )
812 {
813 #ifdef DBG_UTIL
814 ByteString aTraceStr( "SvFileStream::Close(): " );
815 aTraceStr += ByteString(aFilename, osl_getThreadTextEncoding());
816 DBG_TRACE( aTraceStr.GetBuffer() );
817 #endif
818
819 Flush();
820 close( pInstanceData->nHandle );
821 pInstanceData->nHandle = 0;
822 }
823
824 bIsOpen = sal_False;
825 bIsWritable = sal_False;
826 SvStream::ClearBuffer();
827 SvStream::ClearError();
828 }
829
830 /*************************************************************************
831 |*
832 |* SvFileStream::ResetError()
833 |*
834 |* Beschreibung STREAM.SDW; Setzt Filepointer auf Dateianfang
835 |* Ersterstellung OV 15.06.94
836 |* Letzte Aenderung OV 15.06.94
837 |*
838 *************************************************************************/
839
ResetError()840 void SvFileStream::ResetError()
841 {
842 SvStream::ClearError();
843 }
844
845
846 /*************************************************************************
847 |*
848 |* SvFileStream::SetSize()
849 |*
850 |* Beschreibung STREAM.SDW;
851 |* Ersterstellung OV 15.06.94
852 |* Letzte Aenderung OV 15.06.94
853 |*
854 *************************************************************************/
855
SetSize(sal_Size nSize)856 void SvFileStream::SetSize (sal_Size nSize)
857 {
858 if (IsOpen())
859 {
860 int fd = pInstanceData->nHandle;
861 if (::ftruncate (fd, (off_t)nSize) < 0)
862 {
863 // Save original error.
864 sal_uInt32 nErr = ::GetSvError (errno);
865
866 // Check against current size. Fail upon 'shrink'.
867 struct stat aStat;
868 if (::fstat (fd, &aStat) < 0)
869 {
870 SetError (nErr);
871 return;
872 }
873 if ((sal::static_int_cast< sal_sSize >(nSize) <= aStat.st_size))
874 {
875 // Failure upon 'shrink'. Return original error.
876 SetError (nErr);
877 return;
878 }
879
880 // Save current position.
881 sal_Size nCurPos = (sal_Size)::lseek (fd, (off_t)0, SEEK_CUR);
882 if (nCurPos == (sal_Size)(-1))
883 {
884 SetError (nErr);
885 return;
886 }
887
888 // Try 'expand' via 'lseek()' and 'write()'.
889 if (::lseek (fd, (off_t)(nSize - 1), SEEK_SET) < 0)
890 {
891 SetError (nErr);
892 return;
893 }
894 if (::write (fd, (char*)"", (size_t)1) < 0)
895 {
896 // Failure. Restore saved position.
897 if (::lseek (fd, (off_t)nCurPos, SEEK_SET) < 0)
898 {
899 // Double failure.
900 }
901
902 SetError (nErr);
903 return;
904 }
905
906 // Success. Restore saved position.
907 if (::lseek (fd, (off_t)nCurPos, SEEK_SET) < 0)
908 {
909 SetError (nErr);
910 return;
911 }
912 }
913 }
914 }
915
916
917