diff options
Diffstat (limited to 'fs/xfs/xfs_symlink.c')
-rw-r--r-- | fs/xfs/xfs_symlink.c | 102 |
1 files changed, 48 insertions, 54 deletions
diff --git a/fs/xfs/xfs_symlink.c b/fs/xfs/xfs_symlink.c index f622a97a7e33..14e58f2c96bd 100644 --- a/fs/xfs/xfs_symlink.c +++ b/fs/xfs/xfs_symlink.c | |||
@@ -17,31 +17,31 @@ | |||
17 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA | 17 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
18 | */ | 18 | */ |
19 | #include "xfs.h" | 19 | #include "xfs.h" |
20 | #include "xfs_shared.h" | ||
20 | #include "xfs_fs.h" | 21 | #include "xfs_fs.h" |
21 | #include "xfs_format.h" | 22 | #include "xfs_format.h" |
23 | #include "xfs_log_format.h" | ||
24 | #include "xfs_trans_resv.h" | ||
22 | #include "xfs_bit.h" | 25 | #include "xfs_bit.h" |
23 | #include "xfs_log.h" | ||
24 | #include "xfs_trans.h" | ||
25 | #include "xfs_sb.h" | 26 | #include "xfs_sb.h" |
26 | #include "xfs_ag.h" | 27 | #include "xfs_ag.h" |
27 | #include "xfs_mount.h" | 28 | #include "xfs_mount.h" |
28 | #include "xfs_da_btree.h" | 29 | #include "xfs_da_format.h" |
29 | #include "xfs_dir2_format.h" | ||
30 | #include "xfs_dir2.h" | 30 | #include "xfs_dir2.h" |
31 | #include "xfs_bmap_btree.h" | ||
32 | #include "xfs_ialloc_btree.h" | ||
33 | #include "xfs_dinode.h" | ||
34 | #include "xfs_inode.h" | 31 | #include "xfs_inode.h" |
35 | #include "xfs_ialloc.h" | 32 | #include "xfs_ialloc.h" |
36 | #include "xfs_alloc.h" | 33 | #include "xfs_alloc.h" |
37 | #include "xfs_bmap.h" | 34 | #include "xfs_bmap.h" |
35 | #include "xfs_bmap_btree.h" | ||
38 | #include "xfs_bmap_util.h" | 36 | #include "xfs_bmap_util.h" |
39 | #include "xfs_error.h" | 37 | #include "xfs_error.h" |
40 | #include "xfs_quota.h" | 38 | #include "xfs_quota.h" |
41 | #include "xfs_trans_space.h" | 39 | #include "xfs_trans_space.h" |
42 | #include "xfs_trace.h" | 40 | #include "xfs_trace.h" |
43 | #include "xfs_symlink.h" | 41 | #include "xfs_symlink.h" |
44 | #include "xfs_buf_item.h" | 42 | #include "xfs_trans.h" |
43 | #include "xfs_log.h" | ||
44 | #include "xfs_dinode.h" | ||
45 | 45 | ||
46 | /* ----- Kernel only functions below ----- */ | 46 | /* ----- Kernel only functions below ----- */ |
47 | STATIC int | 47 | STATIC int |
@@ -424,8 +424,7 @@ xfs_symlink( | |||
424 | */ | 424 | */ |
425 | STATIC int | 425 | STATIC int |
426 | xfs_inactive_symlink_rmt( | 426 | xfs_inactive_symlink_rmt( |
427 | xfs_inode_t *ip, | 427 | struct xfs_inode *ip) |
428 | xfs_trans_t **tpp) | ||
429 | { | 428 | { |
430 | xfs_buf_t *bp; | 429 | xfs_buf_t *bp; |
431 | int committed; | 430 | int committed; |
@@ -437,11 +436,9 @@ xfs_inactive_symlink_rmt( | |||
437 | xfs_mount_t *mp; | 436 | xfs_mount_t *mp; |
438 | xfs_bmbt_irec_t mval[XFS_SYMLINK_MAPS]; | 437 | xfs_bmbt_irec_t mval[XFS_SYMLINK_MAPS]; |
439 | int nmaps; | 438 | int nmaps; |
440 | xfs_trans_t *ntp; | ||
441 | int size; | 439 | int size; |
442 | xfs_trans_t *tp; | 440 | xfs_trans_t *tp; |
443 | 441 | ||
444 | tp = *tpp; | ||
445 | mp = ip->i_mount; | 442 | mp = ip->i_mount; |
446 | ASSERT(ip->i_df.if_flags & XFS_IFEXTENTS); | 443 | ASSERT(ip->i_df.if_flags & XFS_IFEXTENTS); |
447 | /* | 444 | /* |
@@ -453,6 +450,16 @@ xfs_inactive_symlink_rmt( | |||
453 | */ | 450 | */ |
454 | ASSERT(ip->i_d.di_nextents > 0 && ip->i_d.di_nextents <= 2); | 451 | ASSERT(ip->i_d.di_nextents > 0 && ip->i_d.di_nextents <= 2); |
455 | 452 | ||
453 | tp = xfs_trans_alloc(mp, XFS_TRANS_INACTIVE); | ||
454 | error = xfs_trans_reserve(tp, &M_RES(mp)->tr_itruncate, 0, 0); | ||
455 | if (error) { | ||
456 | xfs_trans_cancel(tp, 0); | ||
457 | return error; | ||
458 | } | ||
459 | |||
460 | xfs_ilock(ip, XFS_ILOCK_EXCL); | ||
461 | xfs_trans_ijoin(tp, ip, 0); | ||
462 | |||
456 | /* | 463 | /* |
457 | * Lock the inode, fix the size, and join it to the transaction. | 464 | * Lock the inode, fix the size, and join it to the transaction. |
458 | * Hold it so in the normal path, we still have it locked for | 465 | * Hold it so in the normal path, we still have it locked for |
@@ -471,7 +478,7 @@ xfs_inactive_symlink_rmt( | |||
471 | error = xfs_bmapi_read(ip, 0, xfs_symlink_blocks(mp, size), | 478 | error = xfs_bmapi_read(ip, 0, xfs_symlink_blocks(mp, size), |
472 | mval, &nmaps, 0); | 479 | mval, &nmaps, 0); |
473 | if (error) | 480 | if (error) |
474 | goto error0; | 481 | goto error_trans_cancel; |
475 | /* | 482 | /* |
476 | * Invalidate the block(s). No validation is done. | 483 | * Invalidate the block(s). No validation is done. |
477 | */ | 484 | */ |
@@ -481,22 +488,24 @@ xfs_inactive_symlink_rmt( | |||
481 | XFS_FSB_TO_BB(mp, mval[i].br_blockcount), 0); | 488 | XFS_FSB_TO_BB(mp, mval[i].br_blockcount), 0); |
482 | if (!bp) { | 489 | if (!bp) { |
483 | error = ENOMEM; | 490 | error = ENOMEM; |
484 | goto error1; | 491 | goto error_bmap_cancel; |
485 | } | 492 | } |
486 | xfs_trans_binval(tp, bp); | 493 | xfs_trans_binval(tp, bp); |
487 | } | 494 | } |
488 | /* | 495 | /* |
489 | * Unmap the dead block(s) to the free_list. | 496 | * Unmap the dead block(s) to the free_list. |
490 | */ | 497 | */ |
491 | if ((error = xfs_bunmapi(tp, ip, 0, size, XFS_BMAPI_METADATA, nmaps, | 498 | error = xfs_bunmapi(tp, ip, 0, size, XFS_BMAPI_METADATA, nmaps, |
492 | &first_block, &free_list, &done))) | 499 | &first_block, &free_list, &done); |
493 | goto error1; | 500 | if (error) |
501 | goto error_bmap_cancel; | ||
494 | ASSERT(done); | 502 | ASSERT(done); |
495 | /* | 503 | /* |
496 | * Commit the first transaction. This logs the EFI and the inode. | 504 | * Commit the first transaction. This logs the EFI and the inode. |
497 | */ | 505 | */ |
498 | if ((error = xfs_bmap_finish(&tp, &free_list, &committed))) | 506 | error = xfs_bmap_finish(&tp, &free_list, &committed); |
499 | goto error1; | 507 | if (error) |
508 | goto error_bmap_cancel; | ||
500 | /* | 509 | /* |
501 | * The transaction must have been committed, since there were | 510 | * The transaction must have been committed, since there were |
502 | * actually extents freed by xfs_bunmapi. See xfs_bmap_finish. | 511 | * actually extents freed by xfs_bunmapi. See xfs_bmap_finish. |
@@ -511,26 +520,13 @@ xfs_inactive_symlink_rmt( | |||
511 | xfs_trans_ijoin(tp, ip, 0); | 520 | xfs_trans_ijoin(tp, ip, 0); |
512 | xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); | 521 | xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); |
513 | /* | 522 | /* |
514 | * Get a new, empty transaction to return to our caller. | ||
515 | */ | ||
516 | ntp = xfs_trans_dup(tp); | ||
517 | /* | ||
518 | * Commit the transaction containing extent freeing and EFDs. | 523 | * Commit the transaction containing extent freeing and EFDs. |
519 | * If we get an error on the commit here or on the reserve below, | ||
520 | * we need to unlock the inode since the new transaction doesn't | ||
521 | * have the inode attached. | ||
522 | */ | 524 | */ |
523 | error = xfs_trans_commit(tp, 0); | 525 | error = xfs_trans_commit(tp, XFS_TRANS_RELEASE_LOG_RES); |
524 | tp = ntp; | ||
525 | if (error) { | 526 | if (error) { |
526 | ASSERT(XFS_FORCED_SHUTDOWN(mp)); | 527 | ASSERT(XFS_FORCED_SHUTDOWN(mp)); |
527 | goto error0; | 528 | goto error_unlock; |
528 | } | 529 | } |
529 | /* | ||
530 | * transaction commit worked ok so we can drop the extra ticket | ||
531 | * reference that we gained in xfs_trans_dup() | ||
532 | */ | ||
533 | xfs_log_ticket_put(tp->t_ticket); | ||
534 | 530 | ||
535 | /* | 531 | /* |
536 | * Remove the memory for extent descriptions (just bookkeeping). | 532 | * Remove the memory for extent descriptions (just bookkeeping). |
@@ -538,23 +534,16 @@ xfs_inactive_symlink_rmt( | |||
538 | if (ip->i_df.if_bytes) | 534 | if (ip->i_df.if_bytes) |
539 | xfs_idata_realloc(ip, -ip->i_df.if_bytes, XFS_DATA_FORK); | 535 | xfs_idata_realloc(ip, -ip->i_df.if_bytes, XFS_DATA_FORK); |
540 | ASSERT(ip->i_df.if_bytes == 0); | 536 | ASSERT(ip->i_df.if_bytes == 0); |
541 | /* | ||
542 | * Put an itruncate log reservation in the new transaction | ||
543 | * for our caller. | ||
544 | */ | ||
545 | error = xfs_trans_reserve(tp, &M_RES(mp)->tr_itruncate, 0, 0); | ||
546 | if (error) { | ||
547 | ASSERT(XFS_FORCED_SHUTDOWN(mp)); | ||
548 | goto error0; | ||
549 | } | ||
550 | 537 | ||
551 | xfs_trans_ijoin(tp, ip, 0); | 538 | xfs_iunlock(ip, XFS_ILOCK_EXCL); |
552 | *tpp = tp; | ||
553 | return 0; | 539 | return 0; |
554 | 540 | ||
555 | error1: | 541 | error_bmap_cancel: |
556 | xfs_bmap_cancel(&free_list); | 542 | xfs_bmap_cancel(&free_list); |
557 | error0: | 543 | error_trans_cancel: |
544 | xfs_trans_cancel(tp, XFS_TRANS_RELEASE_LOG_RES | XFS_TRANS_ABORT); | ||
545 | error_unlock: | ||
546 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | ||
558 | return error; | 547 | return error; |
559 | } | 548 | } |
560 | 549 | ||
@@ -563,41 +552,46 @@ xfs_inactive_symlink_rmt( | |||
563 | */ | 552 | */ |
564 | int | 553 | int |
565 | xfs_inactive_symlink( | 554 | xfs_inactive_symlink( |
566 | struct xfs_inode *ip, | 555 | struct xfs_inode *ip) |
567 | struct xfs_trans **tp) | ||
568 | { | 556 | { |
569 | struct xfs_mount *mp = ip->i_mount; | 557 | struct xfs_mount *mp = ip->i_mount; |
570 | int pathlen; | 558 | int pathlen; |
571 | 559 | ||
572 | trace_xfs_inactive_symlink(ip); | 560 | trace_xfs_inactive_symlink(ip); |
573 | 561 | ||
574 | ASSERT(xfs_isilocked(ip, XFS_ILOCK_EXCL)); | ||
575 | |||
576 | if (XFS_FORCED_SHUTDOWN(mp)) | 562 | if (XFS_FORCED_SHUTDOWN(mp)) |
577 | return XFS_ERROR(EIO); | 563 | return XFS_ERROR(EIO); |
578 | 564 | ||
565 | xfs_ilock(ip, XFS_ILOCK_EXCL); | ||
566 | |||
579 | /* | 567 | /* |
580 | * Zero length symlinks _can_ exist. | 568 | * Zero length symlinks _can_ exist. |
581 | */ | 569 | */ |
582 | pathlen = (int)ip->i_d.di_size; | 570 | pathlen = (int)ip->i_d.di_size; |
583 | if (!pathlen) | 571 | if (!pathlen) { |
572 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | ||
584 | return 0; | 573 | return 0; |
574 | } | ||
585 | 575 | ||
586 | if (pathlen < 0 || pathlen > MAXPATHLEN) { | 576 | if (pathlen < 0 || pathlen > MAXPATHLEN) { |
587 | xfs_alert(mp, "%s: inode (0x%llx) bad symlink length (%d)", | 577 | xfs_alert(mp, "%s: inode (0x%llx) bad symlink length (%d)", |
588 | __func__, (unsigned long long)ip->i_ino, pathlen); | 578 | __func__, (unsigned long long)ip->i_ino, pathlen); |
579 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | ||
589 | ASSERT(0); | 580 | ASSERT(0); |
590 | return XFS_ERROR(EFSCORRUPTED); | 581 | return XFS_ERROR(EFSCORRUPTED); |
591 | } | 582 | } |
592 | 583 | ||
593 | if (ip->i_df.if_flags & XFS_IFINLINE) { | 584 | if (ip->i_df.if_flags & XFS_IFINLINE) { |
594 | if (ip->i_df.if_bytes > 0) | 585 | if (ip->i_df.if_bytes > 0) |
595 | xfs_idata_realloc(ip, -(ip->i_df.if_bytes), | 586 | xfs_idata_realloc(ip, -(ip->i_df.if_bytes), |
596 | XFS_DATA_FORK); | 587 | XFS_DATA_FORK); |
588 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | ||
597 | ASSERT(ip->i_df.if_bytes == 0); | 589 | ASSERT(ip->i_df.if_bytes == 0); |
598 | return 0; | 590 | return 0; |
599 | } | 591 | } |
600 | 592 | ||
593 | xfs_iunlock(ip, XFS_ILOCK_EXCL); | ||
594 | |||
601 | /* remove the remote symlink */ | 595 | /* remove the remote symlink */ |
602 | return xfs_inactive_symlink_rmt(ip, tp); | 596 | return xfs_inactive_symlink_rmt(ip); |
603 | } | 597 | } |