| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * Copyright (C) 2018 Samsung Electronics Co., Ltd. |
| */ |
| |
| #include <linux/list.h> |
| #include <linux/slab.h> |
| #include <linux/rwsem.h> |
| #include <linux/xarray.h> |
| |
| #include "ksmbd_ida.h" |
| #include "user_session.h" |
| #include "user_config.h" |
| #include "tree_connect.h" |
| #include "share_config.h" |
| #include "../transport_ipc.h" |
| #include "../connection.h" |
| #include "../vfs_cache.h" |
| #include "../misc.h" |
| #include "../stats.h" |
| |
| static DEFINE_IDA(session_ida); |
| |
| #define SESSION_HASH_BITS 12 |
| static DEFINE_HASHTABLE(sessions_table, SESSION_HASH_BITS); |
| static DECLARE_RWSEM(sessions_table_lock); |
| |
| struct ksmbd_session_rpc { |
| int id; |
| unsigned int method; |
| }; |
| |
| #ifdef CONFIG_PROC_FS |
| |
| static const struct ksmbd_const_name ksmbd_sess_cap_const_names[] = { |
| {SMB2_GLOBAL_CAP_DFS, "dfs"}, |
| {SMB2_GLOBAL_CAP_LEASING, "lease"}, |
| {SMB2_GLOBAL_CAP_LARGE_MTU, "large-mtu"}, |
| {SMB2_GLOBAL_CAP_MULTI_CHANNEL, "multi-channel"}, |
| {SMB2_GLOBAL_CAP_PERSISTENT_HANDLES, "persistent-handles"}, |
| {SMB2_GLOBAL_CAP_DIRECTORY_LEASING, "dir-lease"}, |
| {SMB2_GLOBAL_CAP_ENCRYPTION, "encryption"} |
| }; |
| |
| static const struct ksmbd_const_name ksmbd_cipher_const_names[] = { |
| {le16_to_cpu(SMB2_ENCRYPTION_AES128_CCM), "aes128-ccm"}, |
| {le16_to_cpu(SMB2_ENCRYPTION_AES128_GCM), "aes128-gcm"}, |
| {le16_to_cpu(SMB2_ENCRYPTION_AES256_CCM), "aes256-ccm"}, |
| {le16_to_cpu(SMB2_ENCRYPTION_AES256_GCM), "aes256-gcm"}, |
| }; |
| |
| static const struct ksmbd_const_name ksmbd_signing_const_names[] = { |
| {SIGNING_ALG_HMAC_SHA256, "hmac-sha256"}, |
| {SIGNING_ALG_AES_CMAC, "aes-cmac"}, |
| {SIGNING_ALG_AES_GMAC, "aes-gmac"}, |
| }; |
| |
| static const char *session_state_string(struct ksmbd_session *session) |
| { |
| switch (session->state) { |
| case SMB2_SESSION_VALID: |
| return "valid"; |
| case SMB2_SESSION_IN_PROGRESS: |
| return "progress"; |
| case SMB2_SESSION_EXPIRED: |
| return "expired"; |
| default: |
| return ""; |
| } |
| } |
| |
| static const char *session_user_name(struct ksmbd_session *session) |
| { |
| if (user_guest(session->user)) |
| return "(Guest)"; |
| else if (ksmbd_anonymous_user(session->user)) |
| return "(Anonymous)"; |
| return session->user->name; |
| } |
| |
| static int show_proc_session(struct seq_file *m, void *v) |
| { |
| struct ksmbd_session *sess; |
| struct ksmbd_tree_connect *tree_conn; |
| struct ksmbd_share_config *share_conf; |
| struct channel *chan; |
| unsigned long id; |
| int i = 0; |
| |
| sess = (struct ksmbd_session *)m->private; |
| ksmbd_user_session_get(sess); |
| |
| i = 0; |
| down_read(&sess->chann_lock); |
| xa_for_each(&sess->ksmbd_chann_list, id, chan) { |
| #if IS_ENABLED(CONFIG_IPV6) |
| if (chan->conn->inet_addr) |
| seq_printf(m, "%-20s\t%pI4\n", "client", |
| &chan->conn->inet_addr); |
| else |
| seq_printf(m, "%-20s\t%pI6c\n", "client", |
| &chan->conn->inet6_addr); |
| #else |
| seq_printf(m, "%-20s\t%pI4\n", "client", |
| &chan->conn->inet_addr); |
| #endif |
| seq_printf(m, "%-20s\t%s\n", "user", session_user_name(sess)); |
| seq_printf(m, "%-20s\t%llu\n", "id", sess->id); |
| seq_printf(m, "%-20s\t%s\n", "state", |
| session_state_string(sess)); |
| |
| seq_printf(m, "%-20s\t", "capabilities"); |
| ksmbd_proc_show_flag_names(m, |
| ksmbd_sess_cap_const_names, |
| ARRAY_SIZE(ksmbd_sess_cap_const_names), |
| chan->conn->vals->req_capabilities); |
| |
| if (sess->sign) { |
| seq_printf(m, "%-20s\t", "signing"); |
| ksmbd_proc_show_const_name(m, "%s\t", |
| ksmbd_signing_const_names, |
| ARRAY_SIZE(ksmbd_signing_const_names), |
| le16_to_cpu(chan->conn->signing_algorithm)); |
| } else if (sess->enc) { |
| seq_printf(m, "%-20s\t", "encryption"); |
| ksmbd_proc_show_const_name(m, "%s\t", |
| ksmbd_cipher_const_names, |
| ARRAY_SIZE(ksmbd_cipher_const_names), |
| le16_to_cpu(chan->conn->cipher_type)); |
| } |
| i++; |
| } |
| up_read(&sess->chann_lock); |
| |
| seq_printf(m, "%-20s\t%d\n", "channels", i); |
| |
| i = 0; |
| down_read(&sess->tree_conns_lock); |
| xa_for_each(&sess->tree_conns, id, tree_conn) { |
| share_conf = tree_conn->share_conf; |
| seq_printf(m, "%-20s\t%s\t%8d", "share", |
| share_conf->name, tree_conn->id); |
| if (test_share_config_flag(share_conf, KSMBD_SHARE_FLAG_PIPE)) |
| seq_printf(m, " %s ", "pipe"); |
| else |
| seq_printf(m, " %s ", "disk"); |
| seq_putc(m, '\n'); |
| } |
| up_read(&sess->tree_conns_lock); |
| |
| ksmbd_user_session_put(sess); |
| return 0; |
| } |
| |
| void ksmbd_proc_show_flag_names(struct seq_file *m, |
| const struct ksmbd_const_name *table, |
| int count, |
| unsigned int flags) |
| { |
| int i; |
| |
| for (i = 0; i < count; i++) { |
| if (table[i].const_value & flags) |
| seq_printf(m, "0x%08x\t", table[i].const_value); |
| } |
| seq_putc(m, '\n'); |
| } |
| |
| void ksmbd_proc_show_const_name(struct seq_file *m, |
| const char *format, |
| const struct ksmbd_const_name *table, |
| int count, |
| unsigned int const_value) |
| { |
| int i; |
| |
| for (i = 0; i < count; i++) { |
| if (table[i].const_value & const_value) |
| seq_printf(m, format, table[i].name); |
| } |
| seq_putc(m, '\n'); |
| } |
| |
| static int create_proc_session(struct ksmbd_session *sess) |
| { |
| char name[30]; |
| |
| snprintf(name, sizeof(name), "sessions/%llu", sess->id); |
| sess->proc_entry = ksmbd_proc_create(name, |
| show_proc_session, sess); |
| return 0; |
| } |
| |
| static void delete_proc_session(struct ksmbd_session *sess) |
| { |
| if (sess->proc_entry) |
| proc_remove(sess->proc_entry); |
| } |
| |
| static int show_proc_sessions(struct seq_file *m, void *v) |
| { |
| struct ksmbd_session *session; |
| struct channel *chan; |
| int i; |
| unsigned long id; |
| |
| seq_printf(m, "#%-40s %-15s %-10s %-10s\n", |
| "<client>", "<user>", "<sess_id>", "<state>"); |
| |
| down_read(&sessions_table_lock); |
| hash_for_each(sessions_table, i, session, hlist) { |
| down_read(&session->chann_lock); |
| xa_for_each(&session->ksmbd_chann_list, id, chan) { |
| down_read(&chan->conn->session_lock); |
| ksmbd_user_session_get(session); |
| |
| #if IS_ENABLED(CONFIG_IPV6) |
| if (!chan->conn->inet_addr) |
| seq_printf(m, " %-40pI6c", &chan->conn->inet6_addr); |
| else |
| #endif |
| seq_printf(m, " %-40pI4", &chan->conn->inet_addr); |
| seq_printf(m, " %-15s %-10llu %-10s\n", |
| session_user_name(session), |
| session->id, |
| session_state_string(session)); |
| |
| ksmbd_user_session_put(session); |
| up_read(&chan->conn->session_lock); |
| } |
| up_read(&session->chann_lock); |
| } |
| up_read(&sessions_table_lock); |
| return 0; |
| } |
| |
| int create_proc_sessions(void) |
| { |
| if (!ksmbd_proc_create("sessions/sessions", |
| show_proc_sessions, NULL)) |
| return -ENOMEM; |
| return 0; |
| } |
| #else |
| int create_proc_sessions(void) { return 0; } |
| static int create_proc_session(struct ksmbd_session *sess) { return 0; } |
| static void delete_proc_session(struct ksmbd_session *sess) {} |
| #endif |
| |
| static void free_channel_list(struct ksmbd_session *sess) |
| { |
| struct channel *chann; |
| unsigned long index; |
| |
| down_write(&sess->chann_lock); |
| xa_for_each(&sess->ksmbd_chann_list, index, chann) { |
| xa_erase(&sess->ksmbd_chann_list, index); |
| kfree(chann); |
| } |
| |
| xa_destroy(&sess->ksmbd_chann_list); |
| up_write(&sess->chann_lock); |
| } |
| |
| static void __session_rpc_close(struct ksmbd_session *sess, |
| struct ksmbd_session_rpc *entry) |
| { |
| struct ksmbd_rpc_command *resp; |
| |
| resp = ksmbd_rpc_close(sess, entry->id); |
| if (!resp) |
| pr_err("Unable to close RPC pipe %d\n", entry->id); |
| |
| kvfree(resp); |
| ksmbd_rpc_id_free(entry->id); |
| kfree(entry); |
| } |
| |
| static void ksmbd_session_rpc_clear_list(struct ksmbd_session *sess) |
| { |
| struct ksmbd_session_rpc *entry; |
| long index; |
| |
| down_write(&sess->rpc_lock); |
| xa_for_each(&sess->rpc_handle_list, index, entry) { |
| xa_erase(&sess->rpc_handle_list, index); |
| __session_rpc_close(sess, entry); |
| } |
| up_write(&sess->rpc_lock); |
| |
| xa_destroy(&sess->rpc_handle_list); |
| } |
| |
| static int __rpc_method(char *rpc_name) |
| { |
| if (!strcmp(rpc_name, "\\srvsvc") || !strcmp(rpc_name, "srvsvc")) |
| return KSMBD_RPC_SRVSVC_METHOD_INVOKE; |
| |
| if (!strcmp(rpc_name, "\\wkssvc") || !strcmp(rpc_name, "wkssvc")) |
| return KSMBD_RPC_WKSSVC_METHOD_INVOKE; |
| |
| if (!strcmp(rpc_name, "LANMAN") || !strcmp(rpc_name, "lanman")) |
| return KSMBD_RPC_RAP_METHOD; |
| |
| if (!strcmp(rpc_name, "\\samr") || !strcmp(rpc_name, "samr")) |
| return KSMBD_RPC_SAMR_METHOD_INVOKE; |
| |
| if (!strcmp(rpc_name, "\\lsarpc") || !strcmp(rpc_name, "lsarpc")) |
| return KSMBD_RPC_LSARPC_METHOD_INVOKE; |
| |
| pr_err("Unsupported RPC: %s\n", rpc_name); |
| return 0; |
| } |
| |
| int ksmbd_session_rpc_open(struct ksmbd_session *sess, char *rpc_name) |
| { |
| struct ksmbd_session_rpc *entry, *old; |
| struct ksmbd_rpc_command *resp; |
| int method, id; |
| |
| method = __rpc_method(rpc_name); |
| if (!method) |
| return -EINVAL; |
| |
| entry = kzalloc_obj(struct ksmbd_session_rpc, KSMBD_DEFAULT_GFP); |
| if (!entry) |
| return -ENOMEM; |
| |
| entry->method = method; |
| entry->id = id = ksmbd_ipc_id_alloc(); |
| if (id < 0) |
| goto free_entry; |
| |
| down_write(&sess->rpc_lock); |
| old = xa_store(&sess->rpc_handle_list, id, entry, KSMBD_DEFAULT_GFP); |
| if (xa_is_err(old)) { |
| up_write(&sess->rpc_lock); |
| goto free_id; |
| } |
| |
| resp = ksmbd_rpc_open(sess, id); |
| if (!resp) { |
| xa_erase(&sess->rpc_handle_list, entry->id); |
| up_write(&sess->rpc_lock); |
| goto free_id; |
| } |
| |
| up_write(&sess->rpc_lock); |
| kvfree(resp); |
| return id; |
| free_id: |
| ksmbd_rpc_id_free(entry->id); |
| free_entry: |
| kfree(entry); |
| return -EINVAL; |
| } |
| |
| void ksmbd_session_rpc_close(struct ksmbd_session *sess, int id) |
| { |
| struct ksmbd_session_rpc *entry; |
| |
| down_write(&sess->rpc_lock); |
| entry = xa_erase(&sess->rpc_handle_list, id); |
| if (entry) |
| __session_rpc_close(sess, entry); |
| up_write(&sess->rpc_lock); |
| } |
| |
| int ksmbd_session_rpc_method(struct ksmbd_session *sess, int id) |
| { |
| struct ksmbd_session_rpc *entry; |
| |
| lockdep_assert_held(&sess->rpc_lock); |
| entry = xa_load(&sess->rpc_handle_list, id); |
| |
| return entry ? entry->method : 0; |
| } |
| |
| void ksmbd_session_destroy(struct ksmbd_session *sess) |
| { |
| if (!sess) |
| return; |
| |
| delete_proc_session(sess); |
| |
| if (sess->user) |
| ksmbd_free_user(sess->user); |
| |
| ksmbd_tree_conn_session_logoff(sess); |
| ksmbd_destroy_file_table(&sess->file_table); |
| ksmbd_launch_ksmbd_durable_scavenger(); |
| ksmbd_session_rpc_clear_list(sess); |
| free_channel_list(sess); |
| kfree(sess->Preauth_HashValue); |
| ksmbd_release_id(&session_ida, sess->id); |
| kfree(sess); |
| } |
| |
| struct ksmbd_session *__session_lookup(unsigned long long id) |
| { |
| struct ksmbd_session *sess; |
| |
| hash_for_each_possible(sessions_table, sess, hlist, id) { |
| if (id == sess->id) { |
| sess->last_active = jiffies; |
| return sess; |
| } |
| } |
| return NULL; |
| } |
| |
| static void ksmbd_expire_session(struct ksmbd_conn *conn) |
| { |
| unsigned long id; |
| struct ksmbd_session *sess; |
| |
| down_write(&sessions_table_lock); |
| down_write(&conn->session_lock); |
| xa_for_each(&conn->sessions, id, sess) { |
| if (atomic_read(&sess->refcnt) <= 1 && |
| (sess->state != SMB2_SESSION_VALID || |
| time_after(jiffies, |
| sess->last_active + SMB2_SESSION_TIMEOUT))) { |
| xa_erase(&conn->sessions, sess->id); |
| hash_del(&sess->hlist); |
| ksmbd_session_destroy(sess); |
| continue; |
| } |
| } |
| up_write(&conn->session_lock); |
| up_write(&sessions_table_lock); |
| } |
| |
| int ksmbd_session_register(struct ksmbd_conn *conn, |
| struct ksmbd_session *sess) |
| { |
| sess->dialect = conn->dialect; |
| memcpy(sess->ClientGUID, conn->ClientGUID, SMB2_CLIENT_GUID_SIZE); |
| ksmbd_expire_session(conn); |
| return xa_err(xa_store(&conn->sessions, sess->id, sess, KSMBD_DEFAULT_GFP)); |
| } |
| |
| static int ksmbd_chann_del(struct ksmbd_conn *conn, struct ksmbd_session *sess) |
| { |
| struct channel *chann; |
| |
| down_write(&sess->chann_lock); |
| chann = xa_erase(&sess->ksmbd_chann_list, (long)conn); |
| up_write(&sess->chann_lock); |
| if (!chann) |
| return -ENOENT; |
| |
| kfree(chann); |
| return 0; |
| } |
| |
| void ksmbd_sessions_deregister(struct ksmbd_conn *conn) |
| { |
| struct ksmbd_session *sess; |
| unsigned long id; |
| |
| down_write(&sessions_table_lock); |
| if (conn->binding) { |
| int bkt; |
| struct hlist_node *tmp; |
| |
| hash_for_each_safe(sessions_table, bkt, tmp, sess, hlist) { |
| if (!ksmbd_chann_del(conn, sess) && |
| xa_empty(&sess->ksmbd_chann_list)) { |
| hash_del(&sess->hlist); |
| down_write(&conn->session_lock); |
| xa_erase(&conn->sessions, sess->id); |
| up_write(&conn->session_lock); |
| if (atomic_dec_and_test(&sess->refcnt)) |
| ksmbd_session_destroy(sess); |
| } |
| } |
| } |
| |
| down_write(&conn->session_lock); |
| xa_for_each(&conn->sessions, id, sess) { |
| unsigned long chann_id; |
| struct channel *chann; |
| |
| xa_for_each(&sess->ksmbd_chann_list, chann_id, chann) { |
| if (chann->conn != conn) |
| ksmbd_conn_set_exiting(chann->conn); |
| } |
| |
| ksmbd_chann_del(conn, sess); |
| if (xa_empty(&sess->ksmbd_chann_list)) { |
| xa_erase(&conn->sessions, sess->id); |
| hash_del(&sess->hlist); |
| if (atomic_dec_and_test(&sess->refcnt)) |
| ksmbd_session_destroy(sess); |
| } |
| } |
| up_write(&conn->session_lock); |
| up_write(&sessions_table_lock); |
| } |
| |
| bool is_ksmbd_session_in_connection(struct ksmbd_conn *conn, |
| unsigned long long id) |
| { |
| struct ksmbd_session *sess; |
| |
| down_read(&conn->session_lock); |
| sess = xa_load(&conn->sessions, id); |
| if (sess) { |
| up_read(&conn->session_lock); |
| return true; |
| } |
| up_read(&conn->session_lock); |
| |
| return false; |
| } |
| |
| struct ksmbd_session *ksmbd_session_lookup(struct ksmbd_conn *conn, |
| unsigned long long id) |
| { |
| struct ksmbd_session *sess; |
| |
| down_read(&conn->session_lock); |
| sess = xa_load(&conn->sessions, id); |
| if (sess) { |
| sess->last_active = jiffies; |
| ksmbd_user_session_get(sess); |
| } |
| up_read(&conn->session_lock); |
| return sess; |
| } |
| |
| struct ksmbd_session *ksmbd_session_lookup_slowpath(unsigned long long id) |
| { |
| struct ksmbd_session *sess; |
| |
| down_read(&sessions_table_lock); |
| sess = __session_lookup(id); |
| if (sess) |
| ksmbd_user_session_get(sess); |
| up_read(&sessions_table_lock); |
| |
| return sess; |
| } |
| |
| struct ksmbd_session *ksmbd_session_lookup_all(struct ksmbd_conn *conn, |
| unsigned long long id) |
| { |
| struct ksmbd_session *sess; |
| |
| sess = ksmbd_session_lookup(conn, id); |
| if (!sess && conn->binding) |
| sess = ksmbd_session_lookup_slowpath(id); |
| if (sess && sess->state != SMB2_SESSION_VALID) { |
| ksmbd_user_session_put(sess); |
| sess = NULL; |
| } |
| return sess; |
| } |
| |
| void ksmbd_user_session_get(struct ksmbd_session *sess) |
| { |
| atomic_inc(&sess->refcnt); |
| } |
| |
| void ksmbd_user_session_put(struct ksmbd_session *sess) |
| { |
| if (!sess) |
| return; |
| |
| if (atomic_read(&sess->refcnt) <= 0) |
| WARN_ON(1); |
| else if (atomic_dec_and_test(&sess->refcnt)) |
| ksmbd_session_destroy(sess); |
| } |
| |
| struct preauth_session *ksmbd_preauth_session_alloc(struct ksmbd_conn *conn, |
| u64 sess_id) |
| { |
| struct preauth_session *sess; |
| |
| sess = kmalloc_obj(struct preauth_session, KSMBD_DEFAULT_GFP); |
| if (!sess) |
| return NULL; |
| |
| sess->id = sess_id; |
| memcpy(sess->Preauth_HashValue, conn->preauth_info->Preauth_HashValue, |
| PREAUTH_HASHVALUE_SIZE); |
| list_add(&sess->preauth_entry, &conn->preauth_sess_table); |
| |
| return sess; |
| } |
| |
| void destroy_previous_session(struct ksmbd_conn *conn, |
| struct ksmbd_user *user, u64 id) |
| { |
| struct ksmbd_session *prev_sess; |
| struct ksmbd_user *prev_user; |
| int err; |
| |
| down_write(&sessions_table_lock); |
| down_write(&conn->session_lock); |
| prev_sess = __session_lookup(id); |
| if (!prev_sess || prev_sess->state == SMB2_SESSION_EXPIRED) |
| goto out; |
| |
| prev_user = prev_sess->user; |
| if (!prev_user || |
| strcmp(user->name, prev_user->name) || |
| user->passkey_sz != prev_user->passkey_sz || |
| memcmp(user->passkey, prev_user->passkey, user->passkey_sz)) |
| goto out; |
| |
| ksmbd_all_conn_set_status(id, KSMBD_SESS_NEED_RECONNECT); |
| err = ksmbd_conn_wait_idle_sess_id(conn, id); |
| if (err) { |
| ksmbd_all_conn_set_status(id, KSMBD_SESS_NEED_SETUP); |
| goto out; |
| } |
| |
| ksmbd_destroy_file_table(&prev_sess->file_table); |
| prev_sess->state = SMB2_SESSION_EXPIRED; |
| ksmbd_all_conn_set_status(id, KSMBD_SESS_NEED_SETUP); |
| ksmbd_launch_ksmbd_durable_scavenger(); |
| out: |
| up_write(&conn->session_lock); |
| up_write(&sessions_table_lock); |
| } |
| |
| static bool ksmbd_preauth_session_id_match(struct preauth_session *sess, |
| unsigned long long id) |
| { |
| return sess->id == id; |
| } |
| |
| struct preauth_session *ksmbd_preauth_session_lookup(struct ksmbd_conn *conn, |
| unsigned long long id) |
| { |
| struct preauth_session *sess = NULL; |
| |
| list_for_each_entry(sess, &conn->preauth_sess_table, preauth_entry) { |
| if (ksmbd_preauth_session_id_match(sess, id)) |
| return sess; |
| } |
| return NULL; |
| } |
| |
| static int __init_smb2_session(struct ksmbd_session *sess) |
| { |
| int id = ksmbd_acquire_smb2_uid(&session_ida); |
| |
| if (id < 0) |
| return -EINVAL; |
| sess->id = id; |
| return 0; |
| } |
| |
| static struct ksmbd_session *__session_create(int protocol) |
| { |
| struct ksmbd_session *sess; |
| int ret; |
| |
| if (protocol != CIFDS_SESSION_FLAG_SMB2) |
| return NULL; |
| |
| sess = kzalloc_obj(struct ksmbd_session, KSMBD_DEFAULT_GFP); |
| if (!sess) |
| return NULL; |
| |
| if (ksmbd_init_file_table(&sess->file_table)) |
| goto error; |
| |
| sess->last_active = jiffies; |
| sess->state = SMB2_SESSION_IN_PROGRESS; |
| set_session_flag(sess, protocol); |
| xa_init(&sess->tree_conns); |
| xa_init(&sess->ksmbd_chann_list); |
| xa_init(&sess->rpc_handle_list); |
| sess->sequence_number = 1; |
| atomic_set(&sess->refcnt, 2); |
| init_rwsem(&sess->tree_conns_lock); |
| init_rwsem(&sess->rpc_lock); |
| init_rwsem(&sess->chann_lock); |
| |
| ret = __init_smb2_session(sess); |
| if (ret) |
| goto error; |
| |
| ida_init(&sess->tree_conn_ida); |
| |
| down_write(&sessions_table_lock); |
| hash_add(sessions_table, &sess->hlist, sess->id); |
| up_write(&sessions_table_lock); |
| |
| create_proc_session(sess); |
| ksmbd_counter_inc(KSMBD_COUNTER_SESSIONS); |
| return sess; |
| |
| error: |
| ksmbd_session_destroy(sess); |
| return NULL; |
| } |
| |
| struct ksmbd_session *ksmbd_smb2_session_create(void) |
| { |
| return __session_create(CIFDS_SESSION_FLAG_SMB2); |
| } |
| |
| int ksmbd_acquire_tree_conn_id(struct ksmbd_session *sess) |
| { |
| int id = -EINVAL; |
| |
| if (test_session_flag(sess, CIFDS_SESSION_FLAG_SMB2)) |
| id = ksmbd_acquire_smb2_tid(&sess->tree_conn_ida); |
| |
| return id; |
| } |
| |
| void ksmbd_release_tree_conn_id(struct ksmbd_session *sess, int id) |
| { |
| if (id >= 0) |
| ksmbd_release_id(&sess->tree_conn_ida, id); |
| } |