xref: /linux/fs/filesystems.c (revision 6e7c1770)
1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds  *  linux/fs/filesystems.c
41da177e4SLinus Torvalds  *
51da177e4SLinus Torvalds  *  Copyright (C) 1991, 1992  Linus Torvalds
61da177e4SLinus Torvalds  *
71da177e4SLinus Torvalds  *  table of configured filesystems
81da177e4SLinus Torvalds  */
91da177e4SLinus Torvalds 
101da177e4SLinus Torvalds #include <linux/syscalls.h>
111da177e4SLinus Torvalds #include <linux/fs.h>
1268274007SAlexey Dobriyan #include <linux/proc_fs.h>
1368274007SAlexey Dobriyan #include <linux/seq_file.h>
141da177e4SLinus Torvalds #include <linux/kmod.h>
151da177e4SLinus Torvalds #include <linux/init.h>
161da177e4SLinus Torvalds #include <linux/module.h>
175a0e3ad6STejun Heo #include <linux/slab.h>
187c0f6ba6SLinus Torvalds #include <linux/uaccess.h>
193e1aeb00SDavid Howells #include <linux/fs_parser.h>
201da177e4SLinus Torvalds 
211da177e4SLinus Torvalds /*
221da177e4SLinus Torvalds  * Handling of filesystem drivers list.
231da177e4SLinus Torvalds  * Rules:
241da177e4SLinus Torvalds  *	Inclusion to/removals from/scanning of list are protected by spinlock.
251da177e4SLinus Torvalds  *	During the unload module must call unregister_filesystem().
261da177e4SLinus Torvalds  *	We can access the fields of list element if:
271da177e4SLinus Torvalds  *		1) spinlock is held or
281da177e4SLinus Torvalds  *		2) we hold the reference to the module.
291da177e4SLinus Torvalds  *	The latter can be guaranteed by call of try_module_get(); if it
301da177e4SLinus Torvalds  *	returned 0 we must skip the element, otherwise we got the reference.
311da177e4SLinus Torvalds  *	Once the reference is obtained we can drop the spinlock.
321da177e4SLinus Torvalds  */
331da177e4SLinus Torvalds 
341da177e4SLinus Torvalds static struct file_system_type *file_systems;
351da177e4SLinus Torvalds static DEFINE_RWLOCK(file_systems_lock);
361da177e4SLinus Torvalds 
371da177e4SLinus Torvalds /* WARNING: This can be used only if we _already_ own a reference */
get_filesystem(struct file_system_type * fs)38ee416bcdSDavid Howells struct file_system_type *get_filesystem(struct file_system_type *fs)
391da177e4SLinus Torvalds {
401da177e4SLinus Torvalds 	__module_get(fs->owner);
41ee416bcdSDavid Howells 	return fs;
421da177e4SLinus Torvalds }
431da177e4SLinus Torvalds 
put_filesystem(struct file_system_type * fs)441da177e4SLinus Torvalds void put_filesystem(struct file_system_type *fs)
451da177e4SLinus Torvalds {
461da177e4SLinus Torvalds 	module_put(fs->owner);
471da177e4SLinus Torvalds }
481da177e4SLinus Torvalds 
find_filesystem(const char * name,unsigned len)4979c0b2dfSMiklos Szeredi static struct file_system_type **find_filesystem(const char *name, unsigned len)
501da177e4SLinus Torvalds {
511da177e4SLinus Torvalds 	struct file_system_type **p;
521da177e4SLinus Torvalds 	for (p = &file_systems; *p; p = &(*p)->next)
53558041d8SAl Viro 		if (strncmp((*p)->name, name, len) == 0 &&
54558041d8SAl Viro 		    !(*p)->name[len])
551da177e4SLinus Torvalds 			break;
561da177e4SLinus Torvalds 	return p;
571da177e4SLinus Torvalds }
581da177e4SLinus Torvalds 
591da177e4SLinus Torvalds /**
601da177e4SLinus Torvalds  *	register_filesystem - register a new filesystem
611da177e4SLinus Torvalds  *	@fs: the file system structure
621da177e4SLinus Torvalds  *
631da177e4SLinus Torvalds  *	Adds the file system passed to the list of file systems the kernel
641da177e4SLinus Torvalds  *	is aware of for mount and other syscalls. Returns 0 on success,
651da177e4SLinus Torvalds  *	or a negative errno code on an error.
661da177e4SLinus Torvalds  *
671da177e4SLinus Torvalds  *	The &struct file_system_type that is passed is linked into the kernel
681da177e4SLinus Torvalds  *	structures and must not be freed until the file system has been
691da177e4SLinus Torvalds  *	unregistered.
701da177e4SLinus Torvalds  */
711da177e4SLinus Torvalds 
register_filesystem(struct file_system_type * fs)721da177e4SLinus Torvalds int register_filesystem(struct file_system_type * fs)
731da177e4SLinus Torvalds {
741da177e4SLinus Torvalds 	int res = 0;
751da177e4SLinus Torvalds 	struct file_system_type ** p;
761da177e4SLinus Torvalds 
7796cafb9cSEric Sandeen 	if (fs->parameters &&
7896cafb9cSEric Sandeen 	    !fs_validate_description(fs->name, fs->parameters))
793e1aeb00SDavid Howells 		return -EINVAL;
803e1aeb00SDavid Howells 
8179c0b2dfSMiklos Szeredi 	BUG_ON(strchr(fs->name, '.'));
821da177e4SLinus Torvalds 	if (fs->next)
831da177e4SLinus Torvalds 		return -EBUSY;
841da177e4SLinus Torvalds 	write_lock(&file_systems_lock);
8579c0b2dfSMiklos Szeredi 	p = find_filesystem(fs->name, strlen(fs->name));
861da177e4SLinus Torvalds 	if (*p)
871da177e4SLinus Torvalds 		res = -EBUSY;
881da177e4SLinus Torvalds 	else
891da177e4SLinus Torvalds 		*p = fs;
901da177e4SLinus Torvalds 	write_unlock(&file_systems_lock);
911da177e4SLinus Torvalds 	return res;
921da177e4SLinus Torvalds }
931da177e4SLinus Torvalds 
941da177e4SLinus Torvalds EXPORT_SYMBOL(register_filesystem);
951da177e4SLinus Torvalds 
961da177e4SLinus Torvalds /**
971da177e4SLinus Torvalds  *	unregister_filesystem - unregister a file system
981da177e4SLinus Torvalds  *	@fs: filesystem to unregister
991da177e4SLinus Torvalds  *
1001da177e4SLinus Torvalds  *	Remove a file system that was previously successfully registered
1011da177e4SLinus Torvalds  *	with the kernel. An error is returned if the file system is not found.
1021da177e4SLinus Torvalds  *	Zero is returned on a success.
1031da177e4SLinus Torvalds  *
1041da177e4SLinus Torvalds  *	Once this function has returned the &struct file_system_type structure
1051da177e4SLinus Torvalds  *	may be freed or reused.
1061da177e4SLinus Torvalds  */
1071da177e4SLinus Torvalds 
unregister_filesystem(struct file_system_type * fs)1081da177e4SLinus Torvalds int unregister_filesystem(struct file_system_type * fs)
1091da177e4SLinus Torvalds {
1101da177e4SLinus Torvalds 	struct file_system_type ** tmp;
1111da177e4SLinus Torvalds 
1121da177e4SLinus Torvalds 	write_lock(&file_systems_lock);
1131da177e4SLinus Torvalds 	tmp = &file_systems;
1141da177e4SLinus Torvalds 	while (*tmp) {
1151da177e4SLinus Torvalds 		if (fs == *tmp) {
1161da177e4SLinus Torvalds 			*tmp = fs->next;
1171da177e4SLinus Torvalds 			fs->next = NULL;
1181da177e4SLinus Torvalds 			write_unlock(&file_systems_lock);
119fff3e5adSMilton Miller 			synchronize_rcu();
1201da177e4SLinus Torvalds 			return 0;
1211da177e4SLinus Torvalds 		}
1221da177e4SLinus Torvalds 		tmp = &(*tmp)->next;
1231da177e4SLinus Torvalds 	}
1241da177e4SLinus Torvalds 	write_unlock(&file_systems_lock);
12531e6b01fSNick Piggin 
1261da177e4SLinus Torvalds 	return -EINVAL;
1271da177e4SLinus Torvalds }
1281da177e4SLinus Torvalds 
1291da177e4SLinus Torvalds EXPORT_SYMBOL(unregister_filesystem);
1301da177e4SLinus Torvalds 
1316af9f7bfSFabian Frederick #ifdef CONFIG_SYSFS_SYSCALL
fs_index(const char __user * __name)1321da177e4SLinus Torvalds static int fs_index(const char __user * __name)
1331da177e4SLinus Torvalds {
1341da177e4SLinus Torvalds 	struct file_system_type * tmp;
13591a27b2aSJeff Layton 	struct filename *name;
1361da177e4SLinus Torvalds 	int err, index;
1371da177e4SLinus Torvalds 
1381da177e4SLinus Torvalds 	name = getname(__name);
1391da177e4SLinus Torvalds 	err = PTR_ERR(name);
1401da177e4SLinus Torvalds 	if (IS_ERR(name))
1411da177e4SLinus Torvalds 		return err;
1421da177e4SLinus Torvalds 
1431da177e4SLinus Torvalds 	err = -EINVAL;
1441da177e4SLinus Torvalds 	read_lock(&file_systems_lock);
1451da177e4SLinus Torvalds 	for (tmp=file_systems, index=0 ; tmp ; tmp=tmp->next, index++) {
14691a27b2aSJeff Layton 		if (strcmp(tmp->name, name->name) == 0) {
1471da177e4SLinus Torvalds 			err = index;
1481da177e4SLinus Torvalds 			break;
1491da177e4SLinus Torvalds 		}
1501da177e4SLinus Torvalds 	}
1511da177e4SLinus Torvalds 	read_unlock(&file_systems_lock);
1521da177e4SLinus Torvalds 	putname(name);
1531da177e4SLinus Torvalds 	return err;
1541da177e4SLinus Torvalds }
1551da177e4SLinus Torvalds 
fs_name(unsigned int index,char __user * buf)1561da177e4SLinus Torvalds static int fs_name(unsigned int index, char __user * buf)
1571da177e4SLinus Torvalds {
1581da177e4SLinus Torvalds 	struct file_system_type * tmp;
1591da177e4SLinus Torvalds 	int len, res;
1601da177e4SLinus Torvalds 
1611da177e4SLinus Torvalds 	read_lock(&file_systems_lock);
1621da177e4SLinus Torvalds 	for (tmp = file_systems; tmp; tmp = tmp->next, index--)
1631da177e4SLinus Torvalds 		if (index <= 0 && try_module_get(tmp->owner))
1641da177e4SLinus Torvalds 			break;
1651da177e4SLinus Torvalds 	read_unlock(&file_systems_lock);
1661da177e4SLinus Torvalds 	if (!tmp)
1671da177e4SLinus Torvalds 		return -EINVAL;
1681da177e4SLinus Torvalds 
1691da177e4SLinus Torvalds 	/* OK, we got the reference, so we can safely block */
1701da177e4SLinus Torvalds 	len = strlen(tmp->name) + 1;
1711da177e4SLinus Torvalds 	res = copy_to_user(buf, tmp->name, len) ? -EFAULT : 0;
1721da177e4SLinus Torvalds 	put_filesystem(tmp);
1731da177e4SLinus Torvalds 	return res;
1741da177e4SLinus Torvalds }
1751da177e4SLinus Torvalds 
fs_maxindex(void)1761da177e4SLinus Torvalds static int fs_maxindex(void)
1771da177e4SLinus Torvalds {
1781da177e4SLinus Torvalds 	struct file_system_type * tmp;
1791da177e4SLinus Torvalds 	int index;
1801da177e4SLinus Torvalds 
1811da177e4SLinus Torvalds 	read_lock(&file_systems_lock);
1821da177e4SLinus Torvalds 	for (tmp = file_systems, index = 0 ; tmp ; tmp = tmp->next, index++)
1831da177e4SLinus Torvalds 		;
1841da177e4SLinus Torvalds 	read_unlock(&file_systems_lock);
1851da177e4SLinus Torvalds 	return index;
1861da177e4SLinus Torvalds }
1871da177e4SLinus Torvalds 
1881da177e4SLinus Torvalds /*
1891da177e4SLinus Torvalds  * Whee.. Weird sysv syscall.
1901da177e4SLinus Torvalds  */
SYSCALL_DEFINE3(sysfs,int,option,unsigned long,arg1,unsigned long,arg2)1911e7bfb21SHeiko Carstens SYSCALL_DEFINE3(sysfs, int, option, unsigned long, arg1, unsigned long, arg2)
1921da177e4SLinus Torvalds {
1931da177e4SLinus Torvalds 	int retval = -EINVAL;
1941da177e4SLinus Torvalds 
1951da177e4SLinus Torvalds 	switch (option) {
1961da177e4SLinus Torvalds 		case 1:
1971da177e4SLinus Torvalds 			retval = fs_index((const char __user *) arg1);
1981da177e4SLinus Torvalds 			break;
1991da177e4SLinus Torvalds 
2001da177e4SLinus Torvalds 		case 2:
2011da177e4SLinus Torvalds 			retval = fs_name(arg1, (char __user *) arg2);
2021da177e4SLinus Torvalds 			break;
2031da177e4SLinus Torvalds 
2041da177e4SLinus Torvalds 		case 3:
2051da177e4SLinus Torvalds 			retval = fs_maxindex();
2061da177e4SLinus Torvalds 			break;
2071da177e4SLinus Torvalds 	}
2081da177e4SLinus Torvalds 	return retval;
2091da177e4SLinus Torvalds }
2106af9f7bfSFabian Frederick #endif
2111da177e4SLinus Torvalds 
list_bdev_fs_names(char * buf,size_t size)212*6e7c1770SChristoph Hellwig int __init list_bdev_fs_names(char *buf, size_t size)
2131da177e4SLinus Torvalds {
214*6e7c1770SChristoph Hellwig 	struct file_system_type *p;
215*6e7c1770SChristoph Hellwig 	size_t len;
216*6e7c1770SChristoph Hellwig 	int count = 0;
2171da177e4SLinus Torvalds 
2181da177e4SLinus Torvalds 	read_lock(&file_systems_lock);
219*6e7c1770SChristoph Hellwig 	for (p = file_systems; p; p = p->next) {
220*6e7c1770SChristoph Hellwig 		if (!(p->fs_flags & FS_REQUIRES_DEV))
221*6e7c1770SChristoph Hellwig 			continue;
222*6e7c1770SChristoph Hellwig 		len = strlen(p->name) + 1;
223*6e7c1770SChristoph Hellwig 		if (len > size) {
224*6e7c1770SChristoph Hellwig 			pr_warn("%s: truncating file system list\n", __func__);
225*6e7c1770SChristoph Hellwig 			break;
226*6e7c1770SChristoph Hellwig 		}
227*6e7c1770SChristoph Hellwig 		memcpy(buf, p->name, len);
228*6e7c1770SChristoph Hellwig 		buf += len;
229*6e7c1770SChristoph Hellwig 		size -= len;
230*6e7c1770SChristoph Hellwig 		count++;
2311da177e4SLinus Torvalds 	}
2321da177e4SLinus Torvalds 	read_unlock(&file_systems_lock);
233*6e7c1770SChristoph Hellwig 	return count;
2341da177e4SLinus Torvalds }
2351da177e4SLinus Torvalds 
23668274007SAlexey Dobriyan #ifdef CONFIG_PROC_FS
filesystems_proc_show(struct seq_file * m,void * v)23768274007SAlexey Dobriyan static int filesystems_proc_show(struct seq_file *m, void *v)
23868274007SAlexey Dobriyan {
23968274007SAlexey Dobriyan 	struct file_system_type * tmp;
24068274007SAlexey Dobriyan 
24168274007SAlexey Dobriyan 	read_lock(&file_systems_lock);
24268274007SAlexey Dobriyan 	tmp = file_systems;
24368274007SAlexey Dobriyan 	while (tmp) {
24468274007SAlexey Dobriyan 		seq_printf(m, "%s\t%s\n",
24568274007SAlexey Dobriyan 			(tmp->fs_flags & FS_REQUIRES_DEV) ? "" : "nodev",
24668274007SAlexey Dobriyan 			tmp->name);
24768274007SAlexey Dobriyan 		tmp = tmp->next;
24868274007SAlexey Dobriyan 	}
24968274007SAlexey Dobriyan 	read_unlock(&file_systems_lock);
25068274007SAlexey Dobriyan 	return 0;
25168274007SAlexey Dobriyan }
25268274007SAlexey Dobriyan 
proc_filesystems_init(void)25368274007SAlexey Dobriyan static int __init proc_filesystems_init(void)
25468274007SAlexey Dobriyan {
2553f3942acSChristoph Hellwig 	proc_create_single("filesystems", 0, NULL, filesystems_proc_show);
25668274007SAlexey Dobriyan 	return 0;
25768274007SAlexey Dobriyan }
25868274007SAlexey Dobriyan module_init(proc_filesystems_init);
25968274007SAlexey Dobriyan #endif
26068274007SAlexey Dobriyan 
__get_fs_type(const char * name,int len)261d8e9650dSLi Zefan static struct file_system_type *__get_fs_type(const char *name, int len)
2621da177e4SLinus Torvalds {
2631da177e4SLinus Torvalds 	struct file_system_type *fs;
2641da177e4SLinus Torvalds 
2651da177e4SLinus Torvalds 	read_lock(&file_systems_lock);
26679c0b2dfSMiklos Szeredi 	fs = *(find_filesystem(name, len));
2671da177e4SLinus Torvalds 	if (fs && !try_module_get(fs->owner))
2681da177e4SLinus Torvalds 		fs = NULL;
2691da177e4SLinus Torvalds 	read_unlock(&file_systems_lock);
270d8e9650dSLi Zefan 	return fs;
2711da177e4SLinus Torvalds }
27279c0b2dfSMiklos Szeredi 
get_fs_type(const char * name)273d8e9650dSLi Zefan struct file_system_type *get_fs_type(const char *name)
274d8e9650dSLi Zefan {
275d8e9650dSLi Zefan 	struct file_system_type *fs;
276d8e9650dSLi Zefan 	const char *dot = strchr(name, '.');
277d8e9650dSLi Zefan 	int len = dot ? dot - name : strlen(name);
278d8e9650dSLi Zefan 
279d8e9650dSLi Zefan 	fs = __get_fs_type(name, len);
28041124db8SLuis R. Rodriguez 	if (!fs && (request_module("fs-%.*s", len, name) == 0)) {
281d8e9650dSLi Zefan 		fs = __get_fs_type(name, len);
28226c5d78cSEric Biggers 		if (!fs)
28326c5d78cSEric Biggers 			pr_warn_once("request_module fs-%.*s succeeded, but still no fs?\n",
28426c5d78cSEric Biggers 				     len, name);
28541124db8SLuis R. Rodriguez 	}
286d8e9650dSLi Zefan 
28779c0b2dfSMiklos Szeredi 	if (dot && fs && !(fs->fs_flags & FS_HAS_SUBTYPE)) {
28879c0b2dfSMiklos Szeredi 		put_filesystem(fs);
28979c0b2dfSMiklos Szeredi 		fs = NULL;
29079c0b2dfSMiklos Szeredi 	}
2911da177e4SLinus Torvalds 	return fs;
2921da177e4SLinus Torvalds }
2931da177e4SLinus Torvalds 
2941da177e4SLinus Torvalds EXPORT_SYMBOL(get_fs_type);
295