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