| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #include "rar.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | File::File() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   hFile=FILE_BAD_HANDLE; | 
					
						
							|  |  |  |   *FileName=0; | 
					
						
							|  |  |  |   NewFile=false; | 
					
						
							|  |  |  |   LastWrite=false; | 
					
						
							|  |  |  |   HandleType=FILE_HANDLENORMAL; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   LineInput=false; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   SkipClose=false; | 
					
						
							|  |  |  |   ErrorType=FILE_SUCCESS; | 
					
						
							|  |  |  |   OpenShared=false; | 
					
						
							|  |  |  |   AllowDelete=true; | 
					
						
							|  |  |  |   AllowExceptions=true; | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |   PreserveAtime=false; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   CreateMode=FMF_UNDEFINED; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |   ReadErrorMode=FREM_ASK; | 
					
						
							|  |  |  |   TruncatedAfterReadError=false; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   CurFilePos=0; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | File::~File() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (hFile!=FILE_BAD_HANDLE && !SkipClose) | 
					
						
							|  |  |  |     if (NewFile) | 
					
						
							|  |  |  |       Delete(); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       Close(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::operator = (File &SrcFile) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   hFile=SrcFile.hFile; | 
					
						
							|  |  |  |   NewFile=SrcFile.NewFile; | 
					
						
							|  |  |  |   LastWrite=SrcFile.LastWrite; | 
					
						
							|  |  |  |   HandleType=SrcFile.HandleType; | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |   TruncatedAfterReadError=SrcFile.TruncatedAfterReadError; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   wcsncpyz(FileName,SrcFile.FileName,ASIZE(FileName)); | 
					
						
							|  |  |  |   SrcFile.SkipClose=true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Open(const wchar *Name,uint Mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ErrorType=FILE_SUCCESS; | 
					
						
							|  |  |  |   FileHandle hNewFile; | 
					
						
							|  |  |  |   bool OpenShared=File::OpenShared || (Mode & FMF_OPENSHARED)!=0; | 
					
						
							|  |  |  |   bool UpdateMode=(Mode & FMF_UPDATE)!=0; | 
					
						
							|  |  |  |   bool WriteMode=(Mode & FMF_WRITE)!=0; | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   uint Access=WriteMode ? GENERIC_WRITE:GENERIC_READ; | 
					
						
							|  |  |  |   if (UpdateMode) | 
					
						
							|  |  |  |     Access|=GENERIC_WRITE; | 
					
						
							|  |  |  |   uint ShareMode=(Mode & FMF_OPENEXCLUSIVE) ? 0 : FILE_SHARE_READ; | 
					
						
							|  |  |  |   if (OpenShared) | 
					
						
							|  |  |  |     ShareMode|=FILE_SHARE_WRITE; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   uint Flags=FILE_FLAG_SEQUENTIAL_SCAN; | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |   FindData FD; | 
					
						
							|  |  |  |   if (PreserveAtime) | 
					
						
							|  |  |  |     Access|=FILE_WRITE_ATTRIBUTES; // Needed to preserve atime.
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   hNewFile=CreateFile(Name,Access,ShareMode,NULL,OPEN_EXISTING,Flags,NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DWORD LastError; | 
					
						
							|  |  |  |   if (hNewFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     LastError=GetLastError(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     wchar LongName[NM]; | 
					
						
							|  |  |  |     if (GetWinLongPath(Name,LongName,ASIZE(LongName))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       hNewFile=CreateFile(LongName,Access,ShareMode,NULL,OPEN_EXISTING,Flags,NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // For archive names longer than 260 characters first CreateFile
 | 
					
						
							|  |  |  |       // (without \\?\) fails and sets LastError to 3 (access denied).
 | 
					
						
							|  |  |  |       // We need the correct "file not found" error code to decide
 | 
					
						
							|  |  |  |       // if we create a new archive or quit with "cannot create" error.
 | 
					
						
							|  |  |  |       // So we need to check the error code after \\?\ CreateFile again,
 | 
					
						
							|  |  |  |       // otherwise we'll fail to create new archives with long names.
 | 
					
						
							|  |  |  |       // But we cannot simply assign the new code to LastError,
 | 
					
						
							|  |  |  |       // because it would break "..\arcname.rar" relative names processing.
 | 
					
						
							|  |  |  |       // First CreateFile returns the correct "file not found" code for such
 | 
					
						
							|  |  |  |       // names, but "\\?\" CreateFile returns ERROR_INVALID_NAME treating
 | 
					
						
							|  |  |  |       // dots as a directory name. So we check only for "file not found"
 | 
					
						
							|  |  |  |       // error here and for other errors use the first CreateFile result.
 | 
					
						
							|  |  |  |       if (GetLastError()==ERROR_FILE_NOT_FOUND) | 
					
						
							|  |  |  |         LastError=ERROR_FILE_NOT_FOUND; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (hNewFile==FILE_BAD_HANDLE && LastError==ERROR_FILE_NOT_FOUND) | 
					
						
							|  |  |  |     ErrorType=FILE_NOTFOUND; | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |   if (PreserveAtime && hNewFile!=FILE_BAD_HANDLE) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     FILETIME ft={0xffffffff,0xffffffff}; // This value prevents atime modification.
 | 
					
						
							|  |  |  |     SetFileTime(hNewFile,NULL,&ft,NULL); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   int flags=UpdateMode ? O_RDWR:(WriteMode ? O_WRONLY:O_RDONLY); | 
					
						
							|  |  |  | #ifdef O_BINARY
 | 
					
						
							|  |  |  |   flags|=O_BINARY; | 
					
						
							|  |  |  | #if defined(_AIX) && defined(_LARGE_FILE_API)
 | 
					
						
							|  |  |  |   flags|=O_LARGEFILE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   // NDK r20 has O_NOATIME, but fails to create files with it in Android 7+.
 | 
					
						
							|  |  |  | #if defined(O_NOATIME)
 | 
					
						
							|  |  |  |   if (PreserveAtime) | 
					
						
							|  |  |  |     flags|=O_NOATIME; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   char NameA[NM]; | 
					
						
							|  |  |  |   WideToChar(Name,NameA,ASIZE(NameA)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int handle=open(NameA,flags); | 
					
						
							|  |  |  | #ifdef LOCK_EX
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef _OSF_SOURCE
 | 
					
						
							|  |  |  |   extern "C" int flock(int, int); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   if (!OpenShared && UpdateMode && handle>=0 && flock(handle,LOCK_EX|LOCK_NB)==-1) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     close(handle); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   if (handle==-1) | 
					
						
							|  |  |  |     hNewFile=FILE_BAD_HANDLE; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |     hNewFile=handle; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     hNewFile=fdopen(handle,UpdateMode ? UPDATEBINARY:READBINARY); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (hNewFile==FILE_BAD_HANDLE && errno==ENOENT) | 
					
						
							|  |  |  |     ErrorType=FILE_NOTFOUND; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   NewFile=false; | 
					
						
							|  |  |  |   HandleType=FILE_HANDLENORMAL; | 
					
						
							|  |  |  |   SkipClose=false; | 
					
						
							|  |  |  |   bool Success=hNewFile!=FILE_BAD_HANDLE; | 
					
						
							|  |  |  |   if (Success) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     hFile=hNewFile; | 
					
						
							|  |  |  |     wcsncpyz(FileName,Name,ASIZE(FileName)); | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |     TruncatedAfterReadError=false; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   } | 
					
						
							|  |  |  |   return Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined(SFX_MODULE)
 | 
					
						
							|  |  |  | void File::TOpen(const wchar *Name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!WOpen(Name)) | 
					
						
							|  |  |  |     ErrHandler.Exit(RARX_OPEN); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::WOpen(const wchar *Name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (Open(Name)) | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   ErrHandler.OpenErrorMsg(Name); | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Create(const wchar *Name,uint Mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // OpenIndiana based NAS and CIFS shares fail to set the file time if file
 | 
					
						
							|  |  |  |   // was created in read+write mode and some data was written and not flushed
 | 
					
						
							|  |  |  |   // before SetFileTime call. So we should use the write only mode if we plan
 | 
					
						
							|  |  |  |   // SetFileTime call and do not need to read from file.
 | 
					
						
							|  |  |  |   bool WriteMode=(Mode & FMF_WRITE)!=0; | 
					
						
							|  |  |  |   bool ShareRead=(Mode & FMF_SHAREREAD)!=0 || File::OpenShared; | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   CreateMode=Mode; | 
					
						
							|  |  |  |   uint Access=WriteMode ? GENERIC_WRITE:GENERIC_READ|GENERIC_WRITE; | 
					
						
							|  |  |  |   DWORD ShareMode=ShareRead ? FILE_SHARE_READ:0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Windows automatically removes dots and spaces in the end of file name,
 | 
					
						
							|  |  |  |   // So we detect such names and process them with \\?\ prefix.
 | 
					
						
							|  |  |  |   wchar *LastChar=PointToLastChar(Name); | 
					
						
							|  |  |  |   bool Special=*LastChar=='.' || *LastChar==' '; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   if (Special && (Mode & FMF_STANDARDNAMES)==0) | 
					
						
							|  |  |  |     hFile=FILE_BAD_HANDLE; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     hFile=CreateFile(Name,Access,ShareMode,NULL,CREATE_ALWAYS,0,NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (hFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     wchar LongName[NM]; | 
					
						
							|  |  |  |     if (GetWinLongPath(Name,LongName,ASIZE(LongName))) | 
					
						
							|  |  |  |       hFile=CreateFile(LongName,Access,ShareMode,NULL,CREATE_ALWAYS,0,NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   char NameA[NM]; | 
					
						
							|  |  |  |   WideToChar(Name,NameA,ASIZE(NameA)); | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |   hFile=open(NameA,(O_CREAT|O_TRUNC) | (WriteMode ? O_WRONLY : O_RDWR),0666); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   hFile=fopen(NameA,WriteMode ? WRITEBINARY:CREATEBINARY); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   NewFile=true; | 
					
						
							|  |  |  |   HandleType=FILE_HANDLENORMAL; | 
					
						
							|  |  |  |   SkipClose=false; | 
					
						
							|  |  |  |   wcsncpyz(FileName,Name,ASIZE(FileName)); | 
					
						
							|  |  |  |   return hFile!=FILE_BAD_HANDLE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined(SFX_MODULE)
 | 
					
						
							|  |  |  | void File::TCreate(const wchar *Name,uint Mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!WCreate(Name,Mode)) | 
					
						
							|  |  |  |     ErrHandler.Exit(RARX_FATAL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::WCreate(const wchar *Name,uint Mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (Create(Name,Mode)) | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   ErrHandler.CreateErrorMsg(Name); | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Close() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   bool Success=true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (hFile!=FILE_BAD_HANDLE) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (!SkipClose) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |       // We use the standard system handle for stdout in Windows
 | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |       // and it must not be closed here.
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |       if (HandleType==FILE_HANDLENORMAL) | 
					
						
							|  |  |  |         Success=CloseHandle(hFile)==TRUE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |       Success=close(hFile)!=-1; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |       Success=fclose(hFile)!=EOF; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     hFile=FILE_BAD_HANDLE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   HandleType=FILE_HANDLENORMAL; | 
					
						
							|  |  |  |   if (!Success && AllowExceptions) | 
					
						
							|  |  |  |     ErrHandler.CloseError(FileName); | 
					
						
							|  |  |  |   return Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Delete() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (HandleType!=FILE_HANDLENORMAL) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   if (hFile!=FILE_BAD_HANDLE) | 
					
						
							|  |  |  |     Close(); | 
					
						
							|  |  |  |   if (!AllowDelete) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   return DelFile(FileName); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Rename(const wchar *NewName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // No need to rename if names are already same.
 | 
					
						
							|  |  |  |   bool Success=wcscmp(FileName,NewName)==0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!Success) | 
					
						
							|  |  |  |     Success=RenameFile(FileName,NewName); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (Success) | 
					
						
							| 
									
										
										
										
											2019-06-12 23:28:12 -04:00
										 |  |  |     wcsncpyz(FileName,NewName,ASIZE(FileName)); | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Write(const void *Data,size_t Size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (Size==0) | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   if (HandleType==FILE_HANDLESTD) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |     hFile=GetStdHandle(STD_OUTPUT_HANDLE); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     // Cannot use the standard stdout here, because it already has wide orientation.
 | 
					
						
							|  |  |  |     if (hFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |       hFile=dup(STDOUT_FILENO); // Open new stdout stream.
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |       hFile=fdopen(dup(STDOUT_FILENO),"w"); // Open new stdout stream.
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   bool Success; | 
					
						
							|  |  |  |   while (1) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     Success=false; | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |     DWORD Written=0; | 
					
						
							|  |  |  |     if (HandleType!=FILE_HANDLENORMAL) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // writing to stdout can fail in old Windows if data block is too large
 | 
					
						
							|  |  |  |       const size_t MaxSize=0x4000; | 
					
						
							|  |  |  |       for (size_t I=0;I<Size;I+=MaxSize) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         Success=WriteFile(hFile,(byte *)Data+I,(DWORD)Min(Size-I,MaxSize),&Written,NULL)==TRUE; | 
					
						
							|  |  |  |         if (!Success) | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       Success=WriteFile(hFile,Data,(DWORD)Size,&Written,NULL)==TRUE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |     ssize_t Written=write(hFile,Data,Size); | 
					
						
							|  |  |  |     Success=Written==Size; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     int Written=fwrite(Data,1,Size,hFile); | 
					
						
							|  |  |  |     Success=Written==Size && !ferror(hFile); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if (!Success && AllowExceptions && HandleType==FILE_HANDLENORMAL) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | #if defined(_WIN_ALL) && !defined(SFX_MODULE) && !defined(RARDLL)
 | 
					
						
							|  |  |  |       int ErrCode=GetLastError(); | 
					
						
							|  |  |  |       int64 FilePos=Tell(); | 
					
						
							|  |  |  |       uint64 FreeSize=GetFreeDisk(FileName); | 
					
						
							|  |  |  |       SetLastError(ErrCode); | 
					
						
							|  |  |  |       if (FreeSize>Size && FilePos-Size<=0xffffffff && FilePos+Size>0xffffffff) | 
					
						
							|  |  |  |         ErrHandler.WriteErrorFAT(FileName); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       if (ErrHandler.AskRepeatWrite(FileName,false)) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | #if !defined(_WIN_ALL) && !defined(FILE_USE_OPEN)
 | 
					
						
							|  |  |  |         clearerr(hFile); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         if (Written<Size && Written>0) | 
					
						
							|  |  |  |           Seek(Tell()-Written,SEEK_SET); | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       ErrHandler.WriteError(NULL,FileName); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   LastWrite=true; | 
					
						
							|  |  |  |   return Success; // It can return false only if AllowExceptions is disabled.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int File::Read(void *Data,size_t Size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |   if (TruncatedAfterReadError) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   int64 FilePos=0; // Initialized only to suppress some compilers warning.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |   if (ReadErrorMode==FREM_IGNORE) | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |     FilePos=Tell(); | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   int TotalRead=0; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   while (true) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |     int ReadSize=DirectRead(Data,Size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |     if (ReadSize==-1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ErrorType=FILE_READERROR; | 
					
						
							|  |  |  |       if (AllowExceptions) | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |         if (ReadErrorMode==FREM_IGNORE) | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |         { | 
					
						
							|  |  |  |           ReadSize=0; | 
					
						
							|  |  |  |           for (size_t I=0;I<Size;I+=512) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             Seek(FilePos+I,SEEK_SET); | 
					
						
							|  |  |  |             size_t SizeToRead=Min(Size-I,512); | 
					
						
							|  |  |  |             int ReadCode=DirectRead(Data,SizeToRead); | 
					
						
							|  |  |  |             ReadSize+=(ReadCode==-1) ? 512:ReadCode; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |             if (ReadSize!=-1) | 
					
						
							|  |  |  |               TotalRead+=ReadSize; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-05-03 13:03:33 -07:00
										 |  |  |           bool Ignore=false,Retry=false,Quit=false; | 
					
						
							|  |  |  |           if (ReadErrorMode==FREM_ASK && HandleType==FILE_HANDLENORMAL) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             ErrHandler.AskRepeatRead(FileName,Ignore,Retry,Quit); | 
					
						
							|  |  |  |             if (Retry) | 
					
						
							|  |  |  |               continue; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (Ignore || ReadErrorMode==FREM_TRUNCATE) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             TruncatedAfterReadError=true; | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |           ErrHandler.ReadError(FileName); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |     TotalRead+=ReadSize; // If ReadSize is -1, TotalRead is also set to -1 here.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (HandleType==FILE_HANDLESTD && !LineInput && ReadSize>0 && (uint)ReadSize<Size) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // Unlike regular files, for pipe we can read only as much as was
 | 
					
						
							|  |  |  |       // written at the other end of pipe. We had seen data coming in small
 | 
					
						
							|  |  |  |       // ~80 byte chunks when piping from 'type arc.rar'. Extraction code
 | 
					
						
							|  |  |  |       // would fail if we read an incomplete archive header from stdin.
 | 
					
						
							|  |  |  |       // So here we ensure that requested size is completely read.
 | 
					
						
							|  |  |  |       // But we return the available data immediately in "line input" mode,
 | 
					
						
							|  |  |  |       // when processing user's input in console prompts. Otherwise apps
 | 
					
						
							|  |  |  |       // piping user responses to multiple Ask() prompts can hang if no more
 | 
					
						
							|  |  |  |       // data is available yet and pipe isn't closed.
 | 
					
						
							|  |  |  |       Data=(byte*)Data+ReadSize; | 
					
						
							|  |  |  |       Size-=ReadSize; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   if (TotalRead>0) // Can be -1 for error and AllowExceptions disabled.
 | 
					
						
							|  |  |  |     CurFilePos+=TotalRead; | 
					
						
							|  |  |  |   return TotalRead; // It can return -1 only if AllowExceptions is disabled.
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Returns -1 in case of error.
 | 
					
						
							|  |  |  | int File::DirectRead(void *Data,size_t Size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   const size_t MaxDeviceRead=20000; | 
					
						
							|  |  |  |   const size_t MaxLockedRead=32768; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   if (HandleType==FILE_HANDLESTD) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  | //    if (Size>MaxDeviceRead)
 | 
					
						
							|  |  |  | //      Size=MaxDeviceRead;
 | 
					
						
							|  |  |  |     hFile=GetStdHandle(STD_INPUT_HANDLE); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |     hFile=STDIN_FILENO; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     hFile=stdin; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   // For pipes like 'type file.txt | rar -si arcname' ReadFile may return
 | 
					
						
							|  |  |  |   // data in small ~4KB blocks. It may slightly reduce the compression ratio.
 | 
					
						
							|  |  |  |   DWORD Read; | 
					
						
							|  |  |  |   if (!ReadFile(hFile,Data,(DWORD)Size,&Read,NULL)) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (IsDevice() && Size>MaxDeviceRead) | 
					
						
							|  |  |  |       return DirectRead(Data,MaxDeviceRead); | 
					
						
							|  |  |  |     if (HandleType==FILE_HANDLESTD && GetLastError()==ERROR_BROKEN_PIPE) | 
					
						
							|  |  |  |       return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We had a bug report about failure to archive 1C database lock file
 | 
					
						
							|  |  |  |     // 1Cv8tmp.1CL, which is a zero length file with a region above 200 KB
 | 
					
						
							|  |  |  |     // permanently locked. If our first read request uses too large buffer
 | 
					
						
							|  |  |  |     // and if we are in -dh mode, so we were able to open the file,
 | 
					
						
							|  |  |  |     // we'll fail with "Read error". So now we use try a smaller buffer size
 | 
					
						
							|  |  |  |     // in case of lock error.
 | 
					
						
							|  |  |  |     if (HandleType==FILE_HANDLENORMAL && Size>MaxLockedRead && | 
					
						
							|  |  |  |         GetLastError()==ERROR_LOCK_VIOLATION) | 
					
						
							|  |  |  |       return DirectRead(Data,MaxLockedRead); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return Read; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |   ssize_t ReadSize=read(hFile,Data,Size); | 
					
						
							|  |  |  |   if (ReadSize==-1) | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   return (int)ReadSize; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (LastWrite) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     fflush(hFile); | 
					
						
							|  |  |  |     LastWrite=false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   clearerr(hFile); | 
					
						
							|  |  |  |   size_t ReadSize=fread(Data,1,Size,hFile); | 
					
						
							|  |  |  |   if (ferror(hFile)) | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   return (int)ReadSize; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::Seek(int64 Offset,int Method) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!RawSeek(Offset,Method) && AllowExceptions) | 
					
						
							|  |  |  |     ErrHandler.SeekError(FileName); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::RawSeek(int64 Offset,int Method) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (hFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |   if (!IsSeekable()) // To extract archives from stdin with -si.
 | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |     // We tried to dynamically allocate 32 KB buffer here, but it improved
 | 
					
						
							|  |  |  |     // speed in Windows 10 by mere ~1.5%.
 | 
					
						
							|  |  |  |     byte Buf[4096]; | 
					
						
							|  |  |  |     if (Method==SEEK_CUR || Method==SEEK_SET && Offset>=CurFilePos) | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |       uint64 SkipSize=Method==SEEK_CUR ? Offset:Offset-CurFilePos; | 
					
						
							|  |  |  |       while (SkipSize>0) // Reading to emulate seek forward.
 | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |       { | 
					
						
							|  |  |  |         int ReadSize=Read(Buf,(size_t)Min(SkipSize,ASIZE(Buf))); | 
					
						
							|  |  |  |         if (ReadSize<=0) | 
					
						
							|  |  |  |           return false; | 
					
						
							|  |  |  |         SkipSize-=ReadSize; | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |         CurFilePos+=ReadSize; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // May need it in FileLength() in Archive::UnexpEndArcMsg() when unpacking
 | 
					
						
							|  |  |  |     // RAR 4.x archives without the end of archive block created with -en.
 | 
					
						
							|  |  |  |     if (Method==SEEK_END) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       int ReadSize; | 
					
						
							|  |  |  |       while ((ReadSize=Read(Buf,ASIZE(Buf)))>0) | 
					
						
							|  |  |  |         CurFilePos+=ReadSize; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |     return false; // Backward seek on unseekable file.
 | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   if (Offset<0 && Method!=SEEK_SET) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     Offset=(Method==SEEK_CUR ? Tell():FileLength())+Offset; | 
					
						
							|  |  |  |     Method=SEEK_SET; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   LONG HighDist=(LONG)(Offset>>32); | 
					
						
							|  |  |  |   if (SetFilePointer(hFile,(LONG)Offset,&HighDist,Method)==0xffffffff && | 
					
						
							|  |  |  |       GetLastError()!=NO_ERROR) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   LastWrite=false; | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |   if (lseek(hFile,(off_t)Offset,Method)==-1) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #elif defined(_LARGEFILE_SOURCE) && !defined(_OSF_SOURCE) && !defined(__VMS)
 | 
					
						
							|  |  |  |   if (fseeko(hFile,Offset,Method)!=0) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (fseek(hFile,(long)Offset,Method)!=0) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int64 File::Tell() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (hFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |     if (AllowExceptions) | 
					
						
							|  |  |  |       ErrHandler.SeekError(FileName); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       return -1; | 
					
						
							| 
									
										
										
										
											2022-07-14 17:09:53 -07:00
										 |  |  |   if (!IsSeekable()) | 
					
						
							|  |  |  |     return CurFilePos; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   LONG HighDist=0; | 
					
						
							|  |  |  |   uint LowDist=SetFilePointer(hFile,0,&HighDist,FILE_CURRENT); | 
					
						
							|  |  |  |   if (LowDist==0xffffffff && GetLastError()!=NO_ERROR) | 
					
						
							|  |  |  |     if (AllowExceptions) | 
					
						
							|  |  |  |       ErrHandler.SeekError(FileName); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  |   return INT32TO64(HighDist,LowDist); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef FILE_USE_OPEN
 | 
					
						
							|  |  |  |   return lseek(hFile,0,SEEK_CUR); | 
					
						
							|  |  |  | #elif defined(_LARGEFILE_SOURCE) && !defined(_OSF_SOURCE)
 | 
					
						
							|  |  |  |   return ftello(hFile); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return ftell(hFile); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::Prealloc(int64 Size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   if (RawSeek(Size,SEEK_SET)) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     Truncate(); | 
					
						
							|  |  |  |     Seek(0,SEEK_SET); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(_UNIX) && defined(USE_FALLOCATE)
 | 
					
						
							|  |  |  |   // fallocate is rather new call. Only latest kernels support it.
 | 
					
						
							|  |  |  |   // So we are not using it by default yet.
 | 
					
						
							|  |  |  |   int fd = GetFD(); | 
					
						
							|  |  |  |   if (fd >= 0) | 
					
						
							|  |  |  |     fallocate(fd, 0, 0, Size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | byte File::GetByte() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   byte Byte=0; | 
					
						
							|  |  |  |   Read(&Byte,1); | 
					
						
							|  |  |  |   return Byte; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::PutByte(byte Byte) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Write(&Byte,1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::Truncate() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   return SetEndOfFile(hFile)==TRUE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return ftruncate(GetFD(),(off_t)Tell())==0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::Flush() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   FlushFileBuffers(hFile); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifndef FILE_USE_OPEN
 | 
					
						
							|  |  |  |   fflush(hFile); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   fsync(GetFD()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::SetOpenFileTime(RarTime *ftm,RarTime *ftc,RarTime *fta) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   // Workaround for OpenIndiana NAS time bug. If we cannot create a file
 | 
					
						
							|  |  |  |   // in write only mode, we need to flush the write buffer before calling
 | 
					
						
							|  |  |  |   // SetFileTime or file time will not be changed.
 | 
					
						
							|  |  |  |   if (CreateMode!=FMF_UNDEFINED && (CreateMode & FMF_WRITE)==0) | 
					
						
							|  |  |  |     FlushFileBuffers(hFile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool sm=ftm!=NULL && ftm->IsSet(); | 
					
						
							|  |  |  |   bool sc=ftc!=NULL && ftc->IsSet(); | 
					
						
							|  |  |  |   bool sa=fta!=NULL && fta->IsSet(); | 
					
						
							|  |  |  |   FILETIME fm,fc,fa; | 
					
						
							|  |  |  |   if (sm) | 
					
						
							|  |  |  |     ftm->GetWinFT(&fm); | 
					
						
							|  |  |  |   if (sc) | 
					
						
							|  |  |  |     ftc->GetWinFT(&fc); | 
					
						
							|  |  |  |   if (sa) | 
					
						
							|  |  |  |     fta->GetWinFT(&fa); | 
					
						
							|  |  |  |   SetFileTime(hFile,sc ? &fc:NULL,sa ? &fa:NULL,sm ? &fm:NULL); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::SetCloseFileTime(RarTime *ftm,RarTime *fta) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | // Android APP_PLATFORM := android-14 does not support futimens and futimes.
 | 
					
						
							|  |  |  | // Newer platforms support futimens, but fail on Android 4.2.
 | 
					
						
							|  |  |  | // We have to use utime for Android.
 | 
					
						
							|  |  |  | // Also we noticed futimens fail to set timestamps on NTFS partition
 | 
					
						
							|  |  |  | // mounted to virtual Linux x86 machine, but utimensat worked correctly.
 | 
					
						
							|  |  |  | // So we set timestamps for already closed files in Unix.
 | 
					
						
							|  |  |  | #ifdef _UNIX
 | 
					
						
							|  |  |  |   SetCloseFileTimeByName(FileName,ftm,fta); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::SetCloseFileTimeByName(const wchar *Name,RarTime *ftm,RarTime *fta) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _UNIX
 | 
					
						
							|  |  |  |   bool setm=ftm!=NULL && ftm->IsSet(); | 
					
						
							|  |  |  |   bool seta=fta!=NULL && fta->IsSet(); | 
					
						
							|  |  |  |   if (setm || seta) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     char NameA[NM]; | 
					
						
							|  |  |  |     WideToChar(Name,NameA,ASIZE(NameA)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef UNIX_TIME_NS
 | 
					
						
							|  |  |  |     timespec times[2]; | 
					
						
							|  |  |  |     times[0].tv_sec=seta ? fta->GetUnix() : 0; | 
					
						
							|  |  |  |     times[0].tv_nsec=seta ? long(fta->GetUnixNS()%1000000000) : UTIME_NOW; | 
					
						
							|  |  |  |     times[1].tv_sec=setm ? ftm->GetUnix() : 0; | 
					
						
							|  |  |  |     times[1].tv_nsec=setm ? long(ftm->GetUnixNS()%1000000000) : UTIME_NOW; | 
					
						
							|  |  |  |     utimensat(AT_FDCWD,NameA,times,0); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     utimbuf ut; | 
					
						
							|  |  |  |     if (setm) | 
					
						
							|  |  |  |       ut.modtime=ftm->GetUnix(); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       ut.modtime=fta->GetUnix(); // Need to set something, cannot left it 0.
 | 
					
						
							|  |  |  |     if (seta) | 
					
						
							|  |  |  |       ut.actime=fta->GetUnix(); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       ut.actime=ut.modtime; // Need to set something, cannot left it 0.
 | 
					
						
							|  |  |  |     utime(NameA,&ut); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  | #ifdef _UNIX
 | 
					
						
							|  |  |  | void File::StatToRarTime(struct stat &st,RarTime *ftm,RarTime *ftc,RarTime *fta) | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  | #ifdef UNIX_TIME_NS
 | 
					
						
							|  |  |  | #if defined(_APPLE)
 | 
					
						
							|  |  |  |   if (ftm!=NULL) ftm->SetUnixNS(st.st_mtimespec.tv_sec*(uint64)1000000000+st.st_mtimespec.tv_nsec); | 
					
						
							|  |  |  |   if (ftc!=NULL) ftc->SetUnixNS(st.st_ctimespec.tv_sec*(uint64)1000000000+st.st_ctimespec.tv_nsec); | 
					
						
							|  |  |  |   if (fta!=NULL) fta->SetUnixNS(st.st_atimespec.tv_sec*(uint64)1000000000+st.st_atimespec.tv_nsec); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (ftm!=NULL) ftm->SetUnixNS(st.st_mtim.tv_sec*(uint64)1000000000+st.st_mtim.tv_nsec); | 
					
						
							|  |  |  |   if (ftc!=NULL) ftc->SetUnixNS(st.st_ctim.tv_sec*(uint64)1000000000+st.st_ctim.tv_nsec); | 
					
						
							|  |  |  |   if (fta!=NULL) fta->SetUnixNS(st.st_atim.tv_sec*(uint64)1000000000+st.st_atim.tv_nsec); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (ftm!=NULL) ftm->SetUnix(st.st_mtime); | 
					
						
							|  |  |  |   if (ftc!=NULL) ftc->SetUnix(st.st_ctime); | 
					
						
							|  |  |  |   if (fta!=NULL) fta->SetUnix(st.st_atime); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void File::GetOpenFileTime(RarTime *ftm,RarTime *ftc,RarTime *fta) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   FILETIME ctime,atime,mtime; | 
					
						
							|  |  |  |   GetFileTime(hFile,&ctime,&atime,&mtime); | 
					
						
							|  |  |  |   if (ftm!=NULL) ftm->SetWinFT(&mtime); | 
					
						
							|  |  |  |   if (ftc!=NULL) ftc->SetWinFT(&ctime); | 
					
						
							|  |  |  |   if (fta!=NULL) fta->SetWinFT(&atime); | 
					
						
							|  |  |  | #elif defined(_UNIX)
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   struct stat st; | 
					
						
							|  |  |  |   fstat(GetFD(),&st); | 
					
						
							| 
									
										
										
										
											2023-08-22 10:26:55 -07:00
										 |  |  |   StatToRarTime(st,ftm,ftc,fta); | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int64 File::FileLength() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |   int64 SavePos=Tell(); | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  |   Seek(0,SEEK_END); | 
					
						
							| 
									
										
										
										
											2020-05-01 14:22:17 -07:00
										 |  |  |   int64 Length=Tell(); | 
					
						
							|  |  |  |   Seek(SavePos,SEEK_SET); | 
					
						
							|  |  |  |   return Length; | 
					
						
							| 
									
										
										
										
											2018-07-30 20:19:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool File::IsDevice() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (hFile==FILE_BAD_HANDLE) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |   uint Type=GetFileType(hFile); | 
					
						
							|  |  |  |   return Type==FILE_TYPE_CHAR || Type==FILE_TYPE_PIPE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return isatty(GetFD()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef SFX_MODULE
 | 
					
						
							|  |  |  | int64 File::Copy(File &Dest,int64 Length) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Array<byte> Buffer(File::CopyBufferSize()); | 
					
						
							|  |  |  |   int64 CopySize=0; | 
					
						
							|  |  |  |   bool CopyAll=(Length==INT64NDF); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (CopyAll || Length>0) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     Wait(); | 
					
						
							|  |  |  |     size_t SizeToRead=(!CopyAll && Length<(int64)Buffer.Size()) ? (size_t)Length:Buffer.Size(); | 
					
						
							|  |  |  |     byte *Buf=&Buffer[0]; | 
					
						
							|  |  |  |     int ReadSize=Read(Buf,SizeToRead); | 
					
						
							|  |  |  |     if (ReadSize==0) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     size_t WriteSize=ReadSize; | 
					
						
							|  |  |  | #ifdef _WIN_ALL
 | 
					
						
							|  |  |  |     // For FAT32 USB flash drives in Windows if first write is 4 KB or more,
 | 
					
						
							|  |  |  |     // write caching is disabled and "write through" is enabled, resulting
 | 
					
						
							|  |  |  |     // in bad performance, especially for many small files. It happens when
 | 
					
						
							|  |  |  |     // we create SFX archive on USB drive, because SFX module is written first.
 | 
					
						
							|  |  |  |     // So we split the first write to small 1 KB followed by rest of data.
 | 
					
						
							|  |  |  |     if (CopySize==0 && WriteSize>=4096) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       const size_t FirstWrite=1024; | 
					
						
							|  |  |  |       Dest.Write(Buf,FirstWrite); | 
					
						
							|  |  |  |       Buf+=FirstWrite; | 
					
						
							|  |  |  |       WriteSize-=FirstWrite; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     Dest.Write(Buf,WriteSize); | 
					
						
							|  |  |  |     CopySize+=ReadSize; | 
					
						
							|  |  |  |     if (!CopyAll) | 
					
						
							|  |  |  |       Length-=ReadSize; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return CopySize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 |