// SPDX-License-Identifier: GPL-2.0 /* * Copyright (C) 2005-2020 Junjiro R. Okajima */ /* * file private data */ #include "aufs.h" void au_hfput(struct au_hfile *hf, int execed) { if (execed) allow_write_access(hf->hf_file); fput(hf->hf_file); hf->hf_file = NULL; au_lcnt_dec(&hf->hf_br->br_nfiles); hf->hf_br = NULL; } void au_set_h_fptr(struct file *file, aufs_bindex_t bindex, struct file *val) { struct au_finfo *finfo = au_fi(file); struct au_hfile *hf; struct au_fidir *fidir; fidir = finfo->fi_hdir; if (!fidir) { AuDebugOn(finfo->fi_btop != bindex); hf = &finfo->fi_htop; } else hf = fidir->fd_hfile + bindex; if (hf && hf->hf_file) au_hfput(hf, vfsub_file_execed(file)); if (val) { FiMustWriteLock(file); AuDebugOn(IS_ERR_OR_NULL(file->f_path.dentry)); hf->hf_file = val; hf->hf_br = au_sbr(file->f_path.dentry->d_sb, bindex); } } void au_update_figen(struct file *file) { atomic_set(&au_fi(file)->fi_generation, au_digen(file->f_path.dentry)); /* smp_mb(); */ /* atomic_set */ } /* ---------------------------------------------------------------------- */ struct au_fidir *au_fidir_alloc(struct super_block *sb) { struct au_fidir *fidir; int nbr; nbr = au_sbbot(sb) + 1; if (nbr < 2) nbr = 2; /* initial allocate for 2 branches */ fidir = kzalloc(au_fidir_sz(nbr), GFP_NOFS); if (fidir) { fidir->fd_bbot = -1; fidir->fd_nent = nbr; } return fidir; } int au_fidir_realloc(struct au_finfo *finfo, int nbr, int may_shrink) { int err; struct au_fidir *fidir, *p; AuRwMustWriteLock(&finfo->fi_rwsem); fidir = finfo->fi_hdir; AuDebugOn(!fidir); err = -ENOMEM; p = au_kzrealloc(fidir, au_fidir_sz(fidir->fd_nent), au_fidir_sz(nbr), GFP_NOFS, may_shrink); if (p) { p->fd_nent = nbr; finfo->fi_hdir = p; err = 0; } return err; } /* ---------------------------------------------------------------------- */ void au_finfo_fin(struct file *file) { struct au_finfo *finfo; au_lcnt_dec(&au_sbi(file->f_path.dentry->d_sb)->si_nfiles); finfo = au_fi(file); AuDebugOn(finfo->fi_hdir); AuRwDestroy(&finfo->fi_rwsem); au_cache_free_finfo(finfo); } void au_fi_init_once(void *_finfo) { struct au_finfo *finfo = _finfo; au_rw_init(&finfo->fi_rwsem); } int au_finfo_init(struct file *file, struct au_fidir *fidir) { int err; struct au_finfo *finfo; struct dentry *dentry; err = -ENOMEM; dentry = file->f_path.dentry; finfo = au_cache_alloc_finfo(); if (unlikely(!finfo)) goto out; err = 0; au_lcnt_inc(&au_sbi(dentry->d_sb)->si_nfiles); au_rw_write_lock(&finfo->fi_rwsem); finfo->fi_btop = -1; finfo->fi_hdir = fidir; atomic_set(&finfo->fi_generation, au_digen(dentry)); /* smp_mb(); */ /* atomic_set */ file->private_data = finfo; out: return err; }