xref: /openbmc/linux/fs/smb/server/mgmt/share_config.c (revision 5698ba69)
138c8a9a5SSteve French // SPDX-License-Identifier: GPL-2.0-or-later
238c8a9a5SSteve French /*
338c8a9a5SSteve French  *   Copyright (C) 2018 Samsung Electronics Co., Ltd.
438c8a9a5SSteve French  */
538c8a9a5SSteve French 
638c8a9a5SSteve French #include <linux/list.h>
738c8a9a5SSteve French #include <linux/jhash.h>
838c8a9a5SSteve French #include <linux/slab.h>
938c8a9a5SSteve French #include <linux/rwsem.h>
1038c8a9a5SSteve French #include <linux/parser.h>
1138c8a9a5SSteve French #include <linux/namei.h>
1238c8a9a5SSteve French #include <linux/sched.h>
1338c8a9a5SSteve French #include <linux/mm.h>
1438c8a9a5SSteve French 
1538c8a9a5SSteve French #include "share_config.h"
1638c8a9a5SSteve French #include "user_config.h"
1738c8a9a5SSteve French #include "user_session.h"
1838c8a9a5SSteve French #include "../transport_ipc.h"
1938c8a9a5SSteve French #include "../misc.h"
2038c8a9a5SSteve French 
2138c8a9a5SSteve French #define SHARE_HASH_BITS		3
2238c8a9a5SSteve French static DEFINE_HASHTABLE(shares_table, SHARE_HASH_BITS);
2338c8a9a5SSteve French static DECLARE_RWSEM(shares_table_lock);
2438c8a9a5SSteve French 
2538c8a9a5SSteve French struct ksmbd_veto_pattern {
2638c8a9a5SSteve French 	char			*pattern;
2738c8a9a5SSteve French 	struct list_head	list;
2838c8a9a5SSteve French };
2938c8a9a5SSteve French 
share_name_hash(const char * name)3038c8a9a5SSteve French static unsigned int share_name_hash(const char *name)
3138c8a9a5SSteve French {
3238c8a9a5SSteve French 	return jhash(name, strlen(name), 0);
3338c8a9a5SSteve French }
3438c8a9a5SSteve French 
kill_share(struct ksmbd_share_config * share)3538c8a9a5SSteve French static void kill_share(struct ksmbd_share_config *share)
3638c8a9a5SSteve French {
3738c8a9a5SSteve French 	while (!list_empty(&share->veto_list)) {
3838c8a9a5SSteve French 		struct ksmbd_veto_pattern *p;
3938c8a9a5SSteve French 
4038c8a9a5SSteve French 		p = list_entry(share->veto_list.next,
4138c8a9a5SSteve French 			       struct ksmbd_veto_pattern,
4238c8a9a5SSteve French 			       list);
4338c8a9a5SSteve French 		list_del(&p->list);
4438c8a9a5SSteve French 		kfree(p->pattern);
4538c8a9a5SSteve French 		kfree(p);
4638c8a9a5SSteve French 	}
4738c8a9a5SSteve French 
4838c8a9a5SSteve French 	if (share->path)
4938c8a9a5SSteve French 		path_put(&share->vfs_path);
5038c8a9a5SSteve French 	kfree(share->name);
5138c8a9a5SSteve French 	kfree(share->path);
5238c8a9a5SSteve French 	kfree(share);
5338c8a9a5SSteve French }
5438c8a9a5SSteve French 
ksmbd_share_config_del(struct ksmbd_share_config * share)5538c8a9a5SSteve French void ksmbd_share_config_del(struct ksmbd_share_config *share)
5638c8a9a5SSteve French {
5738c8a9a5SSteve French 	down_write(&shares_table_lock);
5838c8a9a5SSteve French 	hash_del(&share->hlist);
5938c8a9a5SSteve French 	up_write(&shares_table_lock);
6038c8a9a5SSteve French }
6138c8a9a5SSteve French 
__ksmbd_share_config_put(struct ksmbd_share_config * share)6238c8a9a5SSteve French void __ksmbd_share_config_put(struct ksmbd_share_config *share)
6338c8a9a5SSteve French {
6438c8a9a5SSteve French 	ksmbd_share_config_del(share);
6538c8a9a5SSteve French 	kill_share(share);
6638c8a9a5SSteve French }
6738c8a9a5SSteve French 
6838c8a9a5SSteve French static struct ksmbd_share_config *
__get_share_config(struct ksmbd_share_config * share)6938c8a9a5SSteve French __get_share_config(struct ksmbd_share_config *share)
7038c8a9a5SSteve French {
7138c8a9a5SSteve French 	if (!atomic_inc_not_zero(&share->refcount))
7238c8a9a5SSteve French 		return NULL;
7338c8a9a5SSteve French 	return share;
7438c8a9a5SSteve French }
7538c8a9a5SSteve French 
__share_lookup(const char * name)7638c8a9a5SSteve French static struct ksmbd_share_config *__share_lookup(const char *name)
7738c8a9a5SSteve French {
7838c8a9a5SSteve French 	struct ksmbd_share_config *share;
7938c8a9a5SSteve French 	unsigned int key = share_name_hash(name);
8038c8a9a5SSteve French 
8138c8a9a5SSteve French 	hash_for_each_possible(shares_table, share, hlist, key) {
8238c8a9a5SSteve French 		if (!strcmp(name, share->name))
8338c8a9a5SSteve French 			return share;
8438c8a9a5SSteve French 	}
8538c8a9a5SSteve French 	return NULL;
8638c8a9a5SSteve French }
8738c8a9a5SSteve French 
parse_veto_list(struct ksmbd_share_config * share,char * veto_list,int veto_list_sz)8838c8a9a5SSteve French static int parse_veto_list(struct ksmbd_share_config *share,
8938c8a9a5SSteve French 			   char *veto_list,
9038c8a9a5SSteve French 			   int veto_list_sz)
9138c8a9a5SSteve French {
9238c8a9a5SSteve French 	int sz = 0;
9338c8a9a5SSteve French 
9438c8a9a5SSteve French 	if (!veto_list_sz)
9538c8a9a5SSteve French 		return 0;
9638c8a9a5SSteve French 
9738c8a9a5SSteve French 	while (veto_list_sz > 0) {
9838c8a9a5SSteve French 		struct ksmbd_veto_pattern *p;
9938c8a9a5SSteve French 
10038c8a9a5SSteve French 		sz = strlen(veto_list);
10138c8a9a5SSteve French 		if (!sz)
10238c8a9a5SSteve French 			break;
10338c8a9a5SSteve French 
10438c8a9a5SSteve French 		p = kzalloc(sizeof(struct ksmbd_veto_pattern), GFP_KERNEL);
10538c8a9a5SSteve French 		if (!p)
10638c8a9a5SSteve French 			return -ENOMEM;
10738c8a9a5SSteve French 
10838c8a9a5SSteve French 		p->pattern = kstrdup(veto_list, GFP_KERNEL);
10938c8a9a5SSteve French 		if (!p->pattern) {
11038c8a9a5SSteve French 			kfree(p);
11138c8a9a5SSteve French 			return -ENOMEM;
11238c8a9a5SSteve French 		}
11338c8a9a5SSteve French 
11438c8a9a5SSteve French 		list_add(&p->list, &share->veto_list);
11538c8a9a5SSteve French 
11638c8a9a5SSteve French 		veto_list += sz + 1;
11738c8a9a5SSteve French 		veto_list_sz -= (sz + 1);
11838c8a9a5SSteve French 	}
11938c8a9a5SSteve French 
12038c8a9a5SSteve French 	return 0;
12138c8a9a5SSteve French }
12238c8a9a5SSteve French 
share_config_request(struct unicode_map * um,const char * name)12338c8a9a5SSteve French static struct ksmbd_share_config *share_config_request(struct unicode_map *um,
12438c8a9a5SSteve French 						       const char *name)
12538c8a9a5SSteve French {
12638c8a9a5SSteve French 	struct ksmbd_share_config_response *resp;
12738c8a9a5SSteve French 	struct ksmbd_share_config *share = NULL;
12838c8a9a5SSteve French 	struct ksmbd_share_config *lookup;
12938c8a9a5SSteve French 	int ret;
13038c8a9a5SSteve French 
13138c8a9a5SSteve French 	resp = ksmbd_ipc_share_config_request(name);
13238c8a9a5SSteve French 	if (!resp)
13338c8a9a5SSteve French 		return NULL;
13438c8a9a5SSteve French 
13538c8a9a5SSteve French 	if (resp->flags == KSMBD_SHARE_FLAG_INVALID)
13638c8a9a5SSteve French 		goto out;
13738c8a9a5SSteve French 
13838c8a9a5SSteve French 	if (*resp->share_name) {
13938c8a9a5SSteve French 		char *cf_resp_name;
14038c8a9a5SSteve French 		bool equal;
14138c8a9a5SSteve French 
14238c8a9a5SSteve French 		cf_resp_name = ksmbd_casefold_sharename(um, resp->share_name);
14338c8a9a5SSteve French 		if (IS_ERR(cf_resp_name))
14438c8a9a5SSteve French 			goto out;
14538c8a9a5SSteve French 		equal = !strcmp(cf_resp_name, name);
14638c8a9a5SSteve French 		kfree(cf_resp_name);
14738c8a9a5SSteve French 		if (!equal)
14838c8a9a5SSteve French 			goto out;
14938c8a9a5SSteve French 	}
15038c8a9a5SSteve French 
15138c8a9a5SSteve French 	share = kzalloc(sizeof(struct ksmbd_share_config), GFP_KERNEL);
15238c8a9a5SSteve French 	if (!share)
15338c8a9a5SSteve French 		goto out;
15438c8a9a5SSteve French 
15538c8a9a5SSteve French 	share->flags = resp->flags;
15638c8a9a5SSteve French 	atomic_set(&share->refcount, 1);
15738c8a9a5SSteve French 	INIT_LIST_HEAD(&share->veto_list);
15838c8a9a5SSteve French 	share->name = kstrdup(name, GFP_KERNEL);
15938c8a9a5SSteve French 
16038c8a9a5SSteve French 	if (!test_share_config_flag(share, KSMBD_SHARE_FLAG_PIPE)) {
161a637fabaSNamjae Jeon 		int path_len = PATH_MAX;
162a637fabaSNamjae Jeon 
163a637fabaSNamjae Jeon 		if (resp->payload_sz)
164a637fabaSNamjae Jeon 			path_len = resp->payload_sz - resp->veto_list_sz;
165a637fabaSNamjae Jeon 
166a637fabaSNamjae Jeon 		share->path = kstrndup(ksmbd_share_config_path(resp), path_len,
16738c8a9a5SSteve French 				      GFP_KERNEL);
1685698ba69SNandor Kracser 		if (share->path) {
16938c8a9a5SSteve French 			share->path_sz = strlen(share->path);
1705698ba69SNandor Kracser 			while (share->path_sz > 1 &&
1715698ba69SNandor Kracser 			       share->path[share->path_sz - 1] == '/')
1725698ba69SNandor Kracser 				share->path[--share->path_sz] = '\0';
1735698ba69SNandor Kracser 		}
17438c8a9a5SSteve French 		share->create_mask = resp->create_mask;
17538c8a9a5SSteve French 		share->directory_mask = resp->directory_mask;
17638c8a9a5SSteve French 		share->force_create_mode = resp->force_create_mode;
17738c8a9a5SSteve French 		share->force_directory_mode = resp->force_directory_mode;
17838c8a9a5SSteve French 		share->force_uid = resp->force_uid;
17938c8a9a5SSteve French 		share->force_gid = resp->force_gid;
18038c8a9a5SSteve French 		ret = parse_veto_list(share,
18138c8a9a5SSteve French 				      KSMBD_SHARE_CONFIG_VETO_LIST(resp),
18238c8a9a5SSteve French 				      resp->veto_list_sz);
18338c8a9a5SSteve French 		if (!ret && share->path) {
18438c8a9a5SSteve French 			ret = kern_path(share->path, 0, &share->vfs_path);
18538c8a9a5SSteve French 			if (ret) {
18638c8a9a5SSteve French 				ksmbd_debug(SMB, "failed to access '%s'\n",
18738c8a9a5SSteve French 					    share->path);
18838c8a9a5SSteve French 				/* Avoid put_path() */
18938c8a9a5SSteve French 				kfree(share->path);
19038c8a9a5SSteve French 				share->path = NULL;
19138c8a9a5SSteve French 			}
19238c8a9a5SSteve French 		}
19338c8a9a5SSteve French 		if (ret || !share->name) {
19438c8a9a5SSteve French 			kill_share(share);
19538c8a9a5SSteve French 			share = NULL;
19638c8a9a5SSteve French 			goto out;
19738c8a9a5SSteve French 		}
19838c8a9a5SSteve French 	}
19938c8a9a5SSteve French 
20038c8a9a5SSteve French 	down_write(&shares_table_lock);
20138c8a9a5SSteve French 	lookup = __share_lookup(name);
20238c8a9a5SSteve French 	if (lookup)
20338c8a9a5SSteve French 		lookup = __get_share_config(lookup);
20438c8a9a5SSteve French 	if (!lookup) {
20538c8a9a5SSteve French 		hash_add(shares_table, &share->hlist, share_name_hash(name));
20638c8a9a5SSteve French 	} else {
20738c8a9a5SSteve French 		kill_share(share);
20838c8a9a5SSteve French 		share = lookup;
20938c8a9a5SSteve French 	}
21038c8a9a5SSteve French 	up_write(&shares_table_lock);
21138c8a9a5SSteve French 
21238c8a9a5SSteve French out:
21338c8a9a5SSteve French 	kvfree(resp);
21438c8a9a5SSteve French 	return share;
21538c8a9a5SSteve French }
21638c8a9a5SSteve French 
ksmbd_share_config_get(struct unicode_map * um,const char * name)21738c8a9a5SSteve French struct ksmbd_share_config *ksmbd_share_config_get(struct unicode_map *um,
21838c8a9a5SSteve French 						  const char *name)
21938c8a9a5SSteve French {
22038c8a9a5SSteve French 	struct ksmbd_share_config *share;
22138c8a9a5SSteve French 
22238c8a9a5SSteve French 	down_read(&shares_table_lock);
22338c8a9a5SSteve French 	share = __share_lookup(name);
22438c8a9a5SSteve French 	if (share)
22538c8a9a5SSteve French 		share = __get_share_config(share);
22638c8a9a5SSteve French 	up_read(&shares_table_lock);
22738c8a9a5SSteve French 
22838c8a9a5SSteve French 	if (share)
22938c8a9a5SSteve French 		return share;
23038c8a9a5SSteve French 	return share_config_request(um, name);
23138c8a9a5SSteve French }
23238c8a9a5SSteve French 
ksmbd_share_veto_filename(struct ksmbd_share_config * share,const char * filename)23338c8a9a5SSteve French bool ksmbd_share_veto_filename(struct ksmbd_share_config *share,
23438c8a9a5SSteve French 			       const char *filename)
23538c8a9a5SSteve French {
23638c8a9a5SSteve French 	struct ksmbd_veto_pattern *p;
23738c8a9a5SSteve French 
23838c8a9a5SSteve French 	list_for_each_entry(p, &share->veto_list, list) {
23938c8a9a5SSteve French 		if (match_wildcard(p->pattern, filename))
24038c8a9a5SSteve French 			return true;
24138c8a9a5SSteve French 	}
24238c8a9a5SSteve French 	return false;
24338c8a9a5SSteve French }
244