Line data Source code
1 : /*
2 : Unix SMB/Netbios implementation.
3 : Version 3.0
4 : handle NLTMSSP, client server side parsing
5 :
6 : Copyright (C) Andrew Tridgell 2001
7 : Copyright (C) Andrew Bartlett <abartlet@samba.org> 2001-2005
8 : Copyright (C) Stefan Metzmacher 2005
9 :
10 : This program is free software; you can redistribute it and/or modify
11 : it under the terms of the GNU General Public License as published by
12 : the Free Software Foundation; either version 3 of the License, or
13 : (at your option) any later version.
14 :
15 : This program is distributed in the hope that it will be useful,
16 : but WITHOUT ANY WARRANTY; without even the implied warranty of
17 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 : GNU General Public License for more details.
19 :
20 : You should have received a copy of the GNU General Public License
21 : along with this program. If not, see <http://www.gnu.org/licenses/>.
22 : */
23 :
24 : struct auth_session_info;
25 :
26 : #include "includes.h"
27 : #include "auth/ntlmssp/ntlmssp.h"
28 : #include "../libcli/auth/libcli_auth.h"
29 : #include "auth/credentials/credentials.h"
30 : #include "auth/gensec/gensec.h"
31 : #include "auth/gensec/gensec_internal.h"
32 : #include "param/param.h"
33 : #include "auth/ntlmssp/ntlmssp_private.h"
34 : #include "../librpc/gen_ndr/ndr_ntlmssp.h"
35 : #include "../auth/ntlmssp/ntlmssp_ndr.h"
36 : #include "../nsswitch/libwbclient/wbclient.h"
37 :
38 : #include "lib/crypto/gnutls_helpers.h"
39 : #include <gnutls/gnutls.h>
40 : #include <gnutls/crypto.h>
41 :
42 : #undef DBGC_CLASS
43 : #define DBGC_CLASS DBGC_AUTH
44 :
45 : /*********************************************************************
46 : Client side NTLMSSP
47 : *********************************************************************/
48 :
49 : /**
50 : * Next state function for the Initial packet
51 : *
52 : * @param ntlmssp_state NTLMSSP State
53 : * @param out_mem_ctx The DATA_BLOB *out will be allocated on this context
54 : * @param in A NULL data blob (input ignored)
55 : * @param out The initial negotiate request to the server, as an talloc()ed DATA_BLOB, on out_mem_ctx
56 : * @return Errors or NT_STATUS_OK.
57 : */
58 :
59 10190 : NTSTATUS ntlmssp_client_initial(struct gensec_security *gensec_security,
60 : TALLOC_CTX *out_mem_ctx,
61 : DATA_BLOB in, DATA_BLOB *out)
62 : {
63 7314 : struct gensec_ntlmssp_context *gensec_ntlmssp =
64 10190 : talloc_get_type_abort(gensec_security->private_data,
65 : struct gensec_ntlmssp_context);
66 10190 : struct ntlmssp_state *ntlmssp_state = gensec_ntlmssp->ntlmssp_state;
67 : NTSTATUS status;
68 10190 : const DATA_BLOB version_blob = ntlmssp_version_blob();
69 :
70 : /* generate the ntlmssp negotiate packet */
71 10190 : status = msrpc_gen(out_mem_ctx,
72 : out, "CddAAb",
73 : "NTLMSSP",
74 : NTLMSSP_NEGOTIATE,
75 : ntlmssp_state->neg_flags,
76 : "", /* domain */
77 : "", /* workstation */
78 2876 : version_blob.data, version_blob.length);
79 10190 : if (!NT_STATUS_IS_OK(status)) {
80 0 : DEBUG(0, ("ntlmssp_client_initial: failed to generate "
81 : "ntlmssp negotiate packet\n"));
82 0 : return status;
83 : }
84 :
85 10190 : if (DEBUGLEVEL >= 10) {
86 0 : struct NEGOTIATE_MESSAGE *negotiate = talloc(
87 : ntlmssp_state, struct NEGOTIATE_MESSAGE);
88 0 : if (negotiate != NULL) {
89 0 : status = ntlmssp_pull_NEGOTIATE_MESSAGE(
90 : out, negotiate, negotiate);
91 0 : if (NT_STATUS_IS_OK(status)) {
92 0 : NDR_PRINT_DEBUG(NEGOTIATE_MESSAGE,
93 : negotiate);
94 : }
95 0 : TALLOC_FREE(negotiate);
96 : }
97 : }
98 :
99 10190 : ntlmssp_state->negotiate_blob = data_blob_dup_talloc(ntlmssp_state,
100 : *out);
101 10190 : if (ntlmssp_state->negotiate_blob.length != out->length) {
102 0 : return NT_STATUS_NO_MEMORY;
103 : }
104 :
105 10190 : ntlmssp_state->expected_state = NTLMSSP_CHALLENGE;
106 :
107 10190 : return NT_STATUS_MORE_PROCESSING_REQUIRED;
108 : }
109 :
110 12 : NTSTATUS gensec_ntlmssp_resume_ccache(struct gensec_security *gensec_security,
111 : TALLOC_CTX *out_mem_ctx,
112 : DATA_BLOB in, DATA_BLOB *out)
113 : {
114 7 : struct gensec_ntlmssp_context *gensec_ntlmssp =
115 12 : talloc_get_type_abort(gensec_security->private_data,
116 : struct gensec_ntlmssp_context);
117 12 : struct ntlmssp_state *ntlmssp_state = gensec_ntlmssp->ntlmssp_state;
118 12 : uint32_t neg_flags = 0;
119 : uint32_t ntlmssp_command;
120 : NTSTATUS status;
121 : bool ok;
122 :
123 12 : *out = data_blob_null;
124 :
125 12 : if (in.length == 0) {
126 : /*
127 : * This is compat code for older callers
128 : * which were missing the "initial_blob"/"negotiate_blob".
129 : *
130 : * That means we can't calculate the NTLMSSP_MIC
131 : * field correctly and need to force the
132 : * old_spnego behaviour.
133 : */
134 0 : DEBUG(10, ("%s: in.length==%u force_old_spnego!\n",
135 : __func__, (unsigned int)in.length));
136 0 : ntlmssp_state->force_old_spnego = true;
137 0 : ntlmssp_state->neg_flags |= ntlmssp_state->required_flags;
138 0 : ntlmssp_state->required_flags = 0;
139 0 : ntlmssp_state->expected_state = NTLMSSP_CHALLENGE;
140 0 : return NT_STATUS_MORE_PROCESSING_REQUIRED;
141 : }
142 :
143 : /* parse the NTLMSSP packet */
144 :
145 12 : if (in.length > UINT16_MAX) {
146 0 : DEBUG(1, ("%s: reject large request of length %u\n",
147 : __func__, (unsigned int)in.length));
148 0 : return NT_STATUS_INVALID_PARAMETER;
149 : }
150 :
151 12 : ok = msrpc_parse(ntlmssp_state, &in, "Cdd",
152 : "NTLMSSP",
153 : &ntlmssp_command,
154 : &neg_flags);
155 12 : if (!ok) {
156 0 : DEBUG(1, ("%s: failed to parse NTLMSSP Negotiate of length %u\n",
157 : __func__, (unsigned int)in.length));
158 0 : dump_data(2, in.data, in.length);
159 0 : return NT_STATUS_INVALID_PARAMETER;
160 : }
161 :
162 12 : if (ntlmssp_command != NTLMSSP_NEGOTIATE) {
163 0 : DEBUG(1, ("%s: no NTLMSSP Negotiate message (length %u)\n",
164 : __func__, (unsigned int)in.length));
165 0 : dump_data(2, in.data, in.length);
166 0 : return NT_STATUS_INVALID_PARAMETER;
167 : }
168 :
169 12 : ntlmssp_state->neg_flags = neg_flags;
170 12 : DEBUG(3, ("Imported Negotiate flags:\n"));
171 12 : debug_ntlmssp_flags(neg_flags);
172 :
173 12 : if (ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_UNICODE) {
174 12 : ntlmssp_state->unicode = true;
175 : } else {
176 0 : ntlmssp_state->unicode = false;
177 : }
178 :
179 12 : if (ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_SIGN) {
180 8 : gensec_security->want_features |= GENSEC_FEATURE_SIGN;
181 : }
182 :
183 12 : if (ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_SEAL) {
184 0 : gensec_security->want_features |= GENSEC_FEATURE_SEAL;
185 : }
186 :
187 12 : ntlmssp_state->conf_flags = ntlmssp_state->neg_flags;
188 12 : ntlmssp_state->required_flags = 0;
189 :
190 12 : if (DEBUGLEVEL >= 10) {
191 0 : struct NEGOTIATE_MESSAGE *negotiate = talloc(
192 : ntlmssp_state, struct NEGOTIATE_MESSAGE);
193 0 : if (negotiate != NULL) {
194 0 : status = ntlmssp_pull_NEGOTIATE_MESSAGE(
195 : &in, negotiate, negotiate);
196 0 : if (NT_STATUS_IS_OK(status)) {
197 0 : NDR_PRINT_DEBUG(NEGOTIATE_MESSAGE,
198 : negotiate);
199 : }
200 0 : TALLOC_FREE(negotiate);
201 : }
202 : }
203 :
204 12 : ntlmssp_state->negotiate_blob = data_blob_dup_talloc(ntlmssp_state,
205 : in);
206 12 : if (ntlmssp_state->negotiate_blob.length != in.length) {
207 0 : return NT_STATUS_NO_MEMORY;
208 : }
209 :
210 12 : ntlmssp_state->expected_state = NTLMSSP_CHALLENGE;
211 :
212 12 : return NT_STATUS_MORE_PROCESSING_REQUIRED;
213 : }
214 :
215 : /**
216 : * Next state function for the Challenge Packet. Generate an auth packet.
217 : *
218 : * @param gensec_security GENSEC state
219 : * @param out_mem_ctx Memory context for *out
220 : * @param in The server challnege, as a DATA_BLOB. reply.data must be NULL
221 : * @param out The next request (auth packet) to the server, as an allocated DATA_BLOB, on the out_mem_ctx context
222 : * @return Errors or NT_STATUS_OK.
223 : */
224 :
225 10192 : NTSTATUS ntlmssp_client_challenge(struct gensec_security *gensec_security,
226 : TALLOC_CTX *out_mem_ctx,
227 : const DATA_BLOB in, DATA_BLOB *out)
228 : {
229 7314 : struct gensec_ntlmssp_context *gensec_ntlmssp =
230 10192 : talloc_get_type_abort(gensec_security->private_data,
231 : struct gensec_ntlmssp_context);
232 10192 : struct ntlmssp_state *ntlmssp_state = gensec_ntlmssp->ntlmssp_state;
233 10192 : uint32_t chal_flags, ntlmssp_command, unkn1 = 0, unkn2 = 0;
234 : DATA_BLOB server_domain_blob;
235 : DATA_BLOB challenge_blob;
236 10192 : DATA_BLOB target_info = data_blob(NULL, 0);
237 : char *server_domain;
238 : const char *chal_parse_string;
239 10192 : const char *chal_parse_string_short = NULL;
240 : const char *auth_gen_string;
241 10192 : DATA_BLOB lm_response = data_blob(NULL, 0);
242 10192 : DATA_BLOB nt_response = data_blob(NULL, 0);
243 10192 : DATA_BLOB session_key = data_blob(NULL, 0);
244 10192 : DATA_BLOB lm_session_key = data_blob(NULL, 0);
245 10192 : DATA_BLOB encrypted_session_key = data_blob(NULL, 0);
246 : NTSTATUS nt_status;
247 10192 : int flags = 0;
248 10192 : const char *user = NULL, *domain = NULL, *workstation = NULL;
249 10192 : bool is_anonymous = false;
250 10192 : const DATA_BLOB version_blob = ntlmssp_version_blob();
251 10192 : const NTTIME *server_timestamp = NULL;
252 10192 : uint8_t mic_buffer[NTLMSSP_MIC_SIZE] = { 0, };
253 10192 : DATA_BLOB mic_blob = data_blob_const(mic_buffer, sizeof(mic_buffer));
254 10192 : gnutls_hmac_hd_t hmac_hnd = NULL;
255 : int rc;
256 :
257 10192 : TALLOC_CTX *mem_ctx = talloc_new(out_mem_ctx);
258 10192 : if (!mem_ctx) {
259 0 : return NT_STATUS_NO_MEMORY;
260 : }
261 :
262 10192 : if (!msrpc_parse(mem_ctx,
263 : &in, "CdBd",
264 : "NTLMSSP",
265 : &ntlmssp_command,
266 : &server_domain_blob,
267 : &chal_flags)) {
268 0 : DEBUG(1, ("Failed to parse the NTLMSSP Challenge: (#1)\n"));
269 0 : dump_data(2, in.data, in.length);
270 0 : talloc_free(mem_ctx);
271 :
272 0 : return NT_STATUS_INVALID_PARAMETER;
273 : }
274 :
275 10192 : data_blob_free(&server_domain_blob);
276 :
277 10192 : DEBUG(3, ("Got challenge flags:\n"));
278 10192 : debug_ntlmssp_flags(chal_flags);
279 :
280 10192 : nt_status = ntlmssp_handle_neg_flags(ntlmssp_state,
281 : chal_flags, "challenge");
282 10192 : if (!NT_STATUS_IS_OK(nt_status)) {
283 0 : return nt_status;
284 : }
285 :
286 10192 : if (ntlmssp_state->unicode) {
287 10192 : if (chal_flags & NTLMSSP_NEGOTIATE_TARGET_INFO) {
288 10192 : chal_parse_string = "CdUdbddB";
289 : } else {
290 0 : chal_parse_string = "CdUdbdd";
291 0 : chal_parse_string_short = "CdUdb";
292 : }
293 10192 : auth_gen_string = "CdBBUUUBdbb";
294 : } else {
295 0 : if (chal_flags & NTLMSSP_NEGOTIATE_TARGET_INFO) {
296 0 : chal_parse_string = "CdAdbddB";
297 : } else {
298 0 : chal_parse_string = "CdAdbdd";
299 0 : chal_parse_string_short = "CdAdb";
300 : }
301 :
302 0 : auth_gen_string = "CdBBAAABdbb";
303 : }
304 :
305 10192 : if (!msrpc_parse(mem_ctx,
306 : &in, chal_parse_string,
307 : "NTLMSSP",
308 : &ntlmssp_command,
309 : &server_domain,
310 : &chal_flags,
311 : &challenge_blob, 8,
312 : &unkn1, &unkn2,
313 : &target_info)) {
314 :
315 0 : bool ok = false;
316 :
317 0 : DEBUG(1, ("Failed to parse the NTLMSSP Challenge: (#2)\n"));
318 :
319 0 : if (chal_parse_string_short != NULL) {
320 : /*
321 : * In the case where NTLMSSP_NEGOTIATE_TARGET_INFO
322 : * is not used, some NTLMSSP servers don't return
323 : * the unused unkn1 and unkn2 fields.
324 : * See bug:
325 : * https://bugzilla.samba.org/show_bug.cgi?id=10016
326 : * for packet traces.
327 : * Try and parse again without them.
328 : */
329 0 : ok = msrpc_parse(mem_ctx,
330 : &in, chal_parse_string_short,
331 : "NTLMSSP",
332 : &ntlmssp_command,
333 : &server_domain,
334 : &chal_flags,
335 : &challenge_blob, 8);
336 0 : if (!ok) {
337 0 : DEBUG(1, ("Failed to short parse "
338 : "the NTLMSSP Challenge: (#2)\n"));
339 : }
340 : }
341 :
342 0 : if (!ok) {
343 0 : dump_data(2, in.data, in.length);
344 0 : talloc_free(mem_ctx);
345 0 : return NT_STATUS_INVALID_PARAMETER;
346 : }
347 : }
348 :
349 10192 : if (DEBUGLEVEL >= 10) {
350 0 : struct CHALLENGE_MESSAGE *challenge =
351 0 : talloc(ntlmssp_state, struct CHALLENGE_MESSAGE);
352 0 : if (challenge != NULL) {
353 : NTSTATUS status;
354 0 : challenge->NegotiateFlags = chal_flags;
355 0 : status = ntlmssp_pull_CHALLENGE_MESSAGE(
356 : &in, challenge, challenge);
357 0 : if (NT_STATUS_IS_OK(status)) {
358 0 : NDR_PRINT_DEBUG(CHALLENGE_MESSAGE,
359 : challenge);
360 : }
361 0 : TALLOC_FREE(challenge);
362 : }
363 : }
364 :
365 10192 : if (chal_flags & NTLMSSP_TARGET_TYPE_SERVER) {
366 27 : ntlmssp_state->server.is_standalone = true;
367 : } else {
368 10165 : ntlmssp_state->server.is_standalone = false;
369 : }
370 : /* TODO: parse struct_blob and fill in the rest */
371 10192 : ntlmssp_state->server.netbios_name = "";
372 10192 : ntlmssp_state->server.netbios_domain = talloc_move(ntlmssp_state, &server_domain);
373 10192 : ntlmssp_state->server.dns_name = "";
374 10192 : ntlmssp_state->server.dns_domain = "";
375 :
376 10192 : if (challenge_blob.length != 8) {
377 0 : talloc_free(mem_ctx);
378 0 : return NT_STATUS_INVALID_PARAMETER;
379 : }
380 :
381 10192 : is_anonymous = cli_credentials_is_anonymous(gensec_security->credentials);
382 10192 : cli_credentials_get_ntlm_username_domain(gensec_security->credentials, mem_ctx,
383 : &user, &domain);
384 :
385 10192 : workstation = cli_credentials_get_workstation(gensec_security->credentials);
386 :
387 10192 : if (user == NULL) {
388 0 : DEBUG(10, ("User is NULL, returning INVALID_PARAMETER\n"));
389 0 : return NT_STATUS_INVALID_PARAMETER;
390 : }
391 :
392 10192 : if (domain == NULL) {
393 0 : DEBUG(10, ("Domain is NULL, returning INVALID_PARAMETER\n"));
394 0 : return NT_STATUS_INVALID_PARAMETER;
395 : }
396 :
397 10192 : if (workstation == NULL) {
398 0 : DEBUG(10, ("Workstation is NULL, returning INVALID_PARAMETER\n"));
399 0 : return NT_STATUS_INVALID_PARAMETER;
400 : }
401 :
402 10192 : if (is_anonymous) {
403 497 : ntlmssp_state->neg_flags |= NTLMSSP_ANONYMOUS;
404 : /*
405 : * don't use the ccache for anonymous auth
406 : */
407 497 : ntlmssp_state->use_ccache = false;
408 : }
409 10192 : if (ntlmssp_state->use_ccache) {
410 12 : struct samr_Password *nt_hash = NULL;
411 :
412 : /*
413 : * If we have a password given we don't
414 : * use the ccache
415 : */
416 12 : nt_hash = cli_credentials_get_nt_hash(gensec_security->credentials,
417 : mem_ctx);
418 12 : if (nt_hash != NULL) {
419 0 : ZERO_STRUCTP(nt_hash);
420 0 : TALLOC_FREE(nt_hash);
421 0 : ntlmssp_state->use_ccache = false;
422 : }
423 : }
424 :
425 10192 : if (ntlmssp_state->use_ccache) {
426 : struct wbcCredentialCacheParams params;
427 12 : struct wbcCredentialCacheInfo *info = NULL;
428 12 : struct wbcAuthErrorInfo *error = NULL;
429 : struct wbcNamedBlob auth_blobs[2];
430 12 : const struct wbcBlob *wbc_auth_blob = NULL;
431 12 : const struct wbcBlob *wbc_session_key = NULL;
432 : wbcErr wbc_status;
433 : size_t i;
434 12 : bool new_spnego = false;
435 :
436 12 : params.account_name = user;
437 12 : params.domain_name = domain;
438 12 : params.level = WBC_CREDENTIAL_CACHE_LEVEL_NTLMSSP;
439 :
440 12 : auth_blobs[0].name = "challenge_blob";
441 12 : auth_blobs[0].flags = 0;
442 12 : auth_blobs[0].blob.data = in.data;
443 12 : auth_blobs[0].blob.length = in.length;
444 12 : auth_blobs[1].name = "negotiate_blob";
445 12 : auth_blobs[1].flags = 0;
446 12 : auth_blobs[1].blob.data = ntlmssp_state->negotiate_blob.data;
447 12 : auth_blobs[1].blob.length = ntlmssp_state->negotiate_blob.length;
448 12 : params.num_blobs = ARRAY_SIZE(auth_blobs);
449 12 : params.blobs = auth_blobs;
450 :
451 12 : wbc_status = wbcCredentialCache(¶ms, &info, &error);
452 12 : wbcFreeMemory(error);
453 12 : if (!WBC_ERROR_IS_OK(wbc_status)) {
454 0 : return NT_STATUS_WRONG_CREDENTIAL_HANDLE;
455 : }
456 :
457 44 : for (i=0; i<info->num_blobs; i++) {
458 32 : if (strequal(info->blobs[i].name, "auth_blob")) {
459 12 : wbc_auth_blob = &info->blobs[i].blob;
460 : }
461 32 : if (strequal(info->blobs[i].name, "session_key")) {
462 12 : wbc_session_key = &info->blobs[i].blob;
463 : }
464 32 : if (strequal(info->blobs[i].name, "new_spnego")) {
465 8 : new_spnego = true;
466 : }
467 : }
468 12 : if ((wbc_auth_blob == NULL) || (wbc_session_key == NULL)) {
469 0 : wbcFreeMemory(info);
470 0 : return NT_STATUS_WRONG_CREDENTIAL_HANDLE;
471 : }
472 :
473 12 : session_key = data_blob_talloc(mem_ctx,
474 : wbc_session_key->data,
475 : wbc_session_key->length);
476 12 : if (session_key.length != wbc_session_key->length) {
477 0 : wbcFreeMemory(info);
478 0 : return NT_STATUS_NO_MEMORY;
479 : }
480 12 : *out = data_blob_talloc(mem_ctx,
481 : wbc_auth_blob->data,
482 : wbc_auth_blob->length);
483 12 : if (out->length != wbc_auth_blob->length) {
484 0 : wbcFreeMemory(info);
485 0 : return NT_STATUS_NO_MEMORY;
486 : }
487 12 : ntlmssp_state->new_spnego = new_spnego;
488 :
489 12 : wbcFreeMemory(info);
490 12 : goto done;
491 : }
492 :
493 10180 : if (ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_NTLM2) {
494 9988 : flags |= CLI_CRED_NTLM2;
495 : }
496 10180 : if (ntlmssp_state->use_ntlmv2) {
497 9913 : flags |= CLI_CRED_NTLMv2_AUTH;
498 : }
499 10180 : if (ntlmssp_state->use_nt_response) {
500 10180 : flags |= CLI_CRED_NTLM_AUTH;
501 : }
502 10180 : if (ntlmssp_state->allow_lm_response) {
503 267 : flags |= CLI_CRED_LANMAN_AUTH;
504 : }
505 :
506 10180 : if (target_info.length != 0 && !is_anonymous) {
507 9683 : struct AV_PAIR *pairs = NULL;
508 9683 : uint32_t count = 0;
509 : enum ndr_err_code err;
510 9683 : struct AV_PAIR *timestamp = NULL;
511 9683 : struct AV_PAIR *eol = NULL;
512 9683 : uint32_t i = 0;
513 9683 : const char *service = NULL;
514 9683 : const char *hostname = NULL;
515 :
516 9683 : err = ndr_pull_struct_blob(&target_info,
517 : ntlmssp_state,
518 9683 : &ntlmssp_state->server.av_pair_list,
519 : (ndr_pull_flags_fn_t)ndr_pull_AV_PAIR_LIST);
520 9683 : if (!NDR_ERR_CODE_IS_SUCCESS(err)) {
521 0 : return ndr_map_error2ntstatus(err);
522 : }
523 :
524 9683 : count = ntlmssp_state->server.av_pair_list.count;
525 : /*
526 : * We need room for Flags, SingleHost,
527 : * ChannelBindings and Target
528 : */
529 9683 : pairs = talloc_zero_array(ntlmssp_state, struct AV_PAIR,
530 : count + 4);
531 9683 : if (pairs == NULL) {
532 0 : return NT_STATUS_NO_MEMORY;
533 : }
534 :
535 66753 : for (i = 0; i < count; i++) {
536 57070 : pairs[i] = ntlmssp_state->server.av_pair_list.pair[i];
537 : }
538 :
539 9683 : ntlmssp_state->client.av_pair_list.count = count;
540 9683 : ntlmssp_state->client.av_pair_list.pair = pairs;
541 :
542 9683 : eol = ndr_ntlmssp_find_av(&ntlmssp_state->client.av_pair_list,
543 : MsvAvEOL);
544 9683 : if (eol == NULL) {
545 0 : return NT_STATUS_INVALID_PARAMETER;
546 : }
547 :
548 9683 : timestamp = ndr_ntlmssp_find_av(&ntlmssp_state->client.av_pair_list,
549 : MsvAvTimestamp);
550 9683 : if (timestamp != NULL) {
551 8655 : uint32_t sign_features =
552 : GENSEC_FEATURE_SESSION_KEY |
553 : GENSEC_FEATURE_SIGN |
554 : GENSEC_FEATURE_SEAL;
555 :
556 8655 : server_timestamp = ×tamp->Value.AvTimestamp;
557 :
558 8655 : if (ntlmssp_state->force_old_spnego) {
559 0 : sign_features = 0;
560 : }
561 :
562 8655 : if (gensec_security->want_features & sign_features) {
563 8499 : struct AV_PAIR *av_flags = NULL;
564 :
565 8499 : av_flags = ndr_ntlmssp_find_av(&ntlmssp_state->client.av_pair_list,
566 : MsvAvFlags);
567 8499 : if (av_flags == NULL) {
568 8499 : av_flags = eol;
569 8499 : eol++;
570 8499 : count++;
571 8499 : *eol = *av_flags;
572 8499 : av_flags->AvId = MsvAvFlags;
573 8499 : av_flags->Value.AvFlags = 0;
574 : }
575 :
576 8499 : av_flags->Value.AvFlags |= NTLMSSP_AVFLAG_MIC_IN_AUTHENTICATE_MESSAGE;
577 8499 : ntlmssp_state->new_spnego = true;
578 : }
579 : }
580 :
581 : {
582 9683 : struct AV_PAIR *SingleHost = NULL;
583 :
584 9683 : SingleHost = eol;
585 9683 : eol++;
586 9683 : count++;
587 9683 : *eol = *SingleHost;
588 :
589 : /*
590 : * This is not really used, but we want to
591 : * add some more random bytes and match
592 : * Windows.
593 : */
594 9683 : SingleHost->AvId = MsvAvSingleHost;
595 9683 : SingleHost->Value.AvSingleHost.token_info.Flags = 0;
596 9683 : SingleHost->Value.AvSingleHost.token_info.TokenIL = 0;
597 9683 : generate_random_buffer(SingleHost->Value.AvSingleHost.token_info.MachineId,
598 : sizeof(SingleHost->Value.AvSingleHost.token_info.MachineId));
599 9683 : SingleHost->Value.AvSingleHost.remaining = data_blob_null;
600 : }
601 :
602 : {
603 9683 : struct AV_PAIR *ChannelBindings = NULL;
604 :
605 9683 : ChannelBindings = eol;
606 9683 : eol++;
607 9683 : count++;
608 9683 : *eol = *ChannelBindings;
609 :
610 : /*
611 : * gensec doesn't support channel bindings yet,
612 : * but we want to match Windows on the wire
613 : */
614 9683 : ChannelBindings->AvId = MsvChannelBindings;
615 9683 : memset(ChannelBindings->Value.ChannelBindings, 0,
616 : sizeof(ChannelBindings->Value.ChannelBindings));
617 : }
618 :
619 9683 : service = gensec_get_target_service(gensec_security);
620 9683 : hostname = gensec_get_target_hostname(gensec_security);
621 9683 : if (service != NULL && hostname != NULL) {
622 9362 : struct AV_PAIR *target = NULL;
623 :
624 9362 : target = eol;
625 9362 : eol++;
626 9362 : count++;
627 9362 : *eol = *target;
628 :
629 9362 : target->AvId = MsvAvTargetName;
630 9362 : target->Value.AvTargetName = talloc_asprintf(pairs, "%s/%s",
631 : service,
632 : hostname);
633 9362 : if (target->Value.AvTargetName == NULL) {
634 0 : return NT_STATUS_NO_MEMORY;
635 : }
636 : }
637 :
638 9683 : ntlmssp_state->client.av_pair_list.count = count;
639 9683 : ntlmssp_state->client.av_pair_list.pair = pairs;
640 :
641 9683 : err = ndr_push_struct_blob(&target_info,
642 : ntlmssp_state,
643 9683 : &ntlmssp_state->client.av_pair_list,
644 : (ndr_push_flags_fn_t)ndr_push_AV_PAIR_LIST);
645 9683 : if (!NDR_ERR_CODE_IS_SUCCESS(err)) {
646 0 : return NT_STATUS_NO_MEMORY;
647 : }
648 : }
649 :
650 10180 : nt_status = cli_credentials_get_ntlm_response(gensec_security->credentials, mem_ctx,
651 : &flags, challenge_blob,
652 : server_timestamp, target_info,
653 : &lm_response, &nt_response,
654 : &lm_session_key, &session_key);
655 10180 : if (!NT_STATUS_IS_OK(nt_status)) {
656 0 : return nt_status;
657 : }
658 :
659 10180 : if (!(flags & CLI_CRED_LANMAN_AUTH)) {
660 : /* LM Key is still possible, just silly, so we do not
661 : * allow it. Fortunetly all LM crypto is off by
662 : * default and we require command line options to end
663 : * up here */
664 9952 : ntlmssp_state->neg_flags &= ~NTLMSSP_NEGOTIATE_LM_KEY;
665 : }
666 :
667 10180 : if (!(flags & CLI_CRED_NTLM2)) {
668 : /* NTLM2 is incompatible... */
669 691 : ntlmssp_state->neg_flags &= ~NTLMSSP_NEGOTIATE_NTLM2;
670 : }
671 :
672 10180 : if ((ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_LM_KEY)
673 0 : && ntlmssp_state->allow_lm_key && lm_session_key.length == 16) {
674 0 : DATA_BLOB new_session_key = data_blob_talloc(mem_ctx, NULL, 16);
675 0 : if (lm_response.length == 24) {
676 0 : nt_status = SMBsesskeygen_lm_sess_key(lm_session_key.data,
677 0 : lm_response.data,
678 : new_session_key.data);
679 0 : if (!NT_STATUS_IS_OK(nt_status)) {
680 0 : return nt_status;
681 : }
682 : } else {
683 : static const uint8_t zeros[24];
684 0 : nt_status = SMBsesskeygen_lm_sess_key(lm_session_key.data,
685 : zeros,
686 : new_session_key.data);
687 0 : if (!NT_STATUS_IS_OK(nt_status)) {
688 0 : return nt_status;
689 : }
690 : }
691 0 : session_key = new_session_key;
692 0 : dump_data_pw("LM session key\n", session_key.data, session_key.length);
693 : }
694 :
695 :
696 : /* Key exchange encryptes a new client-generated session key with
697 : the password-derived key */
698 10180 : if (ntlmssp_state->neg_flags & NTLMSSP_NEGOTIATE_KEY_EXCH) {
699 : /* Make up a new session key */
700 : uint8_t client_session_key[16];
701 : gnutls_cipher_hd_t cipher_hnd;
702 17487 : gnutls_datum_t enc_session_key = {
703 10180 : .data = session_key.data,
704 10180 : .size = session_key.length,
705 : };
706 :
707 10180 : generate_random_buffer(client_session_key, sizeof(client_session_key));
708 :
709 : /* Encrypt the new session key with the old one */
710 10180 : encrypted_session_key = data_blob_talloc(ntlmssp_state,
711 : client_session_key, sizeof(client_session_key));
712 10180 : dump_data_pw("KEY_EXCH session key:\n", encrypted_session_key.data, encrypted_session_key.length);
713 :
714 10180 : rc = gnutls_cipher_init(&cipher_hnd,
715 : GNUTLS_CIPHER_ARCFOUR_128,
716 : &enc_session_key,
717 : NULL);
718 10180 : if (rc < 0) {
719 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
720 0 : ZERO_ARRAY(client_session_key);
721 0 : goto done;
722 : }
723 17487 : rc = gnutls_cipher_encrypt(cipher_hnd,
724 10180 : encrypted_session_key.data,
725 : encrypted_session_key.length);
726 10180 : gnutls_cipher_deinit(cipher_hnd);
727 10180 : if (rc < 0) {
728 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
729 0 : ZERO_ARRAY(client_session_key);
730 0 : goto done;
731 : }
732 :
733 10180 : dump_data_pw("KEY_EXCH session key (enc):\n", encrypted_session_key.data, encrypted_session_key.length);
734 :
735 : /* Mark the new session key as the 'real' session key */
736 10180 : session_key = data_blob_talloc(mem_ctx, client_session_key, sizeof(client_session_key));
737 10180 : ZERO_ARRAY(client_session_key);
738 : }
739 :
740 : /* this generates the actual auth packet */
741 10180 : nt_status = msrpc_gen(mem_ctx,
742 : out, auth_gen_string,
743 : "NTLMSSP",
744 : NTLMSSP_AUTH,
745 : lm_response.data, lm_response.length,
746 : nt_response.data, nt_response.length,
747 : domain,
748 : user,
749 : workstation,
750 : encrypted_session_key.data, encrypted_session_key.length,
751 : ntlmssp_state->neg_flags,
752 2873 : version_blob.data, version_blob.length,
753 : mic_blob.data, mic_blob.length);
754 10180 : if (!NT_STATUS_IS_OK(nt_status)) {
755 0 : talloc_free(mem_ctx);
756 0 : return nt_status;
757 : }
758 :
759 10180 : if (DEBUGLEVEL >= 10) {
760 0 : struct AUTHENTICATE_MESSAGE *authenticate =
761 0 : talloc(ntlmssp_state, struct AUTHENTICATE_MESSAGE);
762 0 : if (authenticate != NULL) {
763 : NTSTATUS status;
764 0 : authenticate->NegotiateFlags = ntlmssp_state->neg_flags;
765 0 : status = ntlmssp_pull_AUTHENTICATE_MESSAGE(
766 : out, authenticate, authenticate);
767 0 : if (NT_STATUS_IS_OK(status)) {
768 0 : NDR_PRINT_DEBUG(AUTHENTICATE_MESSAGE,
769 : authenticate);
770 : }
771 0 : TALLOC_FREE(authenticate);
772 : }
773 : }
774 :
775 : /*
776 : * We always include the MIC, even without:
777 : * av_flags->Value.AvFlags |= NTLMSSP_AVFLAG_MIC_IN_AUTHENTICATE_MESSAGE;
778 : * ntlmssp_state->new_spnego = true;
779 : *
780 : * This matches a Windows client.
781 : */
782 17487 : rc = gnutls_hmac_init(&hmac_hnd,
783 : GNUTLS_MAC_MD5,
784 10180 : session_key.data,
785 10180 : MIN(session_key.length, 64));
786 10180 : if (rc < 0) {
787 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
788 0 : goto done;
789 : }
790 :
791 17487 : rc = gnutls_hmac(hmac_hnd,
792 10180 : ntlmssp_state->negotiate_blob.data,
793 : ntlmssp_state->negotiate_blob.length);
794 10180 : if (rc < 0) {
795 0 : gnutls_hmac_deinit(hmac_hnd, NULL);
796 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
797 0 : goto done;
798 : }
799 10180 : rc = gnutls_hmac(hmac_hnd, in.data, in.length);
800 10180 : if (rc < 0) {
801 0 : gnutls_hmac_deinit(hmac_hnd, NULL);
802 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
803 0 : goto done;
804 : }
805 10180 : rc = gnutls_hmac(hmac_hnd, out->data, out->length);
806 10180 : if (rc < 0) {
807 0 : gnutls_hmac_deinit(hmac_hnd, NULL);
808 0 : nt_status = gnutls_error_to_ntstatus(rc, NT_STATUS_NTLM_BLOCKED);
809 0 : goto done;
810 : }
811 :
812 10180 : gnutls_hmac_deinit(hmac_hnd, mic_buffer);
813 :
814 10180 : memcpy(out->data + NTLMSSP_MIC_OFFSET, mic_buffer, NTLMSSP_MIC_SIZE);
815 10180 : ZERO_ARRAY(mic_buffer);
816 :
817 10180 : nt_status = NT_STATUS_OK;
818 10192 : done:
819 10192 : ZERO_ARRAY_LEN(ntlmssp_state->negotiate_blob.data,
820 : ntlmssp_state->negotiate_blob.length);
821 10192 : data_blob_free(&ntlmssp_state->negotiate_blob);
822 :
823 10192 : ntlmssp_state->session_key = session_key;
824 10192 : talloc_steal(ntlmssp_state, session_key.data);
825 :
826 10192 : DEBUG(3, ("NTLMSSP: Set final flags:\n"));
827 10192 : debug_ntlmssp_flags(ntlmssp_state->neg_flags);
828 :
829 10192 : talloc_steal(out_mem_ctx, out->data);
830 :
831 10192 : ntlmssp_state->expected_state = NTLMSSP_DONE;
832 :
833 10192 : if (gensec_ntlmssp_have_feature(gensec_security, GENSEC_FEATURE_SIGN)) {
834 9963 : nt_status = ntlmssp_sign_init(ntlmssp_state);
835 9963 : if (!NT_STATUS_IS_OK(nt_status)) {
836 0 : DEBUG(1, ("Could not setup NTLMSSP signing/sealing system (error was: %s)\n",
837 : nt_errstr(nt_status)));
838 0 : talloc_free(mem_ctx);
839 0 : return nt_status;
840 : }
841 : }
842 :
843 10192 : talloc_free(mem_ctx);
844 10192 : return nt_status;
845 : }
846 :
847 10307 : NTSTATUS gensec_ntlmssp_client_start(struct gensec_security *gensec_security)
848 : {
849 : struct gensec_ntlmssp_context *gensec_ntlmssp;
850 : struct ntlmssp_state *ntlmssp_state;
851 : NTSTATUS nt_status;
852 :
853 10307 : nt_status = gensec_ntlmssp_start(gensec_security);
854 10307 : NT_STATUS_NOT_OK_RETURN(nt_status);
855 :
856 7391 : gensec_ntlmssp =
857 10307 : talloc_get_type_abort(gensec_security->private_data,
858 : struct gensec_ntlmssp_context);
859 :
860 10307 : ntlmssp_state = talloc_zero(gensec_ntlmssp,
861 : struct ntlmssp_state);
862 10307 : if (!ntlmssp_state) {
863 0 : return NT_STATUS_NO_MEMORY;
864 : }
865 :
866 10307 : gensec_ntlmssp->ntlmssp_state = ntlmssp_state;
867 :
868 10307 : ntlmssp_state = gensec_ntlmssp->ntlmssp_state;
869 :
870 10307 : ntlmssp_state->role = NTLMSSP_CLIENT;
871 :
872 10307 : ntlmssp_state->client.netbios_domain = lpcfg_workgroup(gensec_security->settings->lp_ctx);
873 10307 : ntlmssp_state->client.netbios_name = cli_credentials_get_workstation(gensec_security->credentials);
874 :
875 10307 : ntlmssp_state->unicode = gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "unicode", true);
876 :
877 10307 : ntlmssp_state->use_nt_response = \
878 10307 : gensec_setting_bool(gensec_security->settings,
879 : "ntlmssp_client",
880 : "send_nt_response",
881 : true);
882 :
883 10307 : ntlmssp_state->allow_lm_response = lpcfg_client_lanman_auth(gensec_security->settings->lp_ctx);
884 :
885 20614 : ntlmssp_state->allow_lm_key = (ntlmssp_state->allow_lm_response
886 12687 : && (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "allow_lm_key", false)
887 7950 : || gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "lm_key", false)));
888 :
889 10307 : ntlmssp_state->use_ntlmv2 = lpcfg_client_ntlmv2_auth(gensec_security->settings->lp_ctx);
890 :
891 10307 : ntlmssp_state->force_old_spnego = gensec_setting_bool(gensec_security->settings,
892 : "ntlmssp_client", "force_old_spnego", false);
893 :
894 10307 : ntlmssp_state->expected_state = NTLMSSP_INITIAL;
895 :
896 10307 : ntlmssp_state->neg_flags =
897 : NTLMSSP_NEGOTIATE_NTLM |
898 : NTLMSSP_NEGOTIATE_VERSION |
899 : NTLMSSP_REQUEST_TARGET;
900 :
901 10307 : if (ntlmssp_state->unicode) {
902 10307 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_UNICODE;
903 : } else {
904 0 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_OEM;
905 : }
906 :
907 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "128bit", true)) {
908 10147 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_128;
909 : }
910 :
911 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "56bit", false)) {
912 96 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_56;
913 : }
914 :
915 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "lm_key", false)) {
916 192 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_LM_KEY;
917 : }
918 :
919 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "keyexchange", true)) {
920 10307 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_KEY_EXCH;
921 : }
922 :
923 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "alwayssign", true)) {
924 10307 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_ALWAYS_SIGN;
925 : }
926 :
927 10307 : if (gensec_setting_bool(gensec_security->settings, "ntlmssp_client", "ntlm2", true)) {
928 10115 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_NTLM2;
929 : } else {
930 : /* apparently we can't do ntlmv2 if we don't do ntlm2 */
931 192 : ntlmssp_state->use_ntlmv2 = false;
932 : }
933 :
934 10307 : if (ntlmssp_state->use_ntlmv2) {
935 10040 : ntlmssp_state->required_flags |= NTLMSSP_NEGOTIATE_NTLM2;
936 10040 : ntlmssp_state->allow_lm_response = false;
937 10040 : ntlmssp_state->allow_lm_key = false;
938 : }
939 :
940 10307 : if (ntlmssp_state->allow_lm_key) {
941 96 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_LM_KEY;
942 : }
943 :
944 10307 : if (gensec_security->want_features & GENSEC_FEATURE_SESSION_KEY) {
945 : /*
946 : * We need to set this to allow a later SetPassword
947 : * via the SAMR pipe to succeed. Strange.... We could
948 : * also add NTLMSSP_NEGOTIATE_SEAL here. JRA.
949 : *
950 : * Without this, Windows will not create the master key
951 : * that it thinks is only used for NTLMSSP signing and
952 : * sealing. (It is actually pulled out and used directly)
953 : *
954 : * We don't require this here as some servers (e.g. NetAPP)
955 : * doesn't support this.
956 : */
957 1839 : ntlmssp_state->neg_flags |= NTLMSSP_NEGOTIATE_SIGN;
958 : }
959 10307 : if (gensec_security->want_features & GENSEC_FEATURE_SIGN) {
960 8353 : ntlmssp_state->required_flags |= NTLMSSP_NEGOTIATE_SIGN;
961 :
962 8353 : if (gensec_security->want_features & GENSEC_FEATURE_LDAP_STYLE) {
963 : /*
964 : * We need to handle NTLMSSP_NEGOTIATE_SIGN as
965 : * NTLMSSP_NEGOTIATE_SEAL if GENSEC_FEATURE_LDAP_STYLE
966 : * is requested.
967 : */
968 6678 : ntlmssp_state->force_wrap_seal = true;
969 : }
970 : }
971 10307 : if (ntlmssp_state->force_wrap_seal) {
972 : bool ret;
973 :
974 : /*
975 : * We want also work against old Samba servers
976 : * which didn't had GENSEC_FEATURE_LDAP_STYLE
977 : * we negotiate SEAL too. We may remove this
978 : * in a few years. As all servers should have
979 : * GENSEC_FEATURE_LDAP_STYLE by then.
980 : */
981 6678 : ret = gensec_setting_bool(gensec_security->settings,
982 : "ntlmssp_client",
983 : "ldap_style_send_seal",
984 : true);
985 6678 : if (ret) {
986 6673 : ntlmssp_state->required_flags |= NTLMSSP_NEGOTIATE_SEAL;
987 : }
988 : }
989 10307 : if (gensec_security->want_features & GENSEC_FEATURE_SEAL) {
990 7306 : ntlmssp_state->required_flags |= NTLMSSP_NEGOTIATE_SIGN;
991 7306 : ntlmssp_state->required_flags |= NTLMSSP_NEGOTIATE_SEAL;
992 : }
993 10307 : if (gensec_security->want_features & GENSEC_FEATURE_NTLM_CCACHE) {
994 12 : ntlmssp_state->use_ccache = true;
995 : }
996 :
997 10307 : ntlmssp_state->neg_flags |= ntlmssp_state->required_flags;
998 10307 : ntlmssp_state->conf_flags = ntlmssp_state->neg_flags;
999 :
1000 10307 : return NT_STATUS_OK;
1001 : }
1002 :
1003 12 : NTSTATUS gensec_ntlmssp_resume_ccache_start(struct gensec_security *gensec_security)
1004 : {
1005 12 : struct gensec_ntlmssp_context *gensec_ntlmssp = NULL;
1006 : NTSTATUS status;
1007 :
1008 12 : status = gensec_ntlmssp_client_start(gensec_security);
1009 12 : if (!NT_STATUS_IS_OK(status)) {
1010 0 : return status;
1011 : }
1012 :
1013 12 : gensec_ntlmssp = talloc_get_type_abort(gensec_security->private_data,
1014 : struct gensec_ntlmssp_context);
1015 12 : gensec_ntlmssp->ntlmssp_state->use_ccache = false;
1016 12 : gensec_ntlmssp->ntlmssp_state->resume_ccache = true;
1017 12 : gensec_ntlmssp->ntlmssp_state->expected_state = NTLMSSP_NEGOTIATE;
1018 :
1019 12 : return NT_STATUS_OK;
1020 : }
|