xref: /openbmc/linux/fs/ramfs/inode.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
11da177e4SLinus Torvalds /*
21da177e4SLinus Torvalds  * Resizable simple ram filesystem for Linux.
31da177e4SLinus Torvalds  *
41da177e4SLinus Torvalds  * Copyright (C) 2000 Linus Torvalds.
51da177e4SLinus Torvalds  *               2000 Transmeta Corp.
61da177e4SLinus Torvalds  *
71da177e4SLinus Torvalds  * Usage limits added by David Gibson, Linuxcare Australia.
81da177e4SLinus Torvalds  * This file is released under the GPL.
91da177e4SLinus Torvalds  */
101da177e4SLinus Torvalds 
111da177e4SLinus Torvalds /*
121da177e4SLinus Torvalds  * NOTE! This filesystem is probably most useful
131da177e4SLinus Torvalds  * not as a real filesystem, but as an example of
141da177e4SLinus Torvalds  * how virtual filesystems can be written.
151da177e4SLinus Torvalds  *
161da177e4SLinus Torvalds  * It doesn't get much simpler than this. Consider
171da177e4SLinus Torvalds  * that this file implements the full semantics of
181da177e4SLinus Torvalds  * a POSIX-compliant read-write filesystem.
191da177e4SLinus Torvalds  *
201da177e4SLinus Torvalds  * Note in particular how the filesystem does not
211da177e4SLinus Torvalds  * need to implement any data structures of its own
221da177e4SLinus Torvalds  * to keep track of the virtual data: using the VFS
231da177e4SLinus Torvalds  * caches is sufficient.
241da177e4SLinus Torvalds  */
251da177e4SLinus Torvalds 
261da177e4SLinus Torvalds #include <linux/fs.h>
271da177e4SLinus Torvalds #include <linux/pagemap.h>
281da177e4SLinus Torvalds #include <linux/highmem.h>
298dde0509SAndrew Morton #include <linux/time.h>
301da177e4SLinus Torvalds #include <linux/init.h>
311da177e4SLinus Torvalds #include <linux/string.h>
321da177e4SLinus Torvalds #include <linux/backing-dev.h>
331da177e4SLinus Torvalds #include <linux/ramfs.h>
34e8edc6e0SAlexey Dobriyan #include <linux/sched.h>
35c3b1b1cbSWu Fengguang #include <linux/parser.h>
36a7e3108cSmaximilian attems #include <linux/magic.h>
375a0e3ad6STejun Heo #include <linux/slab.h>
387c0f6ba6SLinus Torvalds #include <linux/uaccess.h>
39f3235626SDavid Howells #include <linux/fs_context.h>
40f3235626SDavid Howells #include <linux/fs_parser.h>
41e41d12f5SChristoph Hellwig #include <linux/seq_file.h>
42642fb4d1SDavid Howells #include "internal.h"
431da177e4SLinus Torvalds 
44604ecf42SDavid Howells struct ramfs_mount_opts {
45604ecf42SDavid Howells 	umode_t mode;
46604ecf42SDavid Howells };
47604ecf42SDavid Howells 
48604ecf42SDavid Howells struct ramfs_fs_info {
49604ecf42SDavid Howells 	struct ramfs_mount_opts mount_opts;
50604ecf42SDavid Howells };
51604ecf42SDavid Howells 
52c3b1b1cbSWu Fengguang #define RAMFS_DEFAULT_MODE	0755
53c3b1b1cbSWu Fengguang 
54ee9b6d61SJosef 'Jeff' Sipek static const struct super_operations ramfs_ops;
55c5ef1c42SArjan van de Ven static const struct inode_operations ramfs_dir_inode_operations;
561da177e4SLinus Torvalds 
ramfs_get_inode(struct super_block * sb,const struct inode * dir,umode_t mode,dev_t dev)57454abafeSDmitry Monakhov struct inode *ramfs_get_inode(struct super_block *sb,
58632861f0SAl Viro 				const struct inode *dir, umode_t mode, dev_t dev)
591da177e4SLinus Torvalds {
601da177e4SLinus Torvalds 	struct inode * inode = new_inode(sb);
611da177e4SLinus Torvalds 
621da177e4SLinus Torvalds 	if (inode) {
6385fe4025SChristoph Hellwig 		inode->i_ino = get_next_ino();
64f2d40141SChristian Brauner 		inode_init_owner(&nop_mnt_idmap, inode, dir, mode);
65c1e3dbe9SChristoph Hellwig 		inode->i_mapping->a_ops = &ram_aops;
66769848c0SMel Gorman 		mapping_set_gfp_mask(inode->i_mapping, GFP_HIGHUSER);
67ba9ddf49SLee Schermerhorn 		mapping_set_unevictable(inode->i_mapping);
68*9f590685SJeff Layton 		inode->i_atime = inode->i_mtime = inode_set_ctime_current(inode);
691da177e4SLinus Torvalds 		switch (mode & S_IFMT) {
701da177e4SLinus Torvalds 		default:
711da177e4SLinus Torvalds 			init_special_inode(inode, mode, dev);
721da177e4SLinus Torvalds 			break;
731da177e4SLinus Torvalds 		case S_IFREG:
741da177e4SLinus Torvalds 			inode->i_op = &ramfs_file_inode_operations;
751da177e4SLinus Torvalds 			inode->i_fop = &ramfs_file_operations;
761da177e4SLinus Torvalds 			break;
771da177e4SLinus Torvalds 		case S_IFDIR:
781da177e4SLinus Torvalds 			inode->i_op = &ramfs_dir_inode_operations;
791da177e4SLinus Torvalds 			inode->i_fop = &simple_dir_operations;
801da177e4SLinus Torvalds 
811da177e4SLinus Torvalds 			/* directory inodes start off with i_nlink == 2 (for "." entry) */
82d8c76e6fSDave Hansen 			inc_nlink(inode);
831da177e4SLinus Torvalds 			break;
841da177e4SLinus Torvalds 		case S_IFLNK:
851da177e4SLinus Torvalds 			inode->i_op = &page_symlink_inode_operations;
8621fc61c7SAl Viro 			inode_nohighmem(inode);
871da177e4SLinus Torvalds 			break;
881da177e4SLinus Torvalds 		}
891da177e4SLinus Torvalds 	}
901da177e4SLinus Torvalds 	return inode;
911da177e4SLinus Torvalds }
921da177e4SLinus Torvalds 
931da177e4SLinus Torvalds /*
941da177e4SLinus Torvalds  * File creation. Allocate an inode, and we're done..
951da177e4SLinus Torvalds  */
961da177e4SLinus Torvalds /* SMP-safe */
971da177e4SLinus Torvalds static int
ramfs_mknod(struct mnt_idmap * idmap,struct inode * dir,struct dentry * dentry,umode_t mode,dev_t dev)985ebb29beSChristian Brauner ramfs_mknod(struct mnt_idmap *idmap, struct inode *dir,
99549c7297SChristian Brauner 	    struct dentry *dentry, umode_t mode, dev_t dev)
1001da177e4SLinus Torvalds {
101454abafeSDmitry Monakhov 	struct inode * inode = ramfs_get_inode(dir->i_sb, dir, mode, dev);
1021da177e4SLinus Torvalds 	int error = -ENOSPC;
1031da177e4SLinus Torvalds 
1041da177e4SLinus Torvalds 	if (inode) {
1051da177e4SLinus Torvalds 		d_instantiate(dentry, inode);
1061da177e4SLinus Torvalds 		dget(dentry);	/* Extra count - pin the dentry in core */
1071da177e4SLinus Torvalds 		error = 0;
108*9f590685SJeff Layton 		dir->i_mtime = inode_set_ctime_current(dir);
1091da177e4SLinus Torvalds 	}
1101da177e4SLinus Torvalds 	return error;
1111da177e4SLinus Torvalds }
1121da177e4SLinus Torvalds 
ramfs_mkdir(struct mnt_idmap * idmap,struct inode * dir,struct dentry * dentry,umode_t mode)113c54bd91eSChristian Brauner static int ramfs_mkdir(struct mnt_idmap *idmap, struct inode *dir,
114549c7297SChristian Brauner 		       struct dentry *dentry, umode_t mode)
1151da177e4SLinus Torvalds {
1165ebb29beSChristian Brauner 	int retval = ramfs_mknod(&nop_mnt_idmap, dir, dentry, mode | S_IFDIR, 0);
1171da177e4SLinus Torvalds 	if (!retval)
118d8c76e6fSDave Hansen 		inc_nlink(dir);
1191da177e4SLinus Torvalds 	return retval;
1201da177e4SLinus Torvalds }
1211da177e4SLinus Torvalds 
ramfs_create(struct mnt_idmap * idmap,struct inode * dir,struct dentry * dentry,umode_t mode,bool excl)1226c960e68SChristian Brauner static int ramfs_create(struct mnt_idmap *idmap, struct inode *dir,
123549c7297SChristian Brauner 			struct dentry *dentry, umode_t mode, bool excl)
1241da177e4SLinus Torvalds {
1255ebb29beSChristian Brauner 	return ramfs_mknod(&nop_mnt_idmap, dir, dentry, mode | S_IFREG, 0);
1261da177e4SLinus Torvalds }
1271da177e4SLinus Torvalds 
ramfs_symlink(struct mnt_idmap * idmap,struct inode * dir,struct dentry * dentry,const char * symname)1287a77db95SChristian Brauner static int ramfs_symlink(struct mnt_idmap *idmap, struct inode *dir,
129549c7297SChristian Brauner 			 struct dentry *dentry, const char *symname)
1301da177e4SLinus Torvalds {
1311da177e4SLinus Torvalds 	struct inode *inode;
1321da177e4SLinus Torvalds 	int error = -ENOSPC;
1331da177e4SLinus Torvalds 
134454abafeSDmitry Monakhov 	inode = ramfs_get_inode(dir->i_sb, dir, S_IFLNK|S_IRWXUGO, 0);
1351da177e4SLinus Torvalds 	if (inode) {
1361da177e4SLinus Torvalds 		int l = strlen(symname)+1;
1371da177e4SLinus Torvalds 		error = page_symlink(inode, symname, l);
1381da177e4SLinus Torvalds 		if (!error) {
1391da177e4SLinus Torvalds 			d_instantiate(dentry, inode);
1401da177e4SLinus Torvalds 			dget(dentry);
141*9f590685SJeff Layton 			dir->i_mtime = inode_set_ctime_current(dir);
1421da177e4SLinus Torvalds 		} else
1431da177e4SLinus Torvalds 			iput(inode);
1441da177e4SLinus Torvalds 	}
1451da177e4SLinus Torvalds 	return error;
1461da177e4SLinus Torvalds }
1471da177e4SLinus Torvalds 
ramfs_tmpfile(struct mnt_idmap * idmap,struct inode * dir,struct file * file,umode_t mode)148011e2b71SChristian Brauner static int ramfs_tmpfile(struct mnt_idmap *idmap,
149863f144fSMiklos Szeredi 			 struct inode *dir, struct file *file, umode_t mode)
15093da4003SAlexey Dobriyan {
15193da4003SAlexey Dobriyan 	struct inode *inode;
15293da4003SAlexey Dobriyan 
15393da4003SAlexey Dobriyan 	inode = ramfs_get_inode(dir->i_sb, dir, mode, 0);
15493da4003SAlexey Dobriyan 	if (!inode)
15593da4003SAlexey Dobriyan 		return -ENOSPC;
156863f144fSMiklos Szeredi 	d_tmpfile(file, inode);
157863f144fSMiklos Szeredi 	return finish_open_simple(file, 0);
15893da4003SAlexey Dobriyan }
15993da4003SAlexey Dobriyan 
160c5ef1c42SArjan van de Ven static const struct inode_operations ramfs_dir_inode_operations = {
1611da177e4SLinus Torvalds 	.create		= ramfs_create,
1621da177e4SLinus Torvalds 	.lookup		= simple_lookup,
1631da177e4SLinus Torvalds 	.link		= simple_link,
1641da177e4SLinus Torvalds 	.unlink		= simple_unlink,
1651da177e4SLinus Torvalds 	.symlink	= ramfs_symlink,
1661da177e4SLinus Torvalds 	.mkdir		= ramfs_mkdir,
1671da177e4SLinus Torvalds 	.rmdir		= simple_rmdir,
1681da177e4SLinus Torvalds 	.mknod		= ramfs_mknod,
1691da177e4SLinus Torvalds 	.rename		= simple_rename,
17093da4003SAlexey Dobriyan 	.tmpfile	= ramfs_tmpfile,
1711da177e4SLinus Torvalds };
1721da177e4SLinus Torvalds 
173604ecf42SDavid Howells /*
174604ecf42SDavid Howells  * Display the mount options in /proc/mounts.
175604ecf42SDavid Howells  */
ramfs_show_options(struct seq_file * m,struct dentry * root)176604ecf42SDavid Howells static int ramfs_show_options(struct seq_file *m, struct dentry *root)
177604ecf42SDavid Howells {
178604ecf42SDavid Howells 	struct ramfs_fs_info *fsi = root->d_sb->s_fs_info;
179604ecf42SDavid Howells 
180604ecf42SDavid Howells 	if (fsi->mount_opts.mode != RAMFS_DEFAULT_MODE)
181604ecf42SDavid Howells 		seq_printf(m, ",mode=%o", fsi->mount_opts.mode);
182604ecf42SDavid Howells 	return 0;
183604ecf42SDavid Howells }
184604ecf42SDavid Howells 
185ee9b6d61SJosef 'Jeff' Sipek static const struct super_operations ramfs_ops = {
1861da177e4SLinus Torvalds 	.statfs		= simple_statfs,
1871da177e4SLinus Torvalds 	.drop_inode	= generic_delete_inode,
188604ecf42SDavid Howells 	.show_options	= ramfs_show_options,
189c3b1b1cbSWu Fengguang };
190c3b1b1cbSWu Fengguang 
191f3235626SDavid Howells enum ramfs_param {
192c3b1b1cbSWu Fengguang 	Opt_mode,
193c3b1b1cbSWu Fengguang };
194c3b1b1cbSWu Fengguang 
195d7167b14SAl Viro const struct fs_parameter_spec ramfs_fs_parameters[] = {
196f3235626SDavid Howells 	fsparam_u32oct("mode",	Opt_mode),
197f3235626SDavid Howells 	{}
198c3b1b1cbSWu Fengguang };
199c3b1b1cbSWu Fengguang 
ramfs_parse_param(struct fs_context * fc,struct fs_parameter * param)200f3235626SDavid Howells static int ramfs_parse_param(struct fs_context *fc, struct fs_parameter *param)
201c3b1b1cbSWu Fengguang {
202f3235626SDavid Howells 	struct fs_parse_result result;
203f3235626SDavid Howells 	struct ramfs_fs_info *fsi = fc->s_fs_info;
204f3235626SDavid Howells 	int opt;
205c3b1b1cbSWu Fengguang 
206d7167b14SAl Viro 	opt = fs_parse(fc, ramfs_fs_parameters, param, &result);
2070858d7daSyangerkun 	if (opt == -ENOPARAM) {
2080858d7daSyangerkun 		opt = vfs_parse_fs_param_source(fc, param);
2090858d7daSyangerkun 		if (opt != -ENOPARAM)
2100858d7daSyangerkun 			return opt;
2110a8eba9bSMike Frysinger 		/*
2120a8eba9bSMike Frysinger 		 * We might like to report bad mount options here;
2130a8eba9bSMike Frysinger 		 * but traditionally ramfs has ignored all mount options,
2140a8eba9bSMike Frysinger 		 * and as it is used as a !CONFIG_SHMEM simple substitute
2150a8eba9bSMike Frysinger 		 * for tmpfs, better continue to ignore other mount options.
2160a8eba9bSMike Frysinger 		 */
2170858d7daSyangerkun 		return 0;
218c3b1b1cbSWu Fengguang 	}
2190858d7daSyangerkun 	if (opt < 0)
2200858d7daSyangerkun 		return opt;
221f3235626SDavid Howells 
222f3235626SDavid Howells 	switch (opt) {
223f3235626SDavid Howells 	case Opt_mode:
224f3235626SDavid Howells 		fsi->mount_opts.mode = result.uint_32 & S_IALLUGO;
225f3235626SDavid Howells 		break;
226c3b1b1cbSWu Fengguang 	}
227c3b1b1cbSWu Fengguang 
228c3b1b1cbSWu Fengguang 	return 0;
229c3b1b1cbSWu Fengguang }
230c3b1b1cbSWu Fengguang 
ramfs_fill_super(struct super_block * sb,struct fs_context * fc)231f3235626SDavid Howells static int ramfs_fill_super(struct super_block *sb, struct fs_context *fc)
2321da177e4SLinus Torvalds {
233f3235626SDavid Howells 	struct ramfs_fs_info *fsi = sb->s_fs_info;
234318ceed0SAl Viro 	struct inode *inode;
2351da177e4SLinus Torvalds 
2361da177e4SLinus Torvalds 	sb->s_maxbytes		= MAX_LFS_FILESIZE;
23709cbfeafSKirill A. Shutemov 	sb->s_blocksize		= PAGE_SIZE;
23809cbfeafSKirill A. Shutemov 	sb->s_blocksize_bits	= PAGE_SHIFT;
2391da177e4SLinus Torvalds 	sb->s_magic		= RAMFS_MAGIC;
2401da177e4SLinus Torvalds 	sb->s_op		= &ramfs_ops;
2411da177e4SLinus Torvalds 	sb->s_time_gran		= 1;
242f8201abcSIngo Molnar 
243454abafeSDmitry Monakhov 	inode = ramfs_get_inode(sb, NULL, S_IFDIR | fsi->mount_opts.mode, 0);
24448fde701SAl Viro 	sb->s_root = d_make_root(inode);
245318ceed0SAl Viro 	if (!sb->s_root)
246318ceed0SAl Viro 		return -ENOMEM;
247f8201abcSIngo Molnar 
2481da177e4SLinus Torvalds 	return 0;
2491da177e4SLinus Torvalds }
2501da177e4SLinus Torvalds 
ramfs_get_tree(struct fs_context * fc)251f3235626SDavid Howells static int ramfs_get_tree(struct fs_context *fc)
2521da177e4SLinus Torvalds {
253f3235626SDavid Howells 	return get_tree_nodev(fc, ramfs_fill_super);
254f3235626SDavid Howells }
255f3235626SDavid Howells 
ramfs_free_fc(struct fs_context * fc)256f3235626SDavid Howells static void ramfs_free_fc(struct fs_context *fc)
257f3235626SDavid Howells {
258f3235626SDavid Howells 	kfree(fc->s_fs_info);
259f3235626SDavid Howells }
260f3235626SDavid Howells 
261f3235626SDavid Howells static const struct fs_context_operations ramfs_context_ops = {
262f3235626SDavid Howells 	.free		= ramfs_free_fc,
263f3235626SDavid Howells 	.parse_param	= ramfs_parse_param,
264f3235626SDavid Howells 	.get_tree	= ramfs_get_tree,
265f3235626SDavid Howells };
266f3235626SDavid Howells 
ramfs_init_fs_context(struct fs_context * fc)267f3235626SDavid Howells int ramfs_init_fs_context(struct fs_context *fc)
268f3235626SDavid Howells {
269f3235626SDavid Howells 	struct ramfs_fs_info *fsi;
270f3235626SDavid Howells 
271f3235626SDavid Howells 	fsi = kzalloc(sizeof(*fsi), GFP_KERNEL);
272f3235626SDavid Howells 	if (!fsi)
273f3235626SDavid Howells 		return -ENOMEM;
274f3235626SDavid Howells 
275f3235626SDavid Howells 	fsi->mount_opts.mode = RAMFS_DEFAULT_MODE;
276f3235626SDavid Howells 	fc->s_fs_info = fsi;
277f3235626SDavid Howells 	fc->ops = &ramfs_context_ops;
278f3235626SDavid Howells 	return 0;
2791da177e4SLinus Torvalds }
2801da177e4SLinus Torvalds 
ramfs_kill_sb(struct super_block * sb)28136ce9d76SRoberto Sassu void ramfs_kill_sb(struct super_block *sb)
282c3b1b1cbSWu Fengguang {
283c3b1b1cbSWu Fengguang 	kfree(sb->s_fs_info);
284c3b1b1cbSWu Fengguang 	kill_litter_super(sb);
285c3b1b1cbSWu Fengguang }
286c3b1b1cbSWu Fengguang 
2871da177e4SLinus Torvalds static struct file_system_type ramfs_fs_type = {
2881da177e4SLinus Torvalds 	.name		= "ramfs",
289f3235626SDavid Howells 	.init_fs_context = ramfs_init_fs_context,
290d7167b14SAl Viro 	.parameters	= ramfs_fs_parameters,
291c3b1b1cbSWu Fengguang 	.kill_sb	= ramfs_kill_sb,
292b3c6761dSEric W. Biederman 	.fs_flags	= FS_USERNS_MOUNT,
2931da177e4SLinus Torvalds };
2941da177e4SLinus Torvalds 
init_ramfs_fs(void)29514a253ceSAl Viro static int __init init_ramfs_fs(void)
2961da177e4SLinus Torvalds {
297b4caecd4SChristoph Hellwig 	return register_filesystem(&ramfs_fs_type);
2981da177e4SLinus Torvalds }
299af52b040SPaul Gortmaker fs_initcall(init_ramfs_fs);
300