if (ip->i_df.if_format == XFS_DINODE_FMT_LOCAL) { /* * The VFS crashes on a NULL pointer, so return -EFSCORRUPTED * if if_data is junk.
*/ if (XFS_IS_CORRUPT(ip->i_mount, !ip->i_df.if_data)) goto out_corrupt;
/* * Check component lengths of the target path name.
*/
pathlen = strlen(target_path); if (pathlen >= XFS_SYMLINK_MAXLEN) /* total string too long */ return -ENAMETOOLONG;
ASSERT(pathlen > 0);
/* Make sure that we have allocated dquot(s) on disk. */
error = xfs_icreate_dqalloc(&args, &udqp, &gdqp, &pdqp); if (error) return error;
/* * The symlink will fit into the inode data fork? * If there are no parent pointers, then there wont't be any attributes. * So we get the whole variable part, and do not need to reserve extra * blocks. Otherwise, we need to reserve the blocks.
*/ if (pathlen <= XFS_LITINO(mp) && !xfs_has_parent(mp))
fs_blocks = 0; else
fs_blocks = xfs_symlink_blocks(mp, pathlen);
resblks = xfs_symlink_space_res(mp, link_name->len, fs_blocks);
error = xfs_parent_start(mp, &du.ppargs); if (error) goto out_release_dquots;
/* * Check whether the directory allows new symlinks or not.
*/ if (dp->i_diflags & XFS_DIFLAG_NOSYMLINKS) {
error = -EPERM; goto out_trans_cancel;
}
/* * Allocate an inode for the symlink.
*/
error = xfs_dialloc(&tp, &args, &ino); if (!error)
error = xfs_icreate(tp, ino, &args, &du.ip); if (error) goto out_trans_cancel;
/* * Now we join the directory inode to the transaction. We do not do it * earlier because xfs_dir_ialloc might commit the previous transaction * (and release all the locks). An error from here on will result in * the transaction cancel unlocking dp so don't do it explicitly in the * error path.
*/
xfs_trans_ijoin(tp, dp, 0);
/* * Also attach the dquot(s) to it, if applicable.
*/
xfs_qm_vop_create_dqattach(tp, du.ip, udqp, gdqp, pdqp);
/* * Create the directory entry for the symlink.
*/
error = xfs_dir_create_child(tp, resblks, &du); if (error) goto out_trans_cancel;
/* * If this is a synchronous mount, make sure that the * symlink transaction goes to disk before returning to * the user.
*/ if (xfs_has_wsync(mp) || xfs_has_dirsync(mp))
xfs_trans_set_sync(tp);
error = xfs_trans_commit(tp); if (error) goto out_release_inode;
out_trans_cancel:
xfs_trans_cancel(tp);
out_release_inode: /* * Wait until after the current transaction is aborted to finish the * setup of the inode and release the inode. This prevents recursive * transactions and deadlocks from xfs_inactive.
*/ if (du.ip) {
xfs_iunlock(du.ip, XFS_ILOCK_EXCL);
xfs_finish_inode_setup(du.ip);
xfs_irele(du.ip);
}
out_parent:
xfs_parent_finish(mp, du.ppargs);
out_release_dquots:
xfs_qm_dqrele(udqp);
xfs_qm_dqrele(gdqp);
xfs_qm_dqrele(pdqp);
if (unlock_dp_on_error)
xfs_iunlock(dp, XFS_ILOCK_EXCL); return error;
}
/* * Free a symlink that has blocks associated with it. * * Note: zero length symlinks are not allowed to exist. When we set the size to * zero, also change it to a regular file so that it does not get written to * disk as a zero length symlink. The inode is on the unlinked list already, so * userspace cannot find this inode anymore, so this change is not user visible * but allows us to catch corrupt zero-length symlinks in the verifiers.
*/ STATICint
xfs_inactive_symlink_rmt( struct xfs_inode *ip)
{ struct xfs_mount *mp = ip->i_mount; struct xfs_trans *tp; int error;
ASSERT(!xfs_need_iread_extents(&ip->i_df)); /* * We're freeing a symlink that has some * blocks allocated to it. Free the * blocks here. We know that we've got * either 1 or 2 extents and that we can * free them all in one bunmapi call.
*/
ASSERT(ip->i_df.if_nextents > 0 && ip->i_df.if_nextents <= 2);
/* * Lock the inode, fix the size, turn it into a regular file and join it * to the transaction. Hold it so in the normal path, we still have it * locked for the second transaction. In the error paths we need it * held so the cancel won't rele it, see below.
*/
ip->i_disk_size = 0;
VFS_I(ip)->i_mode = (VFS_I(ip)->i_mode & ~S_IFMT) | S_IFREG;
xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
error = xfs_symlink_remote_truncate(tp, ip); if (error) goto error_trans_cancel;
error = xfs_trans_commit(tp); if (error) {
ASSERT(xfs_is_shutdown(mp)); goto error_unlock;
}
/* * Remove the memory for extent descriptions (just bookkeeping).
*/ if (ip->i_df.if_bytes)
xfs_idata_realloc(ip, -ip->i_df.if_bytes, XFS_DATA_FORK);
ASSERT(ip->i_df.if_bytes == 0);
/* * Inline fork state gets removed by xfs_difree() so we have nothing to * do here in that case.
*/ if (ip->i_df.if_format == XFS_DINODE_FMT_LOCAL) {
xfs_iunlock(ip, XFS_ILOCK_EXCL); return 0;
}
xfs_iunlock(ip, XFS_ILOCK_EXCL);
/* remove the remote symlink */ return xfs_inactive_symlink_rmt(ip);
}
Messung V0.5
¤ Dauer der Verarbeitung: 0.24 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.