cregit-Linux how code gets into the kernel

Release 4.18 fs/bfs/dir.c

Directory: fs/bfs
// SPDX-License-Identifier: GPL-2.0
/*
 *      fs/bfs/dir.c
 *      BFS directory operations.
 *      Copyright (C) 1999,2000  Tigran Aivazian <tigran@veritas.com>
 *      Made endianness-clean by Andrew Stribblehill <ads@wompom.org> 2005
 */

#include <linux/time.h>
#include <linux/string.h>
#include <linux/fs.h>
#include <linux/buffer_head.h>
#include <linux/sched.h>
#include "bfs.h"

#undef DEBUG

#ifdef DEBUG

#define dprintf(x...)	printf(x)
#else

#define dprintf(x...)
#endif

static int bfs_add_entry(struct inode *dir, const struct qstr *child, int ino);
static struct buffer_head *bfs_find_entry(struct inode *dir,
				const struct qstr *child,
				struct bfs_dirent **res_dir);


static int bfs_readdir(struct file *f, struct dir_context *ctx) { struct inode *dir = file_inode(f); struct buffer_head *bh; struct bfs_dirent *de; unsigned int offset; int block; if (ctx->pos & (BFS_DIRENT_SIZE - 1)) { printf("Bad f_pos=%08lx for %s:%08lx\n", (unsigned long)ctx->pos, dir->i_sb->s_id, dir->i_ino); return -EINVAL; } while (ctx->pos < dir->i_size) { offset = ctx->pos & (BFS_BSIZE - 1); block = BFS_I(dir)->i_sblock + (ctx->pos >> BFS_BSIZE_BITS); bh = sb_bread(dir->i_sb, block); if (!bh) { ctx->pos += BFS_BSIZE - offset; continue; } do { de = (struct bfs_dirent *)(bh->b_data + offset); if (de->ino) { int size = strnlen(de->name, BFS_NAMELEN); if (!dir_emit(ctx, de->name, size, le16_to_cpu(de->ino), DT_UNKNOWN)) { brelse(bh); return 0; } } offset += BFS_DIRENT_SIZE; ctx->pos += BFS_DIRENT_SIZE; } while ((offset < BFS_BSIZE) && (ctx->pos < dir->i_size)); brelse(bh); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)21482.31%433.33%
Al Viro2610.00%216.67%
Linus Torvalds135.00%433.33%
Dmitri Vorobiev41.54%18.33%
Andrew Stribblehill31.15%18.33%
Total260100.00%12100.00%

const struct file_operations bfs_dir_operations = { .read = generic_read_dir, .iterate_shared = bfs_readdir, .fsync = generic_file_fsync, .llseek = generic_file_llseek, };
static int bfs_create(struct inode *dir, struct dentry *dentry, umode_t mode, bool excl) { int err; struct inode *inode; struct super_block *s = dir->i_sb; struct bfs_sb_info *info = BFS_SB(s); unsigned long ino; inode = new_inode(s); if (!inode) return -ENOMEM; mutex_lock(&info->bfs_lock); ino = find_first_zero_bit(info->si_imap, info->si_lasti + 1); if (ino > info->si_lasti) { mutex_unlock(&info->bfs_lock); iput(inode); return -ENOSPC; } set_bit(ino, info->si_imap); info->si_freei--; inode_init_owner(inode, dir, mode); inode->i_mtime = inode->i_atime = inode->i_ctime = current_time(inode); inode->i_blocks = 0; inode->i_op = &bfs_file_inops; inode->i_fop = &bfs_file_operations; inode->i_mapping->a_ops = &bfs_aops; inode->i_ino = ino; BFS_I(inode)->i_dsk_ino = ino; BFS_I(inode)->i_sblock = 0; BFS_I(inode)->i_eblock = 0; insert_inode_hash(inode); mark_inode_dirty(inode); bfs_dump_imap("create", s); err = bfs_add_entry(dir, &dentry->d_name, inode->i_ino); if (err) { inode_dec_link_count(inode); mutex_unlock(&info->bfs_lock); iput(inode); return err; } mutex_unlock(&info->bfs_lock); d_instantiate(dentry, inode); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)20668.90%525.00%
Dmitri Vorobiev289.36%15.00%
Dave Jones206.69%15.00%
Linus Torvalds165.35%15.00%
Al Viro82.68%420.00%
Dmitriy Monakhov62.01%15.00%
Eric Sesterhenn / Snakebyte51.67%15.00%
Deepa Dinamani41.34%15.00%
Akinobu Mita20.67%15.00%
Fabian Frederick10.33%15.00%
Trond Myklebust10.33%15.00%
Sanidhya Kashyap10.33%15.00%
Dave Hansen10.33%15.00%
Total299100.00%20100.00%


static struct dentry *bfs_lookup(struct inode *dir, struct dentry *dentry, unsigned int flags) { struct inode *inode = NULL; struct buffer_head *bh; struct bfs_dirent *de; struct bfs_sb_info *info = BFS_SB(dir->i_sb); if (dentry->d_name.len > BFS_NAMELEN) return ERR_PTR(-ENAMETOOLONG); mutex_lock(&info->bfs_lock); bh = bfs_find_entry(dir, &dentry->d_name, &de); if (bh) { unsigned long ino = (unsigned long)le16_to_cpu(de->ino); brelse(bh); inode = bfs_iget(dir->i_sb, ino); } mutex_unlock(&info->bfs_lock); return d_splice_alias(inode, dentry); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)10170.14%111.11%
Dmitri Vorobiev2618.06%111.11%
Al Viro96.25%444.44%
Andrew Stribblehill53.47%111.11%
David Howells21.39%111.11%
Trond Myklebust10.69%111.11%
Total144100.00%9100.00%


static int bfs_link(struct dentry *old, struct inode *dir, struct dentry *new) { struct inode *inode = d_inode(old); struct bfs_sb_info *info = BFS_SB(inode->i_sb); int err; mutex_lock(&info->bfs_lock); err = bfs_add_entry(dir, &new->d_name, inode->i_ino); if (err) { mutex_unlock(&info->bfs_lock); return err; } inc_nlink(inode); inode->i_ctime = current_time(inode); mark_inode_dirty(inode); ihold(inode); d_instantiate(new, inode); mutex_unlock(&info->bfs_lock); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)7961.24%222.22%
Dmitri Vorobiev3325.58%111.11%
Al Viro75.43%333.33%
Deepa Dinamani43.10%111.11%
David Howells32.33%111.11%
Dave Hansen32.33%111.11%
Total129100.00%9100.00%


static int bfs_unlink(struct inode *dir, struct dentry *dentry) { int error = -ENOENT; struct inode *inode = d_inode(dentry); struct buffer_head *bh; struct bfs_dirent *de; struct bfs_sb_info *info = BFS_SB(inode->i_sb); mutex_lock(&info->bfs_lock); bh = bfs_find_entry(dir, &dentry->d_name, &de); if (!bh || (le16_to_cpu(de->ino) != inode->i_ino)) goto out_brelse; if (!inode->i_nlink) { printf("unlinking non-existent file %s:%lu (nlink=%d)\n", inode->i_sb->s_id, inode->i_ino, inode->i_nlink); set_nlink(inode, 1); } de->ino = 0; mark_buffer_dirty_inode(bh, dir); dir->i_ctime = dir->i_mtime = current_time(dir); mark_inode_dirty(dir); inode->i_ctime = dir->i_ctime; inode_dec_link_count(inode); error = 0; out_brelse: brelse(bh); mutex_unlock(&info->bfs_lock); return error; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)14773.87%215.38%
Dmitri Vorobiev2814.07%215.38%
Al Viro63.02%323.08%
Miklos Szeredi42.01%17.69%
Deepa Dinamani42.01%17.69%
Andrew Stribblehill31.51%17.69%
Linus Torvalds31.51%17.69%
David Howells31.51%17.69%
Dave Hansen10.50%17.69%
Total199100.00%13100.00%


static int bfs_rename(struct inode *old_dir, struct dentry *old_dentry, struct inode *new_dir, struct dentry *new_dentry, unsigned int flags) { struct inode *old_inode, *new_inode; struct buffer_head *old_bh, *new_bh; struct bfs_dirent *old_de, *new_de; struct bfs_sb_info *info; int error = -ENOENT; if (flags & ~RENAME_NOREPLACE) return -EINVAL; old_bh = new_bh = NULL; old_inode = d_inode(old_dentry); if (S_ISDIR(old_inode->i_mode)) return -EINVAL; info = BFS_SB(old_inode->i_sb); mutex_lock(&info->bfs_lock); old_bh = bfs_find_entry(old_dir, &old_dentry->d_name, &old_de); if (!old_bh || (le16_to_cpu(old_de->ino) != old_inode->i_ino)) goto end_rename; error = -EPERM; new_inode = d_inode(new_dentry); new_bh = bfs_find_entry(new_dir, &new_dentry->d_name, &new_de); if (new_bh && !new_inode) { brelse(new_bh); new_bh = NULL; } if (!new_bh) { error = bfs_add_entry(new_dir, &new_dentry->d_name, old_inode->i_ino); if (error) goto end_rename; } old_de->ino = 0; old_dir->i_ctime = old_dir->i_mtime = current_time(old_dir); mark_inode_dirty(old_dir); if (new_inode) { new_inode->i_ctime = current_time(new_inode); inode_dec_link_count(new_inode); } mark_buffer_dirty_inode(old_bh, old_dir); error = 0; end_rename: mutex_unlock(&info->bfs_lock); brelse(old_bh); brelse(new_bh); return error; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)23776.95%18.33%
Dmitri Vorobiev309.74%216.67%
Miklos Szeredi154.87%18.33%
Deepa Dinamani82.60%18.33%
Al Viro82.60%433.33%
David Howells61.95%18.33%
Andrew Stribblehill30.97%18.33%
Dave Hansen10.32%18.33%
Total308100.00%12100.00%

const struct inode_operations bfs_dir_inops = { .create = bfs_create, .lookup = bfs_lookup, .link = bfs_link, .unlink = bfs_unlink, .rename = bfs_rename, };
static int bfs_add_entry(struct inode *dir, const struct qstr *child, int ino) { const unsigned char *name = child->name; int namelen = child->len; struct buffer_head *bh; struct bfs_dirent *de; int block, sblock, eblock, off, pos; int i; dprintf("name=%s, namelen=%d\n", name, namelen); if (!namelen) return -ENOENT; if (namelen > BFS_NAMELEN) return -ENAMETOOLONG; sblock = BFS_I(dir)->i_sblock; eblock = BFS_I(dir)->i_eblock; for (block = sblock; block <= eblock; block++) { bh = sb_bread(dir->i_sb, block); if (!bh) return -EIO; for (off = 0; off < BFS_BSIZE; off += BFS_DIRENT_SIZE) { de = (struct bfs_dirent *)(bh->b_data + off); if (!de->ino) { pos = (block - sblock) * BFS_BSIZE + off; if (pos >= dir->i_size) { dir->i_size += BFS_DIRENT_SIZE; dir->i_ctime = current_time(dir); } dir->i_mtime = current_time(dir); mark_inode_dirty(dir); de->ino = cpu_to_le16((u16)ino); for (i = 0; i < BFS_NAMELEN; i++) de->name[i] = (i < namelen) ? name[i] : 0; mark_buffer_dirty_inode(bh, dir); brelse(bh); return 0; } } brelse(bh); } return -ENOSPC; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)24480.79%218.18%
Al Viro237.62%218.18%
Linus Torvalds123.97%218.18%
Deepa Dinamani82.65%19.09%
Dmitri Vorobiev72.32%19.09%
Andrew Stribblehill61.99%19.09%
Andrew Morton10.33%19.09%
Sanidhya Kashyap10.33%19.09%
Total302100.00%11100.00%


static inline int bfs_namecmp(int len, const unsigned char *name, const char *buffer) { if ((len < BFS_NAMELEN) && buffer[len]) return 0; return !memcmp(name, buffer, len); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)4593.75%133.33%
Dmitri Vorobiev24.17%133.33%
Andrew Stribblehill12.08%133.33%
Total48100.00%3100.00%


static struct buffer_head *bfs_find_entry(struct inode *dir, const struct qstr *child, struct bfs_dirent **res_dir) { unsigned long block = 0, offset = 0; struct buffer_head *bh = NULL; struct bfs_dirent *de; const unsigned char *name = child->name; int namelen = child->len; *res_dir = NULL; if (namelen > BFS_NAMELEN) return NULL; while (block * BFS_BSIZE + offset < dir->i_size) { if (!bh) { bh = sb_bread(dir->i_sb, BFS_I(dir)->i_sblock + block); if (!bh) { block++; continue; } } de = (struct bfs_dirent *)(bh->b_data + offset); offset += BFS_DIRENT_SIZE; if (le16_to_cpu(de->ino) && bfs_namecmp(namelen, name, de->name)) { *res_dir = de; return bh; } if (offset < bh->b_size) continue; brelse(bh); bh = NULL; offset = 0; block++; } brelse(bh); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)17583.33%116.67%
Al Viro209.52%116.67%
Linus Torvalds62.86%233.33%
Dmitri Vorobiev62.86%116.67%
Andrew Stribblehill31.43%116.67%
Total210100.00%6100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)155375.90%915.79%
Dmitri Vorobiev1648.02%23.51%
Al Viro1145.57%1628.07%
Linus Torvalds502.44%47.02%
Deepa Dinamani281.37%11.75%
Andrew Stribblehill251.22%11.75%
Dave Jones211.03%23.51%
Miklos Szeredi190.93%23.51%
Art Haas160.78%11.75%
David Howells140.68%23.51%
Christoph Hellwig140.68%47.02%
Dave Hansen60.29%23.51%
Dmitriy Monakhov60.29%11.75%
Eric Sesterhenn / Snakebyte50.24%11.75%
Arjan van de Ven20.10%23.51%
Akinobu Mita20.10%11.75%
Sanidhya Kashyap20.10%11.75%
Trond Myklebust20.10%23.51%
Greg Kroah-Hartman10.05%11.75%
Andrew Morton10.05%11.75%
Fabian Frederick10.05%11.75%
Total2046100.00%57100.00%
Directory: fs/bfs
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.