/* Unix SMB/CIFS implementation. SMB2 client session handling Copyright (C) Andrew Tridgell 2005 This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. */ #include "includes.h" #include "system/network.h" #include <tevent.h> #include "lib/util/tevent_ntstatus.h" #include "libcli/raw/libcliraw.h" #include "libcli/smb2/smb2.h" #include "libcli/smb2/smb2_calls.h" #include "auth/gensec/gensec.h" #include "../libcli/smb/smbXcli_base.h" #include "../source3/libsmb/smb2cli.h" /** initialise a smb2_session structure */ struct smb2_session *smb2_session_init(struct smb2_transport *transport, struct gensec_settings *settings, TALLOC_CTX *parent_ctx, bool primary) { struct smb2_session *session; NTSTATUS status; session = talloc_zero(parent_ctx, struct smb2_session); if (!session) { return NULL; } if (primary) { session->transport = talloc_steal(session, transport); } else { session->transport = talloc_reference(session, transport); } session->pid = getpid(); session->smbXcli = smbXcli_session_create(session, transport->conn); if (session->smbXcli == NULL) { talloc_free(session); return NULL; } /* prepare a gensec context for later use */ status = gensec_client_start(session, &session->gensec, settings); if (!NT_STATUS_IS_OK(status)) { talloc_free(session); return NULL; } gensec_want_feature(session->gensec, GENSEC_FEATURE_SESSION_KEY); return session; } struct smb2_session_setup_spnego_state { struct tevent_context *ev; struct smb2_session *session; struct cli_credentials *credentials; NTSTATUS gensec_status; DATA_BLOB in_secblob; DATA_BLOB out_secblob; }; static void smb2_session_setup_spnego_done(struct tevent_req *subreq); /* a composite function that does a full SPNEGO session setup */ struct tevent_req *smb2_session_setup_spnego_send(TALLOC_CTX *mem_ctx, struct tevent_context *ev, struct smb2_session *session, struct cli_credentials *credentials) { struct tevent_req *req; struct smb2_session_setup_spnego_state *state; const char *chosen_oid; struct tevent_req *subreq; NTSTATUS status; const DATA_BLOB *server_gss_blob; DATA_BLOB negprot_secblob = data_blob_null; uint32_t timeout_msec; timeout_msec = session->transport->options.request_timeout * 1000; req = tevent_req_create(mem_ctx, &state, struct smb2_session_setup_spnego_state); if (req == NULL) { return NULL; } state->ev = ev; state->session = session; state->credentials = credentials; server_gss_blob = smbXcli_conn_server_gss_blob(session->transport->conn); if (server_gss_blob) { negprot_secblob = *server_gss_blob; } status = gensec_set_credentials(session->gensec, credentials); if (tevent_req_nterror(req, status)) { return tevent_req_post(req, ev); } status = gensec_set_target_hostname(session->gensec, smbXcli_conn_remote_name(session->transport->conn)); if (tevent_req_nterror(req, status)) { return tevent_req_post(req, ev); } status = gensec_set_target_service(session->gensec, "cifs"); if (tevent_req_nterror(req, status)) { return tevent_req_post(req, ev); } if (negprot_secblob.length > 0) { chosen_oid = GENSEC_OID_SPNEGO; } else { chosen_oid = GENSEC_OID_NTLMSSP; } status = gensec_start_mech_by_oid(session->gensec, chosen_oid); if (tevent_req_nterror(req, status)) { return tevent_req_post(req, ev); } status = gensec_update(session->gensec, state, state->ev, negprot_secblob, &state->in_secblob); if (!NT_STATUS_EQUAL(status, NT_STATUS_MORE_PROCESSING_REQUIRED)) { tevent_req_nterror(req, status); return tevent_req_post(req, ev); } state->gensec_status = status; subreq = smb2cli_session_setup_send(state, state->ev, session->transport->conn, timeout_msec, session->smbXcli, 0, /* in_flags */ 0, /* in_capabilities */ 0, /* in_channel */ NULL, /* in_previous_session */ &state->in_secblob); if (tevent_req_nomem(subreq, req)) { return tevent_req_post(req, ev); } tevent_req_set_callback(subreq, smb2_session_setup_spnego_done, req); return req; } /* handle continuations of the spnego session setup */ static void smb2_session_setup_spnego_done(struct tevent_req *subreq) { struct tevent_req *req = tevent_req_callback_data(subreq, struct tevent_req); struct smb2_session_setup_spnego_state *state = tevent_req_data(req, struct smb2_session_setup_spnego_state); struct smb2_session *session = state->session; NTSTATUS peer_status; NTSTATUS status; struct iovec *recv_iov; uint32_t timeout_msec; timeout_msec = session->transport->options.request_timeout * 1000; status = smb2cli_session_setup_recv(subreq, state, &recv_iov, &state->out_secblob); if (!NT_STATUS_IS_OK(status) && !NT_STATUS_EQUAL(status, NT_STATUS_MORE_PROCESSING_REQUIRED)) { tevent_req_nterror(req, status); return; } peer_status = status; if (NT_STATUS_EQUAL(state->gensec_status, NT_STATUS_MORE_PROCESSING_REQUIRED)) { status = gensec_update(session->gensec, state, state->ev, state->out_secblob, &state->in_secblob); state->gensec_status = status; } if (!NT_STATUS_IS_OK(status) && !NT_STATUS_EQUAL(status, NT_STATUS_MORE_PROCESSING_REQUIRED)) { tevent_req_nterror(req, status); return; } if (NT_STATUS_IS_OK(peer_status) && NT_STATUS_IS_OK(state->gensec_status)) { status = gensec_session_key(session->gensec, session, &session->session_key); if (tevent_req_nterror(req, status)) { return; } status = smb2cli_session_update_session_key(session->smbXcli, session->session_key, recv_iov); if (tevent_req_nterror(req, status)) { return; } tevent_req_done(req); return; } subreq = smb2cli_session_setup_send(state, state->ev, session->transport->conn, timeout_msec, session->smbXcli, 0, /* in_flags */ 0, /* in_capabilities */ 0, /* in_channel */ NULL, /* in_previous_session */ &state->in_secblob); if (tevent_req_nomem(subreq, req)) { return; } tevent_req_set_callback(subreq, smb2_session_setup_spnego_done, req); } /* receive a composite session setup reply */ NTSTATUS smb2_session_setup_spnego_recv(struct tevent_req *req) { return tevent_req_simple_recv_ntstatus(req); } /* sync version of smb2_session_setup_spnego */ NTSTATUS smb2_session_setup_spnego(struct smb2_session *session, struct cli_credentials *credentials) { struct tevent_req *subreq; NTSTATUS status; bool ok; TALLOC_CTX *frame = talloc_stackframe(); struct tevent_context *ev = session->transport->ev; if (frame == NULL) { return NT_STATUS_NO_MEMORY; } subreq = smb2_session_setup_spnego_send(frame, ev, session, credentials); if (subreq == NULL) { TALLOC_FREE(frame); return NT_STATUS_NO_MEMORY; } ok = tevent_req_poll(subreq, ev); if (!ok) { status = map_nt_error_from_unix_common(errno); TALLOC_FREE(frame); return status; } status = smb2_session_setup_spnego_recv(subreq); TALLOC_FREE(subreq); if (!NT_STATUS_IS_OK(status)) { TALLOC_FREE(frame); return status; } TALLOC_FREE(frame); return NT_STATUS_OK; }