1*2a82b8beSDavid Chinner /* 2*2a82b8beSDavid Chinner * Copyright (c) 2006-2007 Silicon Graphics, Inc. 3*2a82b8beSDavid Chinner * All Rights Reserved. 4*2a82b8beSDavid Chinner * 5*2a82b8beSDavid Chinner * This program is free software; you can redistribute it and/or 6*2a82b8beSDavid Chinner * modify it under the terms of the GNU General Public License as 7*2a82b8beSDavid Chinner * published by the Free Software Foundation. 8*2a82b8beSDavid Chinner * 9*2a82b8beSDavid Chinner * This program is distributed in the hope that it would be useful, 10*2a82b8beSDavid Chinner * but WITHOUT ANY WARRANTY; without even the implied warranty of 11*2a82b8beSDavid Chinner * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12*2a82b8beSDavid Chinner * GNU General Public License for more details. 13*2a82b8beSDavid Chinner * 14*2a82b8beSDavid Chinner * You should have received a copy of the GNU General Public License 15*2a82b8beSDavid Chinner * along with this program; if not, write the Free Software Foundation, 16*2a82b8beSDavid Chinner * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 17*2a82b8beSDavid Chinner */ 18*2a82b8beSDavid Chinner #include "xfs.h" 19*2a82b8beSDavid Chinner #include "xfs_bmap_btree.h" 20*2a82b8beSDavid Chinner #include "xfs_inum.h" 21*2a82b8beSDavid Chinner #include "xfs_dir2.h" 22*2a82b8beSDavid Chinner #include "xfs_dir2_sf.h" 23*2a82b8beSDavid Chinner #include "xfs_attr_sf.h" 24*2a82b8beSDavid Chinner #include "xfs_dinode.h" 25*2a82b8beSDavid Chinner #include "xfs_inode.h" 26*2a82b8beSDavid Chinner #include "xfs_ag.h" 27*2a82b8beSDavid Chinner #include "xfs_dmapi.h" 28*2a82b8beSDavid Chinner #include "xfs_log.h" 29*2a82b8beSDavid Chinner #include "xfs_trans.h" 30*2a82b8beSDavid Chinner #include "xfs_sb.h" 31*2a82b8beSDavid Chinner #include "xfs_mount.h" 32*2a82b8beSDavid Chinner #include "xfs_bmap.h" 33*2a82b8beSDavid Chinner #include "xfs_alloc.h" 34*2a82b8beSDavid Chinner #include "xfs_utils.h" 35*2a82b8beSDavid Chinner #include "xfs_mru_cache.h" 36*2a82b8beSDavid Chinner #include "xfs_filestream.h" 37*2a82b8beSDavid Chinner 38*2a82b8beSDavid Chinner #ifdef XFS_FILESTREAMS_TRACE 39*2a82b8beSDavid Chinner 40*2a82b8beSDavid Chinner ktrace_t *xfs_filestreams_trace_buf; 41*2a82b8beSDavid Chinner 42*2a82b8beSDavid Chinner STATIC void 43*2a82b8beSDavid Chinner xfs_filestreams_trace( 44*2a82b8beSDavid Chinner xfs_mount_t *mp, /* mount point */ 45*2a82b8beSDavid Chinner int type, /* type of trace */ 46*2a82b8beSDavid Chinner const char *func, /* source function */ 47*2a82b8beSDavid Chinner int line, /* source line number */ 48*2a82b8beSDavid Chinner __psunsigned_t arg0, 49*2a82b8beSDavid Chinner __psunsigned_t arg1, 50*2a82b8beSDavid Chinner __psunsigned_t arg2, 51*2a82b8beSDavid Chinner __psunsigned_t arg3, 52*2a82b8beSDavid Chinner __psunsigned_t arg4, 53*2a82b8beSDavid Chinner __psunsigned_t arg5) 54*2a82b8beSDavid Chinner { 55*2a82b8beSDavid Chinner ktrace_enter(xfs_filestreams_trace_buf, 56*2a82b8beSDavid Chinner (void *)(__psint_t)(type | (line << 16)), 57*2a82b8beSDavid Chinner (void *)func, 58*2a82b8beSDavid Chinner (void *)(__psunsigned_t)current_pid(), 59*2a82b8beSDavid Chinner (void *)mp, 60*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg0, 61*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg1, 62*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg2, 63*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg3, 64*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg4, 65*2a82b8beSDavid Chinner (void *)(__psunsigned_t)arg5, 66*2a82b8beSDavid Chinner NULL, NULL, NULL, NULL, NULL, NULL); 67*2a82b8beSDavid Chinner } 68*2a82b8beSDavid Chinner 69*2a82b8beSDavid Chinner #define TRACE0(mp,t) TRACE6(mp,t,0,0,0,0,0,0) 70*2a82b8beSDavid Chinner #define TRACE1(mp,t,a0) TRACE6(mp,t,a0,0,0,0,0,0) 71*2a82b8beSDavid Chinner #define TRACE2(mp,t,a0,a1) TRACE6(mp,t,a0,a1,0,0,0,0) 72*2a82b8beSDavid Chinner #define TRACE3(mp,t,a0,a1,a2) TRACE6(mp,t,a0,a1,a2,0,0,0) 73*2a82b8beSDavid Chinner #define TRACE4(mp,t,a0,a1,a2,a3) TRACE6(mp,t,a0,a1,a2,a3,0,0) 74*2a82b8beSDavid Chinner #define TRACE5(mp,t,a0,a1,a2,a3,a4) TRACE6(mp,t,a0,a1,a2,a3,a4,0) 75*2a82b8beSDavid Chinner #define TRACE6(mp,t,a0,a1,a2,a3,a4,a5) \ 76*2a82b8beSDavid Chinner xfs_filestreams_trace(mp, t, __FUNCTION__, __LINE__, \ 77*2a82b8beSDavid Chinner (__psunsigned_t)a0, (__psunsigned_t)a1, \ 78*2a82b8beSDavid Chinner (__psunsigned_t)a2, (__psunsigned_t)a3, \ 79*2a82b8beSDavid Chinner (__psunsigned_t)a4, (__psunsigned_t)a5) 80*2a82b8beSDavid Chinner 81*2a82b8beSDavid Chinner #define TRACE_AG_SCAN(mp, ag, ag2) \ 82*2a82b8beSDavid Chinner TRACE2(mp, XFS_FSTRM_KTRACE_AGSCAN, ag, ag2); 83*2a82b8beSDavid Chinner #define TRACE_AG_PICK1(mp, max_ag, maxfree) \ 84*2a82b8beSDavid Chinner TRACE2(mp, XFS_FSTRM_KTRACE_AGPICK1, max_ag, maxfree); 85*2a82b8beSDavid Chinner #define TRACE_AG_PICK2(mp, ag, ag2, cnt, free, scan, flag) \ 86*2a82b8beSDavid Chinner TRACE6(mp, XFS_FSTRM_KTRACE_AGPICK2, ag, ag2, \ 87*2a82b8beSDavid Chinner cnt, free, scan, flag) 88*2a82b8beSDavid Chinner #define TRACE_UPDATE(mp, ip, ag, cnt, ag2, cnt2) \ 89*2a82b8beSDavid Chinner TRACE5(mp, XFS_FSTRM_KTRACE_UPDATE, ip, ag, cnt, ag2, cnt2) 90*2a82b8beSDavid Chinner #define TRACE_FREE(mp, ip, pip, ag, cnt) \ 91*2a82b8beSDavid Chinner TRACE4(mp, XFS_FSTRM_KTRACE_FREE, ip, pip, ag, cnt) 92*2a82b8beSDavid Chinner #define TRACE_LOOKUP(mp, ip, pip, ag, cnt) \ 93*2a82b8beSDavid Chinner TRACE4(mp, XFS_FSTRM_KTRACE_ITEM_LOOKUP, ip, pip, ag, cnt) 94*2a82b8beSDavid Chinner #define TRACE_ASSOCIATE(mp, ip, pip, ag, cnt) \ 95*2a82b8beSDavid Chinner TRACE4(mp, XFS_FSTRM_KTRACE_ASSOCIATE, ip, pip, ag, cnt) 96*2a82b8beSDavid Chinner #define TRACE_MOVEAG(mp, ip, pip, oag, ocnt, nag, ncnt) \ 97*2a82b8beSDavid Chinner TRACE6(mp, XFS_FSTRM_KTRACE_MOVEAG, ip, pip, oag, ocnt, nag, ncnt) 98*2a82b8beSDavid Chinner #define TRACE_ORPHAN(mp, ip, ag) \ 99*2a82b8beSDavid Chinner TRACE2(mp, XFS_FSTRM_KTRACE_ORPHAN, ip, ag); 100*2a82b8beSDavid Chinner 101*2a82b8beSDavid Chinner 102*2a82b8beSDavid Chinner #else 103*2a82b8beSDavid Chinner #define TRACE_AG_SCAN(mp, ag, ag2) 104*2a82b8beSDavid Chinner #define TRACE_AG_PICK1(mp, max_ag, maxfree) 105*2a82b8beSDavid Chinner #define TRACE_AG_PICK2(mp, ag, ag2, cnt, free, scan, flag) 106*2a82b8beSDavid Chinner #define TRACE_UPDATE(mp, ip, ag, cnt, ag2, cnt2) 107*2a82b8beSDavid Chinner #define TRACE_FREE(mp, ip, pip, ag, cnt) 108*2a82b8beSDavid Chinner #define TRACE_LOOKUP(mp, ip, pip, ag, cnt) 109*2a82b8beSDavid Chinner #define TRACE_ASSOCIATE(mp, ip, pip, ag, cnt) 110*2a82b8beSDavid Chinner #define TRACE_MOVEAG(mp, ip, pip, oag, ocnt, nag, ncnt) 111*2a82b8beSDavid Chinner #define TRACE_ORPHAN(mp, ip, ag) 112*2a82b8beSDavid Chinner #endif 113*2a82b8beSDavid Chinner 114*2a82b8beSDavid Chinner static kmem_zone_t *item_zone; 115*2a82b8beSDavid Chinner 116*2a82b8beSDavid Chinner /* 117*2a82b8beSDavid Chinner * Structure for associating a file or a directory with an allocation group. 118*2a82b8beSDavid Chinner * The parent directory pointer is only needed for files, but since there will 119*2a82b8beSDavid Chinner * generally be vastly more files than directories in the cache, using the same 120*2a82b8beSDavid Chinner * data structure simplifies the code with very little memory overhead. 121*2a82b8beSDavid Chinner */ 122*2a82b8beSDavid Chinner typedef struct fstrm_item 123*2a82b8beSDavid Chinner { 124*2a82b8beSDavid Chinner xfs_agnumber_t ag; /* AG currently in use for the file/directory. */ 125*2a82b8beSDavid Chinner xfs_inode_t *ip; /* inode self-pointer. */ 126*2a82b8beSDavid Chinner xfs_inode_t *pip; /* Parent directory inode pointer. */ 127*2a82b8beSDavid Chinner } fstrm_item_t; 128*2a82b8beSDavid Chinner 129*2a82b8beSDavid Chinner 130*2a82b8beSDavid Chinner /* 131*2a82b8beSDavid Chinner * Scan the AGs starting at startag looking for an AG that isn't in use and has 132*2a82b8beSDavid Chinner * at least minlen blocks free. 133*2a82b8beSDavid Chinner */ 134*2a82b8beSDavid Chinner static int 135*2a82b8beSDavid Chinner _xfs_filestream_pick_ag( 136*2a82b8beSDavid Chinner xfs_mount_t *mp, 137*2a82b8beSDavid Chinner xfs_agnumber_t startag, 138*2a82b8beSDavid Chinner xfs_agnumber_t *agp, 139*2a82b8beSDavid Chinner int flags, 140*2a82b8beSDavid Chinner xfs_extlen_t minlen) 141*2a82b8beSDavid Chinner { 142*2a82b8beSDavid Chinner int err, trylock, nscan; 143*2a82b8beSDavid Chinner xfs_extlen_t delta, longest, need, free, minfree, maxfree = 0; 144*2a82b8beSDavid Chinner xfs_agnumber_t ag, max_ag = NULLAGNUMBER; 145*2a82b8beSDavid Chinner struct xfs_perag *pag; 146*2a82b8beSDavid Chinner 147*2a82b8beSDavid Chinner /* 2% of an AG's blocks must be free for it to be chosen. */ 148*2a82b8beSDavid Chinner minfree = mp->m_sb.sb_agblocks / 50; 149*2a82b8beSDavid Chinner 150*2a82b8beSDavid Chinner ag = startag; 151*2a82b8beSDavid Chinner *agp = NULLAGNUMBER; 152*2a82b8beSDavid Chinner 153*2a82b8beSDavid Chinner /* For the first pass, don't sleep trying to init the per-AG. */ 154*2a82b8beSDavid Chinner trylock = XFS_ALLOC_FLAG_TRYLOCK; 155*2a82b8beSDavid Chinner 156*2a82b8beSDavid Chinner for (nscan = 0; 1; nscan++) { 157*2a82b8beSDavid Chinner 158*2a82b8beSDavid Chinner TRACE_AG_SCAN(mp, ag, xfs_filestream_peek_ag(mp, ag)); 159*2a82b8beSDavid Chinner 160*2a82b8beSDavid Chinner pag = mp->m_perag + ag; 161*2a82b8beSDavid Chinner 162*2a82b8beSDavid Chinner if (!pag->pagf_init) { 163*2a82b8beSDavid Chinner err = xfs_alloc_pagf_init(mp, NULL, ag, trylock); 164*2a82b8beSDavid Chinner if (err && !trylock) 165*2a82b8beSDavid Chinner return err; 166*2a82b8beSDavid Chinner } 167*2a82b8beSDavid Chinner 168*2a82b8beSDavid Chinner /* Might fail sometimes during the 1st pass with trylock set. */ 169*2a82b8beSDavid Chinner if (!pag->pagf_init) 170*2a82b8beSDavid Chinner goto next_ag; 171*2a82b8beSDavid Chinner 172*2a82b8beSDavid Chinner /* Keep track of the AG with the most free blocks. */ 173*2a82b8beSDavid Chinner if (pag->pagf_freeblks > maxfree) { 174*2a82b8beSDavid Chinner maxfree = pag->pagf_freeblks; 175*2a82b8beSDavid Chinner max_ag = ag; 176*2a82b8beSDavid Chinner } 177*2a82b8beSDavid Chinner 178*2a82b8beSDavid Chinner /* 179*2a82b8beSDavid Chinner * The AG reference count does two things: it enforces mutual 180*2a82b8beSDavid Chinner * exclusion when examining the suitability of an AG in this 181*2a82b8beSDavid Chinner * loop, and it guards against two filestreams being established 182*2a82b8beSDavid Chinner * in the same AG as each other. 183*2a82b8beSDavid Chinner */ 184*2a82b8beSDavid Chinner if (xfs_filestream_get_ag(mp, ag) > 1) { 185*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, ag); 186*2a82b8beSDavid Chinner goto next_ag; 187*2a82b8beSDavid Chinner } 188*2a82b8beSDavid Chinner 189*2a82b8beSDavid Chinner need = XFS_MIN_FREELIST_PAG(pag, mp); 190*2a82b8beSDavid Chinner delta = need > pag->pagf_flcount ? need - pag->pagf_flcount : 0; 191*2a82b8beSDavid Chinner longest = (pag->pagf_longest > delta) ? 192*2a82b8beSDavid Chinner (pag->pagf_longest - delta) : 193*2a82b8beSDavid Chinner (pag->pagf_flcount > 0 || pag->pagf_longest > 0); 194*2a82b8beSDavid Chinner 195*2a82b8beSDavid Chinner if (((minlen && longest >= minlen) || 196*2a82b8beSDavid Chinner (!minlen && pag->pagf_freeblks >= minfree)) && 197*2a82b8beSDavid Chinner (!pag->pagf_metadata || !(flags & XFS_PICK_USERDATA) || 198*2a82b8beSDavid Chinner (flags & XFS_PICK_LOWSPACE))) { 199*2a82b8beSDavid Chinner 200*2a82b8beSDavid Chinner /* Break out, retaining the reference on the AG. */ 201*2a82b8beSDavid Chinner free = pag->pagf_freeblks; 202*2a82b8beSDavid Chinner *agp = ag; 203*2a82b8beSDavid Chinner break; 204*2a82b8beSDavid Chinner } 205*2a82b8beSDavid Chinner 206*2a82b8beSDavid Chinner /* Drop the reference on this AG, it's not usable. */ 207*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, ag); 208*2a82b8beSDavid Chinner next_ag: 209*2a82b8beSDavid Chinner /* Move to the next AG, wrapping to AG 0 if necessary. */ 210*2a82b8beSDavid Chinner if (++ag >= mp->m_sb.sb_agcount) 211*2a82b8beSDavid Chinner ag = 0; 212*2a82b8beSDavid Chinner 213*2a82b8beSDavid Chinner /* If a full pass of the AGs hasn't been done yet, continue. */ 214*2a82b8beSDavid Chinner if (ag != startag) 215*2a82b8beSDavid Chinner continue; 216*2a82b8beSDavid Chinner 217*2a82b8beSDavid Chinner /* Allow sleeping in xfs_alloc_pagf_init() on the 2nd pass. */ 218*2a82b8beSDavid Chinner if (trylock != 0) { 219*2a82b8beSDavid Chinner trylock = 0; 220*2a82b8beSDavid Chinner continue; 221*2a82b8beSDavid Chinner } 222*2a82b8beSDavid Chinner 223*2a82b8beSDavid Chinner /* Finally, if lowspace wasn't set, set it for the 3rd pass. */ 224*2a82b8beSDavid Chinner if (!(flags & XFS_PICK_LOWSPACE)) { 225*2a82b8beSDavid Chinner flags |= XFS_PICK_LOWSPACE; 226*2a82b8beSDavid Chinner continue; 227*2a82b8beSDavid Chinner } 228*2a82b8beSDavid Chinner 229*2a82b8beSDavid Chinner /* 230*2a82b8beSDavid Chinner * Take the AG with the most free space, regardless of whether 231*2a82b8beSDavid Chinner * it's already in use by another filestream. 232*2a82b8beSDavid Chinner */ 233*2a82b8beSDavid Chinner if (max_ag != NULLAGNUMBER) { 234*2a82b8beSDavid Chinner xfs_filestream_get_ag(mp, max_ag); 235*2a82b8beSDavid Chinner TRACE_AG_PICK1(mp, max_ag, maxfree); 236*2a82b8beSDavid Chinner free = maxfree; 237*2a82b8beSDavid Chinner *agp = max_ag; 238*2a82b8beSDavid Chinner break; 239*2a82b8beSDavid Chinner } 240*2a82b8beSDavid Chinner 241*2a82b8beSDavid Chinner /* take AG 0 if none matched */ 242*2a82b8beSDavid Chinner TRACE_AG_PICK1(mp, max_ag, maxfree); 243*2a82b8beSDavid Chinner *agp = 0; 244*2a82b8beSDavid Chinner return 0; 245*2a82b8beSDavid Chinner } 246*2a82b8beSDavid Chinner 247*2a82b8beSDavid Chinner TRACE_AG_PICK2(mp, startag, *agp, xfs_filestream_peek_ag(mp, *agp), 248*2a82b8beSDavid Chinner free, nscan, flags); 249*2a82b8beSDavid Chinner 250*2a82b8beSDavid Chinner return 0; 251*2a82b8beSDavid Chinner } 252*2a82b8beSDavid Chinner 253*2a82b8beSDavid Chinner /* 254*2a82b8beSDavid Chinner * Set the allocation group number for a file or a directory, updating inode 255*2a82b8beSDavid Chinner * references and per-AG references as appropriate. Must be called with the 256*2a82b8beSDavid Chinner * m_peraglock held in read mode. 257*2a82b8beSDavid Chinner */ 258*2a82b8beSDavid Chinner static int 259*2a82b8beSDavid Chinner _xfs_filestream_update_ag( 260*2a82b8beSDavid Chinner xfs_inode_t *ip, 261*2a82b8beSDavid Chinner xfs_inode_t *pip, 262*2a82b8beSDavid Chinner xfs_agnumber_t ag) 263*2a82b8beSDavid Chinner { 264*2a82b8beSDavid Chinner int err = 0; 265*2a82b8beSDavid Chinner xfs_mount_t *mp; 266*2a82b8beSDavid Chinner xfs_mru_cache_t *cache; 267*2a82b8beSDavid Chinner fstrm_item_t *item; 268*2a82b8beSDavid Chinner xfs_agnumber_t old_ag; 269*2a82b8beSDavid Chinner xfs_inode_t *old_pip; 270*2a82b8beSDavid Chinner 271*2a82b8beSDavid Chinner /* 272*2a82b8beSDavid Chinner * Either ip is a regular file and pip is a directory, or ip is a 273*2a82b8beSDavid Chinner * directory and pip is NULL. 274*2a82b8beSDavid Chinner */ 275*2a82b8beSDavid Chinner ASSERT(ip && (((ip->i_d.di_mode & S_IFREG) && pip && 276*2a82b8beSDavid Chinner (pip->i_d.di_mode & S_IFDIR)) || 277*2a82b8beSDavid Chinner ((ip->i_d.di_mode & S_IFDIR) && !pip))); 278*2a82b8beSDavid Chinner 279*2a82b8beSDavid Chinner mp = ip->i_mount; 280*2a82b8beSDavid Chinner cache = mp->m_filestream; 281*2a82b8beSDavid Chinner 282*2a82b8beSDavid Chinner item = xfs_mru_cache_lookup(cache, ip->i_ino); 283*2a82b8beSDavid Chinner if (item) { 284*2a82b8beSDavid Chinner ASSERT(item->ip == ip); 285*2a82b8beSDavid Chinner old_ag = item->ag; 286*2a82b8beSDavid Chinner item->ag = ag; 287*2a82b8beSDavid Chinner old_pip = item->pip; 288*2a82b8beSDavid Chinner item->pip = pip; 289*2a82b8beSDavid Chinner xfs_mru_cache_done(cache); 290*2a82b8beSDavid Chinner 291*2a82b8beSDavid Chinner /* 292*2a82b8beSDavid Chinner * If the AG has changed, drop the old ref and take a new one, 293*2a82b8beSDavid Chinner * effectively transferring the reference from old to new AG. 294*2a82b8beSDavid Chinner */ 295*2a82b8beSDavid Chinner if (ag != old_ag) { 296*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, old_ag); 297*2a82b8beSDavid Chinner xfs_filestream_get_ag(mp, ag); 298*2a82b8beSDavid Chinner } 299*2a82b8beSDavid Chinner 300*2a82b8beSDavid Chinner /* 301*2a82b8beSDavid Chinner * If ip is a file and its pip has changed, drop the old ref and 302*2a82b8beSDavid Chinner * take a new one. 303*2a82b8beSDavid Chinner */ 304*2a82b8beSDavid Chinner if (pip && pip != old_pip) { 305*2a82b8beSDavid Chinner IRELE(old_pip); 306*2a82b8beSDavid Chinner IHOLD(pip); 307*2a82b8beSDavid Chinner } 308*2a82b8beSDavid Chinner 309*2a82b8beSDavid Chinner TRACE_UPDATE(mp, ip, old_ag, xfs_filestream_peek_ag(mp, old_ag), 310*2a82b8beSDavid Chinner ag, xfs_filestream_peek_ag(mp, ag)); 311*2a82b8beSDavid Chinner return 0; 312*2a82b8beSDavid Chinner } 313*2a82b8beSDavid Chinner 314*2a82b8beSDavid Chinner item = kmem_zone_zalloc(item_zone, KM_MAYFAIL); 315*2a82b8beSDavid Chinner if (!item) 316*2a82b8beSDavid Chinner return ENOMEM; 317*2a82b8beSDavid Chinner 318*2a82b8beSDavid Chinner item->ag = ag; 319*2a82b8beSDavid Chinner item->ip = ip; 320*2a82b8beSDavid Chinner item->pip = pip; 321*2a82b8beSDavid Chinner 322*2a82b8beSDavid Chinner err = xfs_mru_cache_insert(cache, ip->i_ino, item); 323*2a82b8beSDavid Chinner if (err) { 324*2a82b8beSDavid Chinner kmem_zone_free(item_zone, item); 325*2a82b8beSDavid Chinner return err; 326*2a82b8beSDavid Chinner } 327*2a82b8beSDavid Chinner 328*2a82b8beSDavid Chinner /* Take a reference on the AG. */ 329*2a82b8beSDavid Chinner xfs_filestream_get_ag(mp, ag); 330*2a82b8beSDavid Chinner 331*2a82b8beSDavid Chinner /* 332*2a82b8beSDavid Chinner * Take a reference on the inode itself regardless of whether it's a 333*2a82b8beSDavid Chinner * regular file or a directory. 334*2a82b8beSDavid Chinner */ 335*2a82b8beSDavid Chinner IHOLD(ip); 336*2a82b8beSDavid Chinner 337*2a82b8beSDavid Chinner /* 338*2a82b8beSDavid Chinner * In the case of a regular file, take a reference on the parent inode 339*2a82b8beSDavid Chinner * as well to ensure it remains in-core. 340*2a82b8beSDavid Chinner */ 341*2a82b8beSDavid Chinner if (pip) 342*2a82b8beSDavid Chinner IHOLD(pip); 343*2a82b8beSDavid Chinner 344*2a82b8beSDavid Chinner TRACE_UPDATE(mp, ip, ag, xfs_filestream_peek_ag(mp, ag), 345*2a82b8beSDavid Chinner ag, xfs_filestream_peek_ag(mp, ag)); 346*2a82b8beSDavid Chinner 347*2a82b8beSDavid Chinner return 0; 348*2a82b8beSDavid Chinner } 349*2a82b8beSDavid Chinner 350*2a82b8beSDavid Chinner /* xfs_fstrm_free_func(): callback for freeing cached stream items. */ 351*2a82b8beSDavid Chinner void 352*2a82b8beSDavid Chinner xfs_fstrm_free_func( 353*2a82b8beSDavid Chinner xfs_ino_t ino, 354*2a82b8beSDavid Chinner fstrm_item_t *item) 355*2a82b8beSDavid Chinner { 356*2a82b8beSDavid Chinner xfs_inode_t *ip = item->ip; 357*2a82b8beSDavid Chinner int ref; 358*2a82b8beSDavid Chinner 359*2a82b8beSDavid Chinner ASSERT(ip->i_ino == ino); 360*2a82b8beSDavid Chinner 361*2a82b8beSDavid Chinner xfs_iflags_clear(ip, XFS_IFILESTREAM); 362*2a82b8beSDavid Chinner 363*2a82b8beSDavid Chinner /* Drop the reference taken on the AG when the item was added. */ 364*2a82b8beSDavid Chinner ref = xfs_filestream_put_ag(ip->i_mount, item->ag); 365*2a82b8beSDavid Chinner 366*2a82b8beSDavid Chinner ASSERT(ref >= 0); 367*2a82b8beSDavid Chinner TRACE_FREE(ip->i_mount, ip, item->pip, item->ag, 368*2a82b8beSDavid Chinner xfs_filestream_peek_ag(ip->i_mount, item->ag)); 369*2a82b8beSDavid Chinner 370*2a82b8beSDavid Chinner /* 371*2a82b8beSDavid Chinner * _xfs_filestream_update_ag() always takes a reference on the inode 372*2a82b8beSDavid Chinner * itself, whether it's a file or a directory. Release it here. 373*2a82b8beSDavid Chinner * This can result in the inode being freed and so we must 374*2a82b8beSDavid Chinner * not hold any inode locks when freeing filesstreams objects 375*2a82b8beSDavid Chinner * otherwise we can deadlock here. 376*2a82b8beSDavid Chinner */ 377*2a82b8beSDavid Chinner IRELE(ip); 378*2a82b8beSDavid Chinner 379*2a82b8beSDavid Chinner /* 380*2a82b8beSDavid Chinner * In the case of a regular file, _xfs_filestream_update_ag() also 381*2a82b8beSDavid Chinner * takes a ref on the parent inode to keep it in-core. Release that 382*2a82b8beSDavid Chinner * too. 383*2a82b8beSDavid Chinner */ 384*2a82b8beSDavid Chinner if (item->pip) 385*2a82b8beSDavid Chinner IRELE(item->pip); 386*2a82b8beSDavid Chinner 387*2a82b8beSDavid Chinner /* Finally, free the memory allocated for the item. */ 388*2a82b8beSDavid Chinner kmem_zone_free(item_zone, item); 389*2a82b8beSDavid Chinner } 390*2a82b8beSDavid Chinner 391*2a82b8beSDavid Chinner /* 392*2a82b8beSDavid Chinner * xfs_filestream_init() is called at xfs initialisation time to set up the 393*2a82b8beSDavid Chinner * memory zone that will be used for filestream data structure allocation. 394*2a82b8beSDavid Chinner */ 395*2a82b8beSDavid Chinner int 396*2a82b8beSDavid Chinner xfs_filestream_init(void) 397*2a82b8beSDavid Chinner { 398*2a82b8beSDavid Chinner item_zone = kmem_zone_init(sizeof(fstrm_item_t), "fstrm_item"); 399*2a82b8beSDavid Chinner #ifdef XFS_FILESTREAMS_TRACE 400*2a82b8beSDavid Chinner xfs_filestreams_trace_buf = ktrace_alloc(XFS_FSTRM_KTRACE_SIZE, KM_SLEEP); 401*2a82b8beSDavid Chinner #endif 402*2a82b8beSDavid Chinner return item_zone ? 0 : -ENOMEM; 403*2a82b8beSDavid Chinner } 404*2a82b8beSDavid Chinner 405*2a82b8beSDavid Chinner /* 406*2a82b8beSDavid Chinner * xfs_filestream_uninit() is called at xfs termination time to destroy the 407*2a82b8beSDavid Chinner * memory zone that was used for filestream data structure allocation. 408*2a82b8beSDavid Chinner */ 409*2a82b8beSDavid Chinner void 410*2a82b8beSDavid Chinner xfs_filestream_uninit(void) 411*2a82b8beSDavid Chinner { 412*2a82b8beSDavid Chinner #ifdef XFS_FILESTREAMS_TRACE 413*2a82b8beSDavid Chinner ktrace_free(xfs_filestreams_trace_buf); 414*2a82b8beSDavid Chinner #endif 415*2a82b8beSDavid Chinner kmem_zone_destroy(item_zone); 416*2a82b8beSDavid Chinner } 417*2a82b8beSDavid Chinner 418*2a82b8beSDavid Chinner /* 419*2a82b8beSDavid Chinner * xfs_filestream_mount() is called when a file system is mounted with the 420*2a82b8beSDavid Chinner * filestream option. It is responsible for allocating the data structures 421*2a82b8beSDavid Chinner * needed to track the new file system's file streams. 422*2a82b8beSDavid Chinner */ 423*2a82b8beSDavid Chinner int 424*2a82b8beSDavid Chinner xfs_filestream_mount( 425*2a82b8beSDavid Chinner xfs_mount_t *mp) 426*2a82b8beSDavid Chinner { 427*2a82b8beSDavid Chinner int err; 428*2a82b8beSDavid Chinner unsigned int lifetime, grp_count; 429*2a82b8beSDavid Chinner 430*2a82b8beSDavid Chinner /* 431*2a82b8beSDavid Chinner * The filestream timer tunable is currently fixed within the range of 432*2a82b8beSDavid Chinner * one second to four minutes, with five seconds being the default. The 433*2a82b8beSDavid Chinner * group count is somewhat arbitrary, but it'd be nice to adhere to the 434*2a82b8beSDavid Chinner * timer tunable to within about 10 percent. This requires at least 10 435*2a82b8beSDavid Chinner * groups. 436*2a82b8beSDavid Chinner */ 437*2a82b8beSDavid Chinner lifetime = xfs_fstrm_centisecs * 10; 438*2a82b8beSDavid Chinner grp_count = 10; 439*2a82b8beSDavid Chinner 440*2a82b8beSDavid Chinner err = xfs_mru_cache_create(&mp->m_filestream, lifetime, grp_count, 441*2a82b8beSDavid Chinner (xfs_mru_cache_free_func_t)xfs_fstrm_free_func); 442*2a82b8beSDavid Chinner 443*2a82b8beSDavid Chinner return err; 444*2a82b8beSDavid Chinner } 445*2a82b8beSDavid Chinner 446*2a82b8beSDavid Chinner /* 447*2a82b8beSDavid Chinner * xfs_filestream_unmount() is called when a file system that was mounted with 448*2a82b8beSDavid Chinner * the filestream option is unmounted. It drains the data structures created 449*2a82b8beSDavid Chinner * to track the file system's file streams and frees all the memory that was 450*2a82b8beSDavid Chinner * allocated. 451*2a82b8beSDavid Chinner */ 452*2a82b8beSDavid Chinner void 453*2a82b8beSDavid Chinner xfs_filestream_unmount( 454*2a82b8beSDavid Chinner xfs_mount_t *mp) 455*2a82b8beSDavid Chinner { 456*2a82b8beSDavid Chinner xfs_mru_cache_destroy(mp->m_filestream); 457*2a82b8beSDavid Chinner } 458*2a82b8beSDavid Chinner 459*2a82b8beSDavid Chinner /* 460*2a82b8beSDavid Chinner * If the mount point's m_perag array is going to be reallocated, all 461*2a82b8beSDavid Chinner * outstanding cache entries must be flushed to avoid accessing reference count 462*2a82b8beSDavid Chinner * addresses that have been freed. The call to xfs_filestream_flush() must be 463*2a82b8beSDavid Chinner * made inside the block that holds the m_peraglock in write mode to do the 464*2a82b8beSDavid Chinner * reallocation. 465*2a82b8beSDavid Chinner */ 466*2a82b8beSDavid Chinner void 467*2a82b8beSDavid Chinner xfs_filestream_flush( 468*2a82b8beSDavid Chinner xfs_mount_t *mp) 469*2a82b8beSDavid Chinner { 470*2a82b8beSDavid Chinner /* point in time flush, so keep the reaper running */ 471*2a82b8beSDavid Chinner xfs_mru_cache_flush(mp->m_filestream, 1); 472*2a82b8beSDavid Chinner } 473*2a82b8beSDavid Chinner 474*2a82b8beSDavid Chinner /* 475*2a82b8beSDavid Chinner * Return the AG of the filestream the file or directory belongs to, or 476*2a82b8beSDavid Chinner * NULLAGNUMBER otherwise. 477*2a82b8beSDavid Chinner */ 478*2a82b8beSDavid Chinner xfs_agnumber_t 479*2a82b8beSDavid Chinner xfs_filestream_lookup_ag( 480*2a82b8beSDavid Chinner xfs_inode_t *ip) 481*2a82b8beSDavid Chinner { 482*2a82b8beSDavid Chinner xfs_mru_cache_t *cache; 483*2a82b8beSDavid Chinner fstrm_item_t *item; 484*2a82b8beSDavid Chinner xfs_agnumber_t ag; 485*2a82b8beSDavid Chinner int ref; 486*2a82b8beSDavid Chinner 487*2a82b8beSDavid Chinner if (!(ip->i_d.di_mode & (S_IFREG | S_IFDIR))) { 488*2a82b8beSDavid Chinner ASSERT(0); 489*2a82b8beSDavid Chinner return NULLAGNUMBER; 490*2a82b8beSDavid Chinner } 491*2a82b8beSDavid Chinner 492*2a82b8beSDavid Chinner cache = ip->i_mount->m_filestream; 493*2a82b8beSDavid Chinner item = xfs_mru_cache_lookup(cache, ip->i_ino); 494*2a82b8beSDavid Chinner if (!item) { 495*2a82b8beSDavid Chinner TRACE_LOOKUP(ip->i_mount, ip, NULL, NULLAGNUMBER, 0); 496*2a82b8beSDavid Chinner return NULLAGNUMBER; 497*2a82b8beSDavid Chinner } 498*2a82b8beSDavid Chinner 499*2a82b8beSDavid Chinner ASSERT(ip == item->ip); 500*2a82b8beSDavid Chinner ag = item->ag; 501*2a82b8beSDavid Chinner ref = xfs_filestream_peek_ag(ip->i_mount, ag); 502*2a82b8beSDavid Chinner xfs_mru_cache_done(cache); 503*2a82b8beSDavid Chinner 504*2a82b8beSDavid Chinner TRACE_LOOKUP(ip->i_mount, ip, item->pip, ag, ref); 505*2a82b8beSDavid Chinner return ag; 506*2a82b8beSDavid Chinner } 507*2a82b8beSDavid Chinner 508*2a82b8beSDavid Chinner /* 509*2a82b8beSDavid Chinner * xfs_filestream_associate() should only be called to associate a regular file 510*2a82b8beSDavid Chinner * with its parent directory. Calling it with a child directory isn't 511*2a82b8beSDavid Chinner * appropriate because filestreams don't apply to entire directory hierarchies. 512*2a82b8beSDavid Chinner * Creating a file in a child directory of an existing filestream directory 513*2a82b8beSDavid Chinner * starts a new filestream with its own allocation group association. 514*2a82b8beSDavid Chinner * 515*2a82b8beSDavid Chinner * Returns < 0 on error, 0 if successful association occurred, > 0 if 516*2a82b8beSDavid Chinner * we failed to get an association because of locking issues. 517*2a82b8beSDavid Chinner */ 518*2a82b8beSDavid Chinner int 519*2a82b8beSDavid Chinner xfs_filestream_associate( 520*2a82b8beSDavid Chinner xfs_inode_t *pip, 521*2a82b8beSDavid Chinner xfs_inode_t *ip) 522*2a82b8beSDavid Chinner { 523*2a82b8beSDavid Chinner xfs_mount_t *mp; 524*2a82b8beSDavid Chinner xfs_mru_cache_t *cache; 525*2a82b8beSDavid Chinner fstrm_item_t *item; 526*2a82b8beSDavid Chinner xfs_agnumber_t ag, rotorstep, startag; 527*2a82b8beSDavid Chinner int err = 0; 528*2a82b8beSDavid Chinner 529*2a82b8beSDavid Chinner ASSERT(pip->i_d.di_mode & S_IFDIR); 530*2a82b8beSDavid Chinner ASSERT(ip->i_d.di_mode & S_IFREG); 531*2a82b8beSDavid Chinner if (!(pip->i_d.di_mode & S_IFDIR) || !(ip->i_d.di_mode & S_IFREG)) 532*2a82b8beSDavid Chinner return -EINVAL; 533*2a82b8beSDavid Chinner 534*2a82b8beSDavid Chinner mp = pip->i_mount; 535*2a82b8beSDavid Chinner cache = mp->m_filestream; 536*2a82b8beSDavid Chinner down_read(&mp->m_peraglock); 537*2a82b8beSDavid Chinner 538*2a82b8beSDavid Chinner /* 539*2a82b8beSDavid Chinner * We have a problem, Houston. 540*2a82b8beSDavid Chinner * 541*2a82b8beSDavid Chinner * Taking the iolock here violates inode locking order - we already 542*2a82b8beSDavid Chinner * hold the ilock. Hence if we block getting this lock we may never 543*2a82b8beSDavid Chinner * wake. Unfortunately, that means if we can't get the lock, we're 544*2a82b8beSDavid Chinner * screwed in terms of getting a stream association - we can't spin 545*2a82b8beSDavid Chinner * waiting for the lock because someone else is waiting on the lock we 546*2a82b8beSDavid Chinner * hold and we cannot drop that as we are in a transaction here. 547*2a82b8beSDavid Chinner * 548*2a82b8beSDavid Chinner * Lucky for us, this inversion is rarely a problem because it's a 549*2a82b8beSDavid Chinner * directory inode that we are trying to lock here and that means the 550*2a82b8beSDavid Chinner * only place that matters is xfs_sync_inodes() and SYNC_DELWRI is 551*2a82b8beSDavid Chinner * used. i.e. freeze, remount-ro, quotasync or unmount. 552*2a82b8beSDavid Chinner * 553*2a82b8beSDavid Chinner * So, if we can't get the iolock without sleeping then just give up 554*2a82b8beSDavid Chinner */ 555*2a82b8beSDavid Chinner if (!xfs_ilock_nowait(pip, XFS_IOLOCK_EXCL)) { 556*2a82b8beSDavid Chinner up_read(&mp->m_peraglock); 557*2a82b8beSDavid Chinner return 1; 558*2a82b8beSDavid Chinner } 559*2a82b8beSDavid Chinner 560*2a82b8beSDavid Chinner /* If the parent directory is already in the cache, use its AG. */ 561*2a82b8beSDavid Chinner item = xfs_mru_cache_lookup(cache, pip->i_ino); 562*2a82b8beSDavid Chinner if (item) { 563*2a82b8beSDavid Chinner ASSERT(item->ip == pip); 564*2a82b8beSDavid Chinner ag = item->ag; 565*2a82b8beSDavid Chinner xfs_mru_cache_done(cache); 566*2a82b8beSDavid Chinner 567*2a82b8beSDavid Chinner TRACE_LOOKUP(mp, pip, pip, ag, xfs_filestream_peek_ag(mp, ag)); 568*2a82b8beSDavid Chinner err = _xfs_filestream_update_ag(ip, pip, ag); 569*2a82b8beSDavid Chinner 570*2a82b8beSDavid Chinner goto exit; 571*2a82b8beSDavid Chinner } 572*2a82b8beSDavid Chinner 573*2a82b8beSDavid Chinner /* 574*2a82b8beSDavid Chinner * Set the starting AG using the rotor for inode32, otherwise 575*2a82b8beSDavid Chinner * use the directory inode's AG. 576*2a82b8beSDavid Chinner */ 577*2a82b8beSDavid Chinner if (mp->m_flags & XFS_MOUNT_32BITINODES) { 578*2a82b8beSDavid Chinner rotorstep = xfs_rotorstep; 579*2a82b8beSDavid Chinner startag = (mp->m_agfrotor / rotorstep) % mp->m_sb.sb_agcount; 580*2a82b8beSDavid Chinner mp->m_agfrotor = (mp->m_agfrotor + 1) % 581*2a82b8beSDavid Chinner (mp->m_sb.sb_agcount * rotorstep); 582*2a82b8beSDavid Chinner } else 583*2a82b8beSDavid Chinner startag = XFS_INO_TO_AGNO(mp, pip->i_ino); 584*2a82b8beSDavid Chinner 585*2a82b8beSDavid Chinner /* Pick a new AG for the parent inode starting at startag. */ 586*2a82b8beSDavid Chinner err = _xfs_filestream_pick_ag(mp, startag, &ag, 0, 0); 587*2a82b8beSDavid Chinner if (err || ag == NULLAGNUMBER) 588*2a82b8beSDavid Chinner goto exit_did_pick; 589*2a82b8beSDavid Chinner 590*2a82b8beSDavid Chinner /* Associate the parent inode with the AG. */ 591*2a82b8beSDavid Chinner err = _xfs_filestream_update_ag(pip, NULL, ag); 592*2a82b8beSDavid Chinner if (err) 593*2a82b8beSDavid Chinner goto exit_did_pick; 594*2a82b8beSDavid Chinner 595*2a82b8beSDavid Chinner /* Associate the file inode with the AG. */ 596*2a82b8beSDavid Chinner err = _xfs_filestream_update_ag(ip, pip, ag); 597*2a82b8beSDavid Chinner if (err) 598*2a82b8beSDavid Chinner goto exit_did_pick; 599*2a82b8beSDavid Chinner 600*2a82b8beSDavid Chinner TRACE_ASSOCIATE(mp, ip, pip, ag, xfs_filestream_peek_ag(mp, ag)); 601*2a82b8beSDavid Chinner 602*2a82b8beSDavid Chinner exit_did_pick: 603*2a82b8beSDavid Chinner /* 604*2a82b8beSDavid Chinner * If _xfs_filestream_pick_ag() returned a valid AG, remove the 605*2a82b8beSDavid Chinner * reference it took on it, since the file and directory will have taken 606*2a82b8beSDavid Chinner * their own now if they were successfully cached. 607*2a82b8beSDavid Chinner */ 608*2a82b8beSDavid Chinner if (ag != NULLAGNUMBER) 609*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, ag); 610*2a82b8beSDavid Chinner 611*2a82b8beSDavid Chinner exit: 612*2a82b8beSDavid Chinner xfs_iunlock(pip, XFS_IOLOCK_EXCL); 613*2a82b8beSDavid Chinner up_read(&mp->m_peraglock); 614*2a82b8beSDavid Chinner return -err; 615*2a82b8beSDavid Chinner } 616*2a82b8beSDavid Chinner 617*2a82b8beSDavid Chinner /* 618*2a82b8beSDavid Chinner * Pick a new allocation group for the current file and its file stream. This 619*2a82b8beSDavid Chinner * function is called by xfs_bmap_filestreams() with the mount point's per-ag 620*2a82b8beSDavid Chinner * lock held. 621*2a82b8beSDavid Chinner */ 622*2a82b8beSDavid Chinner int 623*2a82b8beSDavid Chinner xfs_filestream_new_ag( 624*2a82b8beSDavid Chinner xfs_bmalloca_t *ap, 625*2a82b8beSDavid Chinner xfs_agnumber_t *agp) 626*2a82b8beSDavid Chinner { 627*2a82b8beSDavid Chinner int flags, err; 628*2a82b8beSDavid Chinner xfs_inode_t *ip, *pip = NULL; 629*2a82b8beSDavid Chinner xfs_mount_t *mp; 630*2a82b8beSDavid Chinner xfs_mru_cache_t *cache; 631*2a82b8beSDavid Chinner xfs_extlen_t minlen; 632*2a82b8beSDavid Chinner fstrm_item_t *dir, *file; 633*2a82b8beSDavid Chinner xfs_agnumber_t ag = NULLAGNUMBER; 634*2a82b8beSDavid Chinner 635*2a82b8beSDavid Chinner ip = ap->ip; 636*2a82b8beSDavid Chinner mp = ip->i_mount; 637*2a82b8beSDavid Chinner cache = mp->m_filestream; 638*2a82b8beSDavid Chinner minlen = ap->alen; 639*2a82b8beSDavid Chinner *agp = NULLAGNUMBER; 640*2a82b8beSDavid Chinner 641*2a82b8beSDavid Chinner /* 642*2a82b8beSDavid Chinner * Look for the file in the cache, removing it if it's found. Doing 643*2a82b8beSDavid Chinner * this allows it to be held across the dir lookup that follows. 644*2a82b8beSDavid Chinner */ 645*2a82b8beSDavid Chinner file = xfs_mru_cache_remove(cache, ip->i_ino); 646*2a82b8beSDavid Chinner if (file) { 647*2a82b8beSDavid Chinner ASSERT(ip == file->ip); 648*2a82b8beSDavid Chinner 649*2a82b8beSDavid Chinner /* Save the file's parent inode and old AG number for later. */ 650*2a82b8beSDavid Chinner pip = file->pip; 651*2a82b8beSDavid Chinner ag = file->ag; 652*2a82b8beSDavid Chinner 653*2a82b8beSDavid Chinner /* Look for the file's directory in the cache. */ 654*2a82b8beSDavid Chinner dir = xfs_mru_cache_lookup(cache, pip->i_ino); 655*2a82b8beSDavid Chinner if (dir) { 656*2a82b8beSDavid Chinner ASSERT(pip == dir->ip); 657*2a82b8beSDavid Chinner 658*2a82b8beSDavid Chinner /* 659*2a82b8beSDavid Chinner * If the directory has already moved on to a new AG, 660*2a82b8beSDavid Chinner * use that AG as the new AG for the file. Don't 661*2a82b8beSDavid Chinner * forget to twiddle the AG refcounts to match the 662*2a82b8beSDavid Chinner * movement. 663*2a82b8beSDavid Chinner */ 664*2a82b8beSDavid Chinner if (dir->ag != file->ag) { 665*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, file->ag); 666*2a82b8beSDavid Chinner xfs_filestream_get_ag(mp, dir->ag); 667*2a82b8beSDavid Chinner *agp = file->ag = dir->ag; 668*2a82b8beSDavid Chinner } 669*2a82b8beSDavid Chinner 670*2a82b8beSDavid Chinner xfs_mru_cache_done(cache); 671*2a82b8beSDavid Chinner } 672*2a82b8beSDavid Chinner 673*2a82b8beSDavid Chinner /* 674*2a82b8beSDavid Chinner * Put the file back in the cache. If this fails, the free 675*2a82b8beSDavid Chinner * function needs to be called to tidy up in the same way as if 676*2a82b8beSDavid Chinner * the item had simply expired from the cache. 677*2a82b8beSDavid Chinner */ 678*2a82b8beSDavid Chinner err = xfs_mru_cache_insert(cache, ip->i_ino, file); 679*2a82b8beSDavid Chinner if (err) { 680*2a82b8beSDavid Chinner xfs_fstrm_free_func(ip->i_ino, file); 681*2a82b8beSDavid Chinner return err; 682*2a82b8beSDavid Chinner } 683*2a82b8beSDavid Chinner 684*2a82b8beSDavid Chinner /* 685*2a82b8beSDavid Chinner * If the file's AG was moved to the directory's new AG, there's 686*2a82b8beSDavid Chinner * nothing more to be done. 687*2a82b8beSDavid Chinner */ 688*2a82b8beSDavid Chinner if (*agp != NULLAGNUMBER) { 689*2a82b8beSDavid Chinner TRACE_MOVEAG(mp, ip, pip, 690*2a82b8beSDavid Chinner ag, xfs_filestream_peek_ag(mp, ag), 691*2a82b8beSDavid Chinner *agp, xfs_filestream_peek_ag(mp, *agp)); 692*2a82b8beSDavid Chinner return 0; 693*2a82b8beSDavid Chinner } 694*2a82b8beSDavid Chinner } 695*2a82b8beSDavid Chinner 696*2a82b8beSDavid Chinner /* 697*2a82b8beSDavid Chinner * If the file's parent directory is known, take its iolock in exclusive 698*2a82b8beSDavid Chinner * mode to prevent two sibling files from racing each other to migrate 699*2a82b8beSDavid Chinner * themselves and their parent to different AGs. 700*2a82b8beSDavid Chinner */ 701*2a82b8beSDavid Chinner if (pip) 702*2a82b8beSDavid Chinner xfs_ilock(pip, XFS_IOLOCK_EXCL); 703*2a82b8beSDavid Chinner 704*2a82b8beSDavid Chinner /* 705*2a82b8beSDavid Chinner * A new AG needs to be found for the file. If the file's parent 706*2a82b8beSDavid Chinner * directory is also known, it will be moved to the new AG as well to 707*2a82b8beSDavid Chinner * ensure that files created inside it in future use the new AG. 708*2a82b8beSDavid Chinner */ 709*2a82b8beSDavid Chinner ag = (ag == NULLAGNUMBER) ? 0 : (ag + 1) % mp->m_sb.sb_agcount; 710*2a82b8beSDavid Chinner flags = (ap->userdata ? XFS_PICK_USERDATA : 0) | 711*2a82b8beSDavid Chinner (ap->low ? XFS_PICK_LOWSPACE : 0); 712*2a82b8beSDavid Chinner 713*2a82b8beSDavid Chinner err = _xfs_filestream_pick_ag(mp, ag, agp, flags, minlen); 714*2a82b8beSDavid Chinner if (err || *agp == NULLAGNUMBER) 715*2a82b8beSDavid Chinner goto exit; 716*2a82b8beSDavid Chinner 717*2a82b8beSDavid Chinner /* 718*2a82b8beSDavid Chinner * If the file wasn't found in the file cache, then its parent directory 719*2a82b8beSDavid Chinner * inode isn't known. For this to have happened, the file must either 720*2a82b8beSDavid Chinner * be pre-existing, or it was created long enough ago that its cache 721*2a82b8beSDavid Chinner * entry has expired. This isn't the sort of usage that the filestreams 722*2a82b8beSDavid Chinner * allocator is trying to optimise, so there's no point trying to track 723*2a82b8beSDavid Chinner * its new AG somehow in the filestream data structures. 724*2a82b8beSDavid Chinner */ 725*2a82b8beSDavid Chinner if (!pip) { 726*2a82b8beSDavid Chinner TRACE_ORPHAN(mp, ip, *agp); 727*2a82b8beSDavid Chinner goto exit; 728*2a82b8beSDavid Chinner } 729*2a82b8beSDavid Chinner 730*2a82b8beSDavid Chinner /* Associate the parent inode with the AG. */ 731*2a82b8beSDavid Chinner err = _xfs_filestream_update_ag(pip, NULL, *agp); 732*2a82b8beSDavid Chinner if (err) 733*2a82b8beSDavid Chinner goto exit; 734*2a82b8beSDavid Chinner 735*2a82b8beSDavid Chinner /* Associate the file inode with the AG. */ 736*2a82b8beSDavid Chinner err = _xfs_filestream_update_ag(ip, pip, *agp); 737*2a82b8beSDavid Chinner if (err) 738*2a82b8beSDavid Chinner goto exit; 739*2a82b8beSDavid Chinner 740*2a82b8beSDavid Chinner TRACE_MOVEAG(mp, ip, pip, NULLAGNUMBER, 0, 741*2a82b8beSDavid Chinner *agp, xfs_filestream_peek_ag(mp, *agp)); 742*2a82b8beSDavid Chinner 743*2a82b8beSDavid Chinner exit: 744*2a82b8beSDavid Chinner /* 745*2a82b8beSDavid Chinner * If _xfs_filestream_pick_ag() returned a valid AG, remove the 746*2a82b8beSDavid Chinner * reference it took on it, since the file and directory will have taken 747*2a82b8beSDavid Chinner * their own now if they were successfully cached. 748*2a82b8beSDavid Chinner */ 749*2a82b8beSDavid Chinner if (*agp != NULLAGNUMBER) 750*2a82b8beSDavid Chinner xfs_filestream_put_ag(mp, *agp); 751*2a82b8beSDavid Chinner else 752*2a82b8beSDavid Chinner *agp = 0; 753*2a82b8beSDavid Chinner 754*2a82b8beSDavid Chinner if (pip) 755*2a82b8beSDavid Chinner xfs_iunlock(pip, XFS_IOLOCK_EXCL); 756*2a82b8beSDavid Chinner 757*2a82b8beSDavid Chinner return err; 758*2a82b8beSDavid Chinner } 759*2a82b8beSDavid Chinner 760*2a82b8beSDavid Chinner /* 761*2a82b8beSDavid Chinner * Remove an association between an inode and a filestream object. 762*2a82b8beSDavid Chinner * Typically this is done on last close of an unlinked file. 763*2a82b8beSDavid Chinner */ 764*2a82b8beSDavid Chinner void 765*2a82b8beSDavid Chinner xfs_filestream_deassociate( 766*2a82b8beSDavid Chinner xfs_inode_t *ip) 767*2a82b8beSDavid Chinner { 768*2a82b8beSDavid Chinner xfs_mru_cache_t *cache = ip->i_mount->m_filestream; 769*2a82b8beSDavid Chinner 770*2a82b8beSDavid Chinner xfs_mru_cache_delete(cache, ip->i_ino); 771*2a82b8beSDavid Chinner } 772