Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 | // SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2022-2024 Oracle. * All rights reserved. */ #include "xfs.h" #include "xfs_fs.h" #include "xfs_format.h" #include "xfs_da_format.h" #include "xfs_log_format.h" #include "xfs_shared.h" #include "xfs_trans_resv.h" #include "xfs_mount.h" #include "xfs_bmap_btree.h" #include "xfs_inode.h" #include "xfs_error.h" #include "xfs_trace.h" #include "xfs_trans.h" #include "xfs_da_btree.h" #include "xfs_attr.h" #include "xfs_dir2.h" #include "xfs_dir2_priv.h" #include "xfs_attr_sf.h" #include "xfs_bmap.h" #include "xfs_defer.h" #include "xfs_log.h" #include "xfs_xattr.h" #include "xfs_parent.h" #include "xfs_trans_space.h" #include "xfs_attr_item.h" #include "xfs_health.h" struct kmem_cache *xfs_parent_args_cache; /* * Parent pointer attribute handling. * * Because the attribute name is a filename component, it will never be longer * than 255 bytes and must not contain nulls or slashes. These are roughly the * same constraints that apply to attribute names. * * The attribute value must always be a struct xfs_parent_rec. This means the * attribute will never be in remote format because 12 bytes is nowhere near * xfs_attr_leaf_entsize_local_max() (~75% of block size). * * Creating a new parent attribute will always create a new attribute - there * should never, ever be an existing attribute in the tree for a new inode. * ENOSPC behavior is problematic - creating the inode without the parent * pointer is effectively a corruption, so we allow parent attribute creation * to dip into the reserve block pool to avoid unexpected ENOSPC errors from * occurring. */ /* Return true if parent pointer attr name is valid. */ bool xfs_parent_namecheck( unsigned int attr_flags, const void *name, size_t length) { /* * Parent pointers always use logged operations, so there should never * be incomplete xattrs. */ if (attr_flags & XFS_ATTR_INCOMPLETE) return false; return xfs_dir2_namecheck(name, length); } /* Return true if parent pointer attr value is valid. */ bool xfs_parent_valuecheck( struct xfs_mount *mp, const void *value, size_t valuelen) { const struct xfs_parent_rec *rec = value; if (!xfs_has_parent(mp)) return false; /* The xattr value must be a parent record. */ if (valuelen != sizeof(struct xfs_parent_rec)) return false; /* The parent record must be local. */ if (value == NULL) return false; /* The parent inumber must be valid. */ if (!xfs_verify_dir_ino(mp, be64_to_cpu(rec->p_ino))) return false; return true; } /* Compute the attribute name hash for a parent pointer. */ xfs_dahash_t xfs_parent_hashval( struct xfs_mount *mp, const uint8_t *name, int namelen, xfs_ino_t parent_ino) { struct xfs_name xname = { .name = name, .len = namelen, }; /* * Use the same dirent name hash as would be used on the directory, but * mix in the parent inode number to avoid collisions on hardlinked * files with identical names but different parents. */ return xfs_dir2_hashname(mp, &xname) ^ upper_32_bits(parent_ino) ^ lower_32_bits(parent_ino); } /* Compute the attribute name hash from the xattr components. */ xfs_dahash_t xfs_parent_hashattr( struct xfs_mount *mp, const uint8_t *name, int namelen, const void *value, int valuelen) { const struct xfs_parent_rec *rec = value; /* Requires a local attr value in xfs_parent_rec format */ if (valuelen != sizeof(struct xfs_parent_rec)) { ASSERT(valuelen == sizeof(struct xfs_parent_rec)); return 0; } if (!value) { ASSERT(value != NULL); return 0; } return xfs_parent_hashval(mp, name, namelen, be64_to_cpu(rec->p_ino)); } /* * Initialize the parent pointer arguments structure. Caller must have zeroed * the contents of @args. @tp is only required for updates. */ static void xfs_parent_da_args_init( struct xfs_da_args *args, struct xfs_trans *tp, struct xfs_parent_rec *rec, struct xfs_inode *child, xfs_ino_t owner, const struct xfs_name *parent_name) { args->geo = child->i_mount->m_attr_geo; args->whichfork = XFS_ATTR_FORK; args->attr_filter = XFS_ATTR_PARENT; args->op_flags = XFS_DA_OP_LOGGED | XFS_DA_OP_OKNOENT; args->trans = tp; args->dp = child; args->owner = owner; args->name = parent_name->name; args->namelen = parent_name->len; args->value = rec; args->valuelen = sizeof(struct xfs_parent_rec); xfs_attr_sethash(args); } /* Make sure the incore state is ready for a parent pointer query/update. */ static inline int xfs_parent_iread_extents( struct xfs_trans *tp, struct xfs_inode *child) { /* Parent pointers require that the attr fork must exist. */ if (XFS_IS_CORRUPT(child->i_mount, !xfs_inode_has_attr_fork(child))) { xfs_inode_mark_sick(child, XFS_SICK_INO_PARENT); return -EFSCORRUPTED; } return xfs_iread_extents(tp, child, XFS_ATTR_FORK); } /* Add a parent pointer to reflect a dirent addition. */ int xfs_parent_addname( struct xfs_trans *tp, struct xfs_parent_args *ppargs, struct xfs_inode *dp, const struct xfs_name *parent_name, struct xfs_inode *child) { int error; error = xfs_parent_iread_extents(tp, child); if (error) return error; xfs_inode_to_parent_rec(&ppargs->rec, dp); xfs_parent_da_args_init(&ppargs->args, tp, &ppargs->rec, child, child->i_ino, parent_name); xfs_attr_defer_add(&ppargs->args, XFS_ATTR_DEFER_SET); return 0; } /* Remove a parent pointer to reflect a dirent removal. */ int xfs_parent_removename( struct xfs_trans *tp, struct xfs_parent_args *ppargs, struct xfs_inode *dp, const struct xfs_name *parent_name, struct xfs_inode *child) { int error; error = xfs_parent_iread_extents(tp, child); if (error) return error; xfs_inode_to_parent_rec(&ppargs->rec, dp); xfs_parent_da_args_init(&ppargs->args, tp, &ppargs->rec, child, child->i_ino, parent_name); xfs_attr_defer_add(&ppargs->args, XFS_ATTR_DEFER_REMOVE); return 0; } /* Replace one parent pointer with another to reflect a rename. */ int xfs_parent_replacename( struct xfs_trans *tp, struct xfs_parent_args *ppargs, struct xfs_inode *old_dp, const struct xfs_name *old_name, struct xfs_inode *new_dp, const struct xfs_name *new_name, struct xfs_inode *child) { int error; error = xfs_parent_iread_extents(tp, child); if (error) return error; xfs_inode_to_parent_rec(&ppargs->rec, old_dp); xfs_parent_da_args_init(&ppargs->args, tp, &ppargs->rec, child, child->i_ino, old_name); xfs_inode_to_parent_rec(&ppargs->new_rec, new_dp); ppargs->args.new_name = new_name->name; ppargs->args.new_namelen = new_name->len; ppargs->args.new_value = &ppargs->new_rec; ppargs->args.new_valuelen = sizeof(struct xfs_parent_rec); xfs_attr_defer_add(&ppargs->args, XFS_ATTR_DEFER_REPLACE); return 0; } /* * Extract parent pointer information from any parent pointer xattr into * @parent_ino/gen. The last two parameters can be NULL pointers. * * Returns 0 if this is not a parent pointer xattr at all; or -EFSCORRUPTED for * garbage. */ int xfs_parent_from_attr( struct xfs_mount *mp, unsigned int attr_flags, const unsigned char *name, unsigned int namelen, const void *value, unsigned int valuelen, xfs_ino_t *parent_ino, uint32_t *parent_gen) { const struct xfs_parent_rec *rec = value; ASSERT(attr_flags & XFS_ATTR_PARENT); if (!xfs_parent_namecheck(attr_flags, name, namelen)) return -EFSCORRUPTED; if (!xfs_parent_valuecheck(mp, value, valuelen)) return -EFSCORRUPTED; if (parent_ino) *parent_ino = be64_to_cpu(rec->p_ino); if (parent_gen) *parent_gen = be32_to_cpu(rec->p_gen); return 0; } /* * Look up a parent pointer record (@parent_name -> @pptr) of @ip. * * Caller must hold at least ILOCK_SHARED. The scratchpad need not be * initialized. * * Returns 0 if the pointer is found, -ENOATTR if there is no match, or a * negative errno. */ int xfs_parent_lookup( struct xfs_trans *tp, struct xfs_inode *ip, const struct xfs_name *parent_name, struct xfs_parent_rec *pptr, struct xfs_da_args *scratch) { memset(scratch, 0, sizeof(struct xfs_da_args)); xfs_parent_da_args_init(scratch, tp, pptr, ip, ip->i_ino, parent_name); return xfs_attr_get_ilocked(scratch); } /* Sanity-check a parent pointer before we try to perform repairs. */ static inline bool xfs_parent_sanity_check( struct xfs_mount *mp, const struct xfs_name *parent_name, const struct xfs_parent_rec *pptr) { if (!xfs_parent_namecheck(XFS_ATTR_PARENT, parent_name->name, parent_name->len)) return false; if (!xfs_parent_valuecheck(mp, pptr, sizeof(*pptr))) return false; return true; } /* * Attach the parent pointer (@parent_name -> @pptr) to @ip immediately. * Caller must not have a transaction or hold the ILOCK. This is for * specialized repair functions only. The scratchpad need not be initialized. */ int xfs_parent_set( struct xfs_inode *ip, xfs_ino_t owner, const struct xfs_name *parent_name, struct xfs_parent_rec *pptr, struct xfs_da_args *scratch) { if (!xfs_parent_sanity_check(ip->i_mount, parent_name, pptr)) { ASSERT(0); return -EFSCORRUPTED; } memset(scratch, 0, sizeof(struct xfs_da_args)); xfs_parent_da_args_init(scratch, NULL, pptr, ip, owner, parent_name); return xfs_attr_set(scratch, XFS_ATTRUPDATE_CREATE, false); } /* * Remove the parent pointer (@parent_name -> @pptr) from @ip immediately. * Caller must not have a transaction or hold the ILOCK. This is for * specialized repair functions only. The scratchpad need not be initialized. */ int xfs_parent_unset( struct xfs_inode *ip, xfs_ino_t owner, const struct xfs_name *parent_name, struct xfs_parent_rec *pptr, struct xfs_da_args *scratch) { if (!xfs_parent_sanity_check(ip->i_mount, parent_name, pptr)) { ASSERT(0); return -EFSCORRUPTED; } memset(scratch, 0, sizeof(struct xfs_da_args)); xfs_parent_da_args_init(scratch, NULL, pptr, ip, owner, parent_name); return xfs_attr_set(scratch, XFS_ATTRUPDATE_REMOVE, false); } |