1 /* 2 * Copyright 2012, The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include "FileBase.h" 18 #include "Log.h" 19 20 #include <sys/file.h> 21 #include <sys/stat.h> 22 #include <unistd.h> 23 24 #include <cerrno> 25 #include <cstring> 26 #include <new> 27 28 using namespace bcc; 29 30 #ifdef _WIN32 31 // TODO: Fix flock usage under windows 32 #define LOCK_SH 0 33 #define LOCK_EX 0 34 #define LOCK_NB 0 35 #define LOCK_UN 0 36 37 int flock(int fd, int operation) { 38 return 0; 39 } 40 #endif // _WIN32 41 42 FileBase::FileBase(const std::string &pFilename, 43 unsigned pOpenFlags, 44 unsigned pFlags) 45 : mFD(-1), 46 mError(), 47 mName(pFilename), mOpenFlags(pOpenFlags), 48 mShouldUnlock(false), 49 mShouldDelete(false) { 50 // Process pFlags 51 #ifdef O_BINARY 52 if (pFlags & kBinary) { 53 mOpenFlags |= O_BINARY; 54 } 55 #endif 56 if (pFlags & kTruncate) { 57 mOpenFlags |= O_TRUNC; 58 } 59 60 if (pFlags & kAppend) { 61 mOpenFlags |= O_APPEND; 62 } 63 64 if (pFlags & kDeleteOnClose) { 65 mShouldDelete = true; 66 } 67 68 // Open the file. 69 open(); 70 71 return; 72 } 73 74 FileBase::~FileBase() { 75 close(); 76 } 77 78 bool FileBase::open() { 79 do { 80 // FIXME: Hard-coded permissions (0644) for newly created file should be 81 // removed and provide a way to let the user configure the value. 82 mFD = ::open(mName.c_str(), mOpenFlags, 0644); 83 if (mFD > 0) { 84 return true; 85 } 86 87 // Some errors occurred ... 88 if (errno != EINTR) { 89 detectError(); 90 return false; 91 } 92 } while (true); 93 // unreachable 94 } 95 96 97 bool FileBase::checkFileIntegrity() { 98 // Check the file integrity by examining whether the inode referring to the mFD 99 // and to the file mName are the same. 100 struct stat fd_stat, file_stat; 101 102 // Get the file status of file descriptor mFD. 103 do { 104 if (::fstat(mFD, &fd_stat) == 0) { 105 break; 106 } else if (errno != EINTR) { 107 detectError(); 108 return false; 109 } 110 } while (true); 111 112 // Get the file status of file mName. 113 do { 114 if (::stat(mName.c_str(), &file_stat) == 0) { 115 break; 116 } else if (errno != EINTR) { 117 detectError(); 118 return false; 119 } 120 } while (true); 121 122 return ((fd_stat.st_dev == file_stat.st_dev) && 123 (fd_stat.st_ino == file_stat.st_ino)); 124 } 125 126 void FileBase::detectError() { 127 // Read error from errno. 128 mError.assign(errno, std::generic_category()); 129 } 130 131 bool FileBase::lock(enum LockModeEnum pMode, 132 bool pNonblocking, 133 unsigned pMaxRetry, 134 useconds_t pRetryInterval) { 135 int lock_operation; 136 unsigned retry = 0; 137 138 // Check the state. 139 if ((mFD < 0) || hasError()) { 140 return false; 141 } 142 143 // Return immediately if it's already locked. 144 if (mShouldUnlock) { 145 return true; 146 } 147 148 // Determine the lock operation (2nd argument) to the flock(). 149 if (pMode == kReadLock) { 150 lock_operation = LOCK_SH; 151 } else if (pMode == kWriteLock) { 152 lock_operation = LOCK_EX; 153 } else { 154 mError = std::make_error_code(std::errc::invalid_argument); 155 return false; 156 } 157 158 if (pNonblocking) { 159 lock_operation |= LOCK_NB; 160 } 161 162 do { 163 if (::flock(mFD, lock_operation) == 0) { 164 mShouldUnlock = true; 165 // Here we got a lock but we need to check whether the mFD still 166 // "represents" the filename (mName) we opened in the contructor. This 167 // check may failed when another process deleted the original file mFD 168 // mapped when we were trying to obtain the lock on the file. 169 if (!checkFileIntegrity()) { 170 if (hasError() || !reopen()) { 171 // Error occurred when check the file integrity or re-open the file. 172 return false; 173 } else { 174 // Wait a while before the next try. 175 ::usleep(pRetryInterval); 176 retry++; 177 continue; 178 } 179 } 180 181 return true; 182 } 183 184 // flock() was not performed successfully. Check the errno to see whether 185 // it's retry-able. 186 if (errno == EINTR) { 187 // flock() was interrupted by delivery of a signal. Restart without 188 // decrement the retry counter. 189 continue; 190 } else if (errno == EWOULDBLOCK) { 191 // The file descriptor was locked by others, wait for a while before next 192 // retry. 193 retry++; 194 ::usleep(pRetryInterval); 195 } else { 196 // There's a fatal error occurs when perform flock(). Return immediately 197 // without further retry. 198 detectError(); 199 return false; 200 } 201 } while (retry <= pMaxRetry); 202 203 return false; 204 } 205 206 void FileBase::unlock() { 207 if (mFD < 0) { 208 return; 209 } 210 211 do { 212 if (::flock(mFD, LOCK_UN) == 0) { 213 mShouldUnlock = false; 214 return; 215 } 216 } while (errno == EINTR); 217 218 detectError(); 219 return; 220 } 221 222 void FileBase::close() { 223 if (mShouldUnlock) { 224 unlock(); 225 mShouldUnlock = false; 226 } 227 if (mFD > 0) { 228 ::close(mFD); 229 mFD = -1; 230 } 231 if (mShouldDelete) { 232 int res = ::remove(mName.c_str()); 233 if (res != 0) { 234 ALOGE("Failed to remove file: %s - %s", mName.c_str(), ::strerror(res)); 235 } 236 } 237 return; 238 } 239