|  | // SPDX-License-Identifier: GPL-2.0+ | 
|  | /* | 
|  | * ifile.c - NILFS inode file | 
|  | * | 
|  | * Copyright (C) 2006-2008 Nippon Telegraph and Telephone Corporation. | 
|  | * | 
|  | * Written by Amagai Yoshiji. | 
|  | * Revised by Ryusuke Konishi. | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include <linux/types.h> | 
|  | #include <linux/buffer_head.h> | 
|  | #include "nilfs.h" | 
|  | #include "mdt.h" | 
|  | #include "alloc.h" | 
|  | #include "ifile.h" | 
|  |  | 
|  | /** | 
|  | * struct nilfs_ifile_info - on-memory private data of ifile | 
|  | * @mi: on-memory private data of metadata file | 
|  | * @palloc_cache: persistent object allocator cache of ifile | 
|  | */ | 
|  | struct nilfs_ifile_info { | 
|  | struct nilfs_mdt_info mi; | 
|  | struct nilfs_palloc_cache palloc_cache; | 
|  | }; | 
|  |  | 
|  | static inline struct nilfs_ifile_info *NILFS_IFILE_I(struct inode *ifile) | 
|  | { | 
|  | return (struct nilfs_ifile_info *)NILFS_MDT(ifile); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * nilfs_ifile_create_inode - create a new disk inode | 
|  | * @ifile: ifile inode | 
|  | * @out_ino: pointer to a variable to store inode number | 
|  | * @out_bh: buffer_head contains newly allocated disk inode | 
|  | * | 
|  | * Return Value: On success, 0 is returned and the newly allocated inode | 
|  | * number is stored in the place pointed by @ino, and buffer_head pointer | 
|  | * that contains newly allocated disk inode structure is stored in the | 
|  | * place pointed by @out_bh | 
|  | * On error, one of the following negative error codes is returned. | 
|  | * | 
|  | * %-EIO - I/O error. | 
|  | * | 
|  | * %-ENOMEM - Insufficient amount of memory available. | 
|  | * | 
|  | * %-ENOSPC - No inode left. | 
|  | */ | 
|  | int nilfs_ifile_create_inode(struct inode *ifile, ino_t *out_ino, | 
|  | struct buffer_head **out_bh) | 
|  | { | 
|  | struct nilfs_palloc_req req; | 
|  | int ret; | 
|  |  | 
|  | req.pr_entry_nr = 0;  /* | 
|  | * 0 says find free inode from beginning | 
|  | * of a group. dull code!! | 
|  | */ | 
|  | req.pr_entry_bh = NULL; | 
|  |  | 
|  | ret = nilfs_palloc_prepare_alloc_entry(ifile, &req); | 
|  | if (!ret) { | 
|  | ret = nilfs_palloc_get_entry_block(ifile, req.pr_entry_nr, 1, | 
|  | &req.pr_entry_bh); | 
|  | if (ret < 0) | 
|  | nilfs_palloc_abort_alloc_entry(ifile, &req); | 
|  | } | 
|  | if (ret < 0) { | 
|  | brelse(req.pr_entry_bh); | 
|  | return ret; | 
|  | } | 
|  | nilfs_palloc_commit_alloc_entry(ifile, &req); | 
|  | mark_buffer_dirty(req.pr_entry_bh); | 
|  | nilfs_mdt_mark_dirty(ifile); | 
|  | *out_ino = (ino_t)req.pr_entry_nr; | 
|  | *out_bh = req.pr_entry_bh; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * nilfs_ifile_delete_inode - delete a disk inode | 
|  | * @ifile: ifile inode | 
|  | * @ino: inode number | 
|  | * | 
|  | * Return Value: On success, 0 is returned. On error, one of the following | 
|  | * negative error codes is returned. | 
|  | * | 
|  | * %-EIO - I/O error. | 
|  | * | 
|  | * %-ENOMEM - Insufficient amount of memory available. | 
|  | * | 
|  | * %-ENOENT - The inode number @ino have not been allocated. | 
|  | */ | 
|  | int nilfs_ifile_delete_inode(struct inode *ifile, ino_t ino) | 
|  | { | 
|  | struct nilfs_palloc_req req = { | 
|  | .pr_entry_nr = ino, .pr_entry_bh = NULL | 
|  | }; | 
|  | struct nilfs_inode *raw_inode; | 
|  | void *kaddr; | 
|  | int ret; | 
|  |  | 
|  | ret = nilfs_palloc_prepare_free_entry(ifile, &req); | 
|  | if (!ret) { | 
|  | ret = nilfs_palloc_get_entry_block(ifile, req.pr_entry_nr, 0, | 
|  | &req.pr_entry_bh); | 
|  | if (ret < 0) | 
|  | nilfs_palloc_abort_free_entry(ifile, &req); | 
|  | } | 
|  | if (ret < 0) { | 
|  | brelse(req.pr_entry_bh); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | kaddr = kmap_atomic(req.pr_entry_bh->b_page); | 
|  | raw_inode = nilfs_palloc_block_get_entry(ifile, req.pr_entry_nr, | 
|  | req.pr_entry_bh, kaddr); | 
|  | raw_inode->i_flags = 0; | 
|  | kunmap_atomic(kaddr); | 
|  |  | 
|  | mark_buffer_dirty(req.pr_entry_bh); | 
|  | brelse(req.pr_entry_bh); | 
|  |  | 
|  | nilfs_palloc_commit_free_entry(ifile, &req); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int nilfs_ifile_get_inode_block(struct inode *ifile, ino_t ino, | 
|  | struct buffer_head **out_bh) | 
|  | { | 
|  | struct super_block *sb = ifile->i_sb; | 
|  | int err; | 
|  |  | 
|  | if (unlikely(!NILFS_VALID_INODE(sb, ino))) { | 
|  | nilfs_error(sb, "bad inode number: %lu", (unsigned long)ino); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = nilfs_palloc_get_entry_block(ifile, ino, 0, out_bh); | 
|  | if (unlikely(err)) | 
|  | nilfs_warn(sb, "error %d reading inode: ino=%lu", | 
|  | err, (unsigned long)ino); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * nilfs_ifile_count_free_inodes - calculate free inodes count | 
|  | * @ifile: ifile inode | 
|  | * @nmaxinodes: current maximum of available inodes count [out] | 
|  | * @nfreeinodes: free inodes count [out] | 
|  | */ | 
|  | int nilfs_ifile_count_free_inodes(struct inode *ifile, | 
|  | u64 *nmaxinodes, u64 *nfreeinodes) | 
|  | { | 
|  | u64 nused; | 
|  | int err; | 
|  |  | 
|  | *nmaxinodes = 0; | 
|  | *nfreeinodes = 0; | 
|  |  | 
|  | nused = atomic64_read(&NILFS_I(ifile)->i_root->inodes_count); | 
|  | err = nilfs_palloc_count_max_entries(ifile, nused, nmaxinodes); | 
|  | if (likely(!err)) | 
|  | *nfreeinodes = *nmaxinodes - nused; | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * nilfs_ifile_read - read or get ifile inode | 
|  | * @sb: super block instance | 
|  | * @root: root object | 
|  | * @inode_size: size of an inode | 
|  | * @raw_inode: on-disk ifile inode | 
|  | * @inodep: buffer to store the inode | 
|  | */ | 
|  | int nilfs_ifile_read(struct super_block *sb, struct nilfs_root *root, | 
|  | size_t inode_size, struct nilfs_inode *raw_inode, | 
|  | struct inode **inodep) | 
|  | { | 
|  | struct inode *ifile; | 
|  | int err; | 
|  |  | 
|  | ifile = nilfs_iget_locked(sb, root, NILFS_IFILE_INO); | 
|  | if (unlikely(!ifile)) | 
|  | return -ENOMEM; | 
|  | if (!(ifile->i_state & I_NEW)) | 
|  | goto out; | 
|  |  | 
|  | err = nilfs_mdt_init(ifile, NILFS_MDT_GFP, | 
|  | sizeof(struct nilfs_ifile_info)); | 
|  | if (err) | 
|  | goto failed; | 
|  |  | 
|  | err = nilfs_palloc_init_blockgroup(ifile, inode_size); | 
|  | if (err) | 
|  | goto failed; | 
|  |  | 
|  | nilfs_palloc_setup_cache(ifile, &NILFS_IFILE_I(ifile)->palloc_cache); | 
|  |  | 
|  | err = nilfs_read_inode_common(ifile, raw_inode); | 
|  | if (err) | 
|  | goto failed; | 
|  |  | 
|  | unlock_new_inode(ifile); | 
|  | out: | 
|  | *inodep = ifile; | 
|  | return 0; | 
|  | failed: | 
|  | iget_failed(ifile); | 
|  | return err; | 
|  | } |