OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 #include "platform/globals.h" | 5 #include "platform/globals.h" |
6 #if defined(TARGET_OS_LINUX) | 6 #if defined(TARGET_OS_LINUX) |
7 | 7 |
8 #include "bin/file.h" | 8 #include "bin/file.h" |
9 | 9 |
10 #include <errno.h> // NOLINT | 10 #include <errno.h> // NOLINT |
(...skipping 248 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
259 FDUtils::SaveErrorAndClose(fd); | 259 FDUtils::SaveErrorAndClose(fd); |
260 return is_file; | 260 return is_file; |
261 } | 261 } |
262 | 262 |
263 | 263 |
264 bool File::CreateLink(const char* name, const char* target) { | 264 bool File::CreateLink(const char* name, const char* target) { |
265 return NO_RETRY_EXPECTED(symlink(target, name)) == 0; | 265 return NO_RETRY_EXPECTED(symlink(target, name)) == 0; |
266 } | 266 } |
267 | 267 |
268 | 268 |
269 bool File::Delete(const char* name) { | 269 File::Type File::GetType(const char* pathname, bool follow_links) { |
270 File::Type type = File::GetType(name, true); | 270 struct stat64 entry_info; |
271 if (type == kIsFile) { | 271 int stat_success; |
272 return NO_RETRY_EXPECTED(unlink(name)) == 0; | 272 if (follow_links) { |
273 } else if (type == kIsDirectory) { | 273 stat_success = TEMP_FAILURE_RETRY(stat64(pathname, &entry_info)); |
274 errno = EISDIR; | |
275 } else { | 274 } else { |
276 errno = ENOENT; | 275 stat_success = TEMP_FAILURE_RETRY(lstat64(pathname, &entry_info)); |
| 276 } |
| 277 if (stat_success == -1) { |
| 278 return File::kDoesNotExist; |
| 279 } |
| 280 if (S_ISDIR(entry_info.st_mode)) { |
| 281 return File::kIsDirectory; |
| 282 } |
| 283 if (S_ISREG(entry_info.st_mode)) { |
| 284 return File::kIsFile; |
| 285 } |
| 286 if (S_ISLNK(entry_info.st_mode)) { |
| 287 return File::kIsLink; |
| 288 } |
| 289 return File::kDoesNotExist; |
| 290 } |
| 291 |
| 292 |
| 293 static bool CheckTypeAndSetErrno(const char* name, |
| 294 File::Type expected, |
| 295 bool follow_links) { |
| 296 File::Type actual = File::GetType(name, follow_links); |
| 297 if (actual == expected) { |
| 298 return true; |
| 299 } |
| 300 switch (actual) { |
| 301 case File::kIsDirectory: |
| 302 errno = EISDIR; |
| 303 break; |
| 304 case File::kDoesNotExist: |
| 305 errno = ENOENT; |
| 306 break; |
| 307 default: |
| 308 errno = EINVAL; |
| 309 break; |
277 } | 310 } |
278 return false; | 311 return false; |
279 } | 312 } |
280 | 313 |
281 | 314 |
| 315 bool File::Delete(const char* name) { |
| 316 return CheckTypeAndSetErrno(name, kIsFile, true) && |
| 317 (NO_RETRY_EXPECTED(unlink(name)) == 0); |
| 318 } |
| 319 |
| 320 |
282 bool File::DeleteLink(const char* name) { | 321 bool File::DeleteLink(const char* name) { |
283 File::Type type = File::GetType(name, false); | 322 return CheckTypeAndSetErrno(name, kIsLink, false) && |
284 if (type == kIsLink) { | 323 (NO_RETRY_EXPECTED(unlink(name)) == 0); |
285 return NO_RETRY_EXPECTED(unlink(name)) == 0; | |
286 } | |
287 errno = EINVAL; | |
288 return false; | |
289 } | 324 } |
290 | 325 |
291 | 326 |
292 bool File::Rename(const char* old_path, const char* new_path) { | 327 bool File::Rename(const char* old_path, const char* new_path) { |
293 File::Type type = File::GetType(old_path, true); | 328 return CheckTypeAndSetErrno(old_path, kIsFile, true) && |
294 if (type == kIsFile) { | 329 (NO_RETRY_EXPECTED(rename(old_path, new_path)) == 0); |
295 return NO_RETRY_EXPECTED(rename(old_path, new_path)) == 0; | |
296 } else if (type == kIsDirectory) { | |
297 errno = EISDIR; | |
298 } else { | |
299 errno = ENOENT; | |
300 } | |
301 return false; | |
302 } | 330 } |
303 | 331 |
304 | 332 |
305 bool File::RenameLink(const char* old_path, const char* new_path) { | 333 bool File::RenameLink(const char* old_path, const char* new_path) { |
306 File::Type type = File::GetType(old_path, false); | 334 return CheckTypeAndSetErrno(old_path, kIsLink, false) && |
307 if (type == kIsLink) { | 335 (NO_RETRY_EXPECTED(rename(old_path, new_path)) == 0); |
308 return NO_RETRY_EXPECTED(rename(old_path, new_path)) == 0; | |
309 } else if (type == kIsDirectory) { | |
310 errno = EISDIR; | |
311 } else { | |
312 errno = EINVAL; | |
313 } | |
314 return false; | |
315 } | 336 } |
316 | 337 |
317 | 338 |
318 bool File::Copy(const char* old_path, const char* new_path) { | 339 bool File::Copy(const char* old_path, const char* new_path) { |
319 File::Type type = File::GetType(old_path, true); | 340 if (!CheckTypeAndSetErrno(old_path, kIsFile, true)) { |
320 if (type == kIsFile) { | 341 return false; |
321 struct stat64 st; | 342 } |
322 if (TEMP_FAILURE_RETRY(stat64(old_path, &st)) != 0) { | 343 struct stat64 st; |
323 return false; | 344 if (TEMP_FAILURE_RETRY(stat64(old_path, &st)) != 0) { |
324 } | 345 return false; |
325 int old_fd = TEMP_FAILURE_RETRY(open64(old_path, O_RDONLY | O_CLOEXEC)); | 346 } |
326 if (old_fd < 0) { | 347 int old_fd = TEMP_FAILURE_RETRY(open64(old_path, O_RDONLY | O_CLOEXEC)); |
327 return false; | 348 if (old_fd < 0) { |
328 } | 349 return false; |
329 int new_fd = TEMP_FAILURE_RETRY( | 350 } |
330 open64(new_path, O_WRONLY | O_TRUNC | O_CREAT | O_CLOEXEC, st.st_mode)); | 351 int new_fd = TEMP_FAILURE_RETRY( |
331 if (new_fd < 0) { | 352 open64(new_path, O_WRONLY | O_TRUNC | O_CREAT | O_CLOEXEC, st.st_mode)); |
332 VOID_TEMP_FAILURE_RETRY(close(old_fd)); | 353 if (new_fd < 0) { |
333 return false; | 354 VOID_TEMP_FAILURE_RETRY(close(old_fd)); |
334 } | 355 return false; |
335 int64_t offset = 0; | 356 } |
336 intptr_t result = 1; | 357 int64_t offset = 0; |
337 while (result > 0) { | 358 intptr_t result = 1; |
338 // Loop to ensure we copy everything, and not only up to 2GB. | 359 while (result > 0) { |
339 result = | 360 // Loop to ensure we copy everything, and not only up to 2GB. |
340 NO_RETRY_EXPECTED(sendfile64(new_fd, old_fd, &offset, kMaxUint32)); | 361 result = NO_RETRY_EXPECTED(sendfile64(new_fd, old_fd, &offset, kMaxUint32)); |
341 } | 362 } |
342 // From sendfile man pages: | 363 // From sendfile man pages: |
343 // Applications may wish to fall back to read(2)/write(2) in the case | 364 // Applications may wish to fall back to read(2)/write(2) in the case |
344 // where sendfile() fails with EINVAL or ENOSYS. | 365 // where sendfile() fails with EINVAL or ENOSYS. |
345 if ((result < 0) && ((errno == EINVAL) || (errno == ENOSYS))) { | 366 if ((result < 0) && ((errno == EINVAL) || (errno == ENOSYS))) { |
346 const intptr_t kBufferSize = 8 * KB; | 367 const intptr_t kBufferSize = 8 * KB; |
347 uint8_t buffer[kBufferSize]; | 368 uint8_t buffer[kBufferSize]; |
348 while ((result = TEMP_FAILURE_RETRY(read(old_fd, buffer, kBufferSize))) > | 369 while ((result = TEMP_FAILURE_RETRY(read(old_fd, buffer, kBufferSize))) > |
349 0) { | 370 0) { |
350 int wrote = TEMP_FAILURE_RETRY(write(new_fd, buffer, result)); | 371 int wrote = TEMP_FAILURE_RETRY(write(new_fd, buffer, result)); |
351 if (wrote != result) { | 372 if (wrote != result) { |
352 result = -1; | 373 result = -1; |
353 break; | 374 break; |
354 } | |
355 } | 375 } |
356 } | 376 } |
357 int e = errno; | |
358 VOID_TEMP_FAILURE_RETRY(close(old_fd)); | |
359 VOID_TEMP_FAILURE_RETRY(close(new_fd)); | |
360 if (result < 0) { | |
361 VOID_NO_RETRY_EXPECTED(unlink(new_path)); | |
362 errno = e; | |
363 return false; | |
364 } | |
365 return true; | |
366 } else if (type == kIsDirectory) { | |
367 errno = EISDIR; | |
368 } else { | |
369 errno = ENOENT; | |
370 } | 377 } |
371 return false; | 378 int e = errno; |
| 379 VOID_TEMP_FAILURE_RETRY(close(old_fd)); |
| 380 VOID_TEMP_FAILURE_RETRY(close(new_fd)); |
| 381 if (result < 0) { |
| 382 VOID_NO_RETRY_EXPECTED(unlink(new_path)); |
| 383 errno = e; |
| 384 return false; |
| 385 } |
| 386 return true; |
372 } | 387 } |
373 | 388 |
374 | 389 |
375 int64_t File::LengthFromPath(const char* name) { | 390 int64_t File::LengthFromPath(const char* name) { |
376 struct stat64 st; | 391 struct stat64 st; |
377 if (TEMP_FAILURE_RETRY(stat64(name, &st)) == 0) { | 392 if (TEMP_FAILURE_RETRY(stat64(name, &st)) == 0) { |
378 // Signal an error if it's a directory. | 393 // Signal an error if it's a directory. |
379 if (S_ISDIR(st.st_mode)) { | 394 if (S_ISDIR(st.st_mode)) { |
380 errno = EISDIR; | 395 errno = EISDIR; |
381 return -1; | 396 return -1; |
(...skipping 125 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
507 if (S_ISSOCK(buf.st_mode)) { | 522 if (S_ISSOCK(buf.st_mode)) { |
508 return kSocket; | 523 return kSocket; |
509 } | 524 } |
510 if (S_ISREG(buf.st_mode)) { | 525 if (S_ISREG(buf.st_mode)) { |
511 return kFile; | 526 return kFile; |
512 } | 527 } |
513 return kOther; | 528 return kOther; |
514 } | 529 } |
515 | 530 |
516 | 531 |
517 File::Type File::GetType(const char* pathname, bool follow_links) { | |
518 struct stat64 entry_info; | |
519 int stat_success; | |
520 if (follow_links) { | |
521 stat_success = TEMP_FAILURE_RETRY(stat64(pathname, &entry_info)); | |
522 } else { | |
523 stat_success = TEMP_FAILURE_RETRY(lstat64(pathname, &entry_info)); | |
524 } | |
525 if (stat_success == -1) { | |
526 return File::kDoesNotExist; | |
527 } | |
528 if (S_ISDIR(entry_info.st_mode)) { | |
529 return File::kIsDirectory; | |
530 } | |
531 if (S_ISREG(entry_info.st_mode)) { | |
532 return File::kIsFile; | |
533 } | |
534 if (S_ISLNK(entry_info.st_mode)) { | |
535 return File::kIsLink; | |
536 } | |
537 return File::kDoesNotExist; | |
538 } | |
539 | |
540 | |
541 File::Identical File::AreIdentical(const char* file_1, const char* file_2) { | 532 File::Identical File::AreIdentical(const char* file_1, const char* file_2) { |
542 struct stat64 file_1_info; | 533 struct stat64 file_1_info; |
543 struct stat64 file_2_info; | 534 struct stat64 file_2_info; |
544 if ((TEMP_FAILURE_RETRY(lstat64(file_1, &file_1_info)) == -1) || | 535 if ((TEMP_FAILURE_RETRY(lstat64(file_1, &file_1_info)) == -1) || |
545 (TEMP_FAILURE_RETRY(lstat64(file_2, &file_2_info)) == -1)) { | 536 (TEMP_FAILURE_RETRY(lstat64(file_2, &file_2_info)) == -1)) { |
546 return File::kError; | 537 return File::kError; |
547 } | 538 } |
548 return ((file_1_info.st_ino == file_2_info.st_ino) && | 539 return ((file_1_info.st_ino == file_2_info.st_ino) && |
549 (file_1_info.st_dev == file_2_info.st_dev)) | 540 (file_1_info.st_dev == file_2_info.st_dev)) |
550 ? File::kIdentical | 541 ? File::kIdentical |
551 : File::kDifferent; | 542 : File::kDifferent; |
552 } | 543 } |
553 | 544 |
554 } // namespace bin | 545 } // namespace bin |
555 } // namespace dart | 546 } // namespace dart |
556 | 547 |
557 #endif // defined(TARGET_OS_LINUX) | 548 #endif // defined(TARGET_OS_LINUX) |
OLD | NEW |