cifsencrypt.c 22.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 *   fs/cifs/cifsencrypt.c
 *
4
 *   Copyright (C) International Business Machines  Corp., 2005,2006
L
Linus Torvalds 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 *   Author(s): Steve French (sfrench@us.ibm.com)
 *
 *   This library is free software; you can redistribute it and/or modify
 *   it under the terms of the GNU Lesser General Public License as published
 *   by the Free Software Foundation; either version 2.1 of the License, or
 *   (at your option) any later version.
 *
 *   This library 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 Lesser General Public License for more details.
 *
 *   You should have received a copy of the GNU Lesser General Public License
 *   along with this library; if not, write to the Free Software
 *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 */

#include <linux/fs.h>
23
#include <linux/slab.h>
L
Linus Torvalds 已提交
24
#include "cifspdu.h"
S
Steve French 已提交
25
#include "cifsglob.h"
L
Linus Torvalds 已提交
26 27 28
#include "cifs_debug.h"
#include "cifs_unicode.h"
#include "cifsproto.h"
29
#include "ntlmssp.h"
30
#include <linux/ctype.h>
S
Steve French 已提交
31
#include <linux/random.h>
L
Linus Torvalds 已提交
32

33 34 35 36 37 38 39
/*
 * Calculate and return the CIFS signature based on the mac key and SMB PDU.
 * The 16 byte signature must be allocated by the caller. Note we only use the
 * 1st eight bytes and that the smb header signature field on input contains
 * the sequence number before this function is called. Also, this function
 * should be called with the server->srv_mutex held.
 */
S
Steve French 已提交
40
static int cifs_calculate_signature(const struct smb_hdr *cifs_pdu,
41
				struct TCP_Server_Info *server, char *signature)
L
Linus Torvalds 已提交
42
{
43
	int rc;
L
Linus Torvalds 已提交
44

45
	if (cifs_pdu == NULL || signature == NULL || server == NULL)
L
Linus Torvalds 已提交
46 47
		return -EINVAL;

48 49 50 51 52 53 54
	if (!server->secmech.sdescmd5) {
		cERROR(1, "%s: Can't generate signature\n", __func__);
		return -1;
	}

	rc = crypto_shash_init(&server->secmech.sdescmd5->shash);
	if (rc) {
55
		cERROR(1, "%s: Could not init md5\n", __func__);
56 57 58
		return rc;
	}

59
	rc = crypto_shash_update(&server->secmech.sdescmd5->shash,
60
		server->session_key.response, server->session_key.len);
61 62 63 64
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		return rc;
	}
65

66
	rc = crypto_shash_update(&server->secmech.sdescmd5->shash,
67
		cifs_pdu->Protocol, be32_to_cpu(cifs_pdu->smb_buf_length));
68 69 70 71
	if (rc) {
		cERROR(1, "%s: Could not update with payload\n", __func__);
		return rc;
	}
72 73

	rc = crypto_shash_final(&server->secmech.sdescmd5->shash, signature);
74 75
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
76

77
	return rc;
L
Linus Torvalds 已提交
78 79
}

80
/* must be called with server->srv_mutex held */
S
Steve French 已提交
81 82
int cifs_sign_smb(struct smb_hdr *cifs_pdu, struct TCP_Server_Info *server,
		  __u32 *pexpected_response_sequence_number)
L
Linus Torvalds 已提交
83 84 85 86
{
	int rc = 0;
	char smb_signature[20];

S
Steve French 已提交
87
	if ((cifs_pdu == NULL) || (server == NULL))
L
Linus Torvalds 已提交
88 89
		return -EINVAL;

90 91
	if (!(cifs_pdu->Flags2 & SMBFLG2_SECURITY_SIGNATURE) ||
	    server->tcpStatus == CifsNeedNegotiate)
L
Linus Torvalds 已提交
92 93
		return rc;

94
	if (!server->session_estab) {
95
		memcpy(cifs_pdu->Signature.SecuritySignature, "BSRSPYL", 8);
96 97 98
		return rc;
	}

99 100
	cifs_pdu->Signature.Sequence.SequenceNumber =
			cpu_to_le32(server->sequence_number);
L
Linus Torvalds 已提交
101
	cifs_pdu->Signature.Sequence.Reserved = 0;
102

103 104
	*pexpected_response_sequence_number = server->sequence_number++;
	server->sequence_number++;
L
Linus Torvalds 已提交
105

106
	rc = cifs_calculate_signature(cifs_pdu, server, smb_signature);
S
Steve French 已提交
107
	if (rc)
L
Linus Torvalds 已提交
108 109 110 111 112 113 114
		memset(cifs_pdu->Signature.SecuritySignature, 0, 8);
	else
		memcpy(cifs_pdu->Signature.SecuritySignature, smb_signature, 8);

	return rc;
}

S
Steve French 已提交
115
static int cifs_calc_signature2(const struct kvec *iov, int n_vec,
116
				struct TCP_Server_Info *server, char *signature)
117
{
118
	int i;
119
	int rc;
120

121
	if (iov == NULL || signature == NULL || server == NULL)
122
		return -EINVAL;
123

124 125 126 127 128 129 130
	if (!server->secmech.sdescmd5) {
		cERROR(1, "%s: Can't generate signature\n", __func__);
		return -1;
	}

	rc = crypto_shash_init(&server->secmech.sdescmd5->shash);
	if (rc) {
131
		cERROR(1, "%s: Could not init md5\n", __func__);
132 133 134
		return rc;
	}

135
	rc = crypto_shash_update(&server->secmech.sdescmd5->shash,
136
		server->session_key.response, server->session_key.len);
137 138 139 140
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		return rc;
	}
141

142
	for (i = 0; i < n_vec; i++) {
143 144
		if (iov[i].iov_len == 0)
			continue;
S
Steve French 已提交
145
		if (iov[i].iov_base == NULL) {
146
			cERROR(1, "null iovec entry");
147
			return -EIO;
148
		}
S
Steve French 已提交
149
		/* The first entry includes a length field (which does not get
150
		   signed that occupies the first 4 bytes before the header */
S
Steve French 已提交
151
		if (i == 0) {
152
			if (iov[0].iov_len <= 8) /* cmd field at offset 9 */
153
				break; /* nothing to sign or corrupt header */
154
			rc =
155 156
			crypto_shash_update(&server->secmech.sdescmd5->shash,
				iov[i].iov_base + 4, iov[i].iov_len - 4);
157 158
		} else {
			rc =
159 160
			crypto_shash_update(&server->secmech.sdescmd5->shash,
				iov[i].iov_base, iov[i].iov_len);
161 162 163 164 165 166
		}
		if (rc) {
			cERROR(1, "%s: Could not update with payload\n",
							__func__);
			return rc;
		}
167
	}
168

169
	rc = crypto_shash_final(&server->secmech.sdescmd5->shash, signature);
170 171
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
172

173
	return rc;
174 175
}

176
/* must be called with server->srv_mutex held */
S
Steve French 已提交
177
int cifs_sign_smb2(struct kvec *iov, int n_vec, struct TCP_Server_Info *server,
178
		   __u32 *pexpected_response_sequence_number)
179 180 181
{
	int rc = 0;
	char smb_signature[20];
S
Steve French 已提交
182
	struct smb_hdr *cifs_pdu = iov[0].iov_base;
183

S
Steve French 已提交
184
	if ((cifs_pdu == NULL) || (server == NULL))
185 186
		return -EINVAL;

187 188
	if (!(cifs_pdu->Flags2 & SMBFLG2_SECURITY_SIGNATURE) ||
	    server->tcpStatus == CifsNeedNegotiate)
189 190
		return rc;

191
	if (!server->session_estab) {
192
		memcpy(cifs_pdu->Signature.SecuritySignature, "BSRSPYL", 8);
193 194 195
		return rc;
	}

S
Steve French 已提交
196
	cifs_pdu->Signature.Sequence.SequenceNumber =
197
				cpu_to_le32(server->sequence_number);
S
Steve French 已提交
198
	cifs_pdu->Signature.Sequence.Reserved = 0;
199

S
Steve French 已提交
200 201
	*pexpected_response_sequence_number = server->sequence_number++;
	server->sequence_number++;
202

203
	rc = cifs_calc_signature2(iov, n_vec, server, smb_signature);
S
Steve French 已提交
204 205 206 207
	if (rc)
		memset(cifs_pdu->Signature.SecuritySignature, 0, 8);
	else
		memcpy(cifs_pdu->Signature.SecuritySignature, smb_signature, 8);
208

S
Steve French 已提交
209
	return rc;
210 211
}

212
int cifs_verify_signature(struct smb_hdr *cifs_pdu,
213
			  struct TCP_Server_Info *server,
S
Steve French 已提交
214
			  __u32 expected_sequence_number)
L
Linus Torvalds 已提交
215
{
216
	unsigned int rc;
L
Linus Torvalds 已提交
217 218 219
	char server_response_sig[8];
	char what_we_think_sig_should_be[20];

220
	if (cifs_pdu == NULL || server == NULL)
L
Linus Torvalds 已提交
221 222
		return -EINVAL;

223
	if (!server->session_estab)
L
Linus Torvalds 已提交
224 225 226
		return 0;

	if (cifs_pdu->Command == SMB_COM_LOCKING_ANDX) {
227
		struct smb_com_lock_req *pSMB =
S
Steve French 已提交
228 229
			(struct smb_com_lock_req *)cifs_pdu;
	    if (pSMB->LockType & LOCKING_ANDX_OPLOCK_RELEASE)
L
Linus Torvalds 已提交
230 231 232
			return 0;
	}

233 234 235
	/* BB what if signatures are supposed to be on for session but
	   server does not send one? BB */

L
Linus Torvalds 已提交
236
	/* Do not need to verify session setups with signature "BSRSPYL "  */
237
	if (memcmp(cifs_pdu->Signature.SecuritySignature, "BSRSPYL ", 8) == 0)
238 239
		cFYI(1, "dummy signature received for smb command 0x%x",
			cifs_pdu->Command);
L
Linus Torvalds 已提交
240 241 242

	/* save off the origiginal signature so we can modify the smb and check
		its signature against what the server sent */
243
	memcpy(server_response_sig, cifs_pdu->Signature.SecuritySignature, 8);
L
Linus Torvalds 已提交
244

245 246
	cifs_pdu->Signature.Sequence.SequenceNumber =
					cpu_to_le32(expected_sequence_number);
L
Linus Torvalds 已提交
247 248
	cifs_pdu->Signature.Sequence.Reserved = 0;

249
	mutex_lock(&server->srv_mutex);
250
	rc = cifs_calculate_signature(cifs_pdu, server,
L
Linus Torvalds 已提交
251
		what_we_think_sig_should_be);
252
	mutex_unlock(&server->srv_mutex);
L
Linus Torvalds 已提交
253

254
	if (rc)
L
Linus Torvalds 已提交
255 256
		return rc;

257 258
/*	cifs_dump_mem("what we think it should be: ",
		      what_we_think_sig_should_be, 16); */
L
Linus Torvalds 已提交
259

260
	if (memcmp(server_response_sig, what_we_think_sig_should_be, 8))
L
Linus Torvalds 已提交
261 262 263 264 265 266
		return -EACCES;
	else
		return 0;

}

267
/* first calculate 24 bytes ntlm response and then 16 byte session key */
268
int setup_ntlm_response(struct cifs_ses *ses)
L
Linus Torvalds 已提交
269
{
270
	int rc = 0;
271 272 273 274
	unsigned int temp_len = CIFS_SESS_KEY_SIZE + CIFS_AUTH_RESP_SIZE;
	char temp_key[CIFS_SESS_KEY_SIZE];

	if (!ses)
L
Linus Torvalds 已提交
275 276
		return -EINVAL;

277 278 279 280 281 282 283
	ses->auth_key.response = kmalloc(temp_len, GFP_KERNEL);
	if (!ses->auth_key.response) {
		cERROR(1, "NTLM can't allocate (%u bytes) memory", temp_len);
		return -ENOMEM;
	}
	ses->auth_key.len = temp_len;

284
	rc = SMBNTencrypt(ses->password, ses->server->cryptkey,
285
			ses->auth_key.response + CIFS_SESS_KEY_SIZE);
286 287 288 289 290 291 292 293 294 295 296
	if (rc) {
		cFYI(1, "%s Can't generate NTLM response, error: %d",
			__func__, rc);
		return rc;
	}

	rc = E_md4hash(ses->password, temp_key);
	if (rc) {
		cFYI(1, "%s Can't generate NT hash, error: %d", __func__, rc);
		return rc;
	}
297

298 299 300 301
	rc = mdfour(ses->auth_key.response, temp_key, CIFS_SESS_KEY_SIZE);
	if (rc)
		cFYI(1, "%s Can't generate NTLM session key, error: %d",
			__func__, rc);
302

303
	return rc;
L
Linus Torvalds 已提交
304 305
}

306
#ifdef CONFIG_CIFS_WEAK_PW_HASH
307
int calc_lanman_hash(const char *password, const char *cryptkey, bool encrypt,
308
			char *lnm_session_key)
309 310
{
	int i;
311
	int rc;
312 313 314
	char password_with_pad[CIFS_ENCPWD_SIZE];

	memset(password_with_pad, 0, CIFS_ENCPWD_SIZE);
315 316 317
	if (password)
		strncpy(password_with_pad, password, CIFS_ENCPWD_SIZE);

318
	if (!encrypt && global_secflags & CIFSSEC_MAY_PLNTXT) {
319 320 321
		memset(lnm_session_key, 0, CIFS_SESS_KEY_SIZE);
		memcpy(lnm_session_key, password_with_pad,
			CIFS_ENCPWD_SIZE);
322
		return 0;
323
	}
324

325 326 327 328 329 330 331 332 333 334 335
	/* calculate old style session key */
	/* calling toupper is less broken than repeatedly
	calling nls_toupper would be since that will never
	work for UTF8, but neither handles multibyte code pages
	but the only alternative would be converting to UCS-16 (Unicode)
	(using a routine something like UniStrupr) then
	uppercasing and then converting back from Unicode - which
	would only worth doing it if we knew it were utf8. Basically
	utf8 and other multibyte codepages each need their own strupper
	function since a byte at a time will ont work. */

336
	for (i = 0; i < CIFS_ENCPWD_SIZE; i++)
337 338
		password_with_pad[i] = toupper(password_with_pad[i]);

339
	rc = SMBencrypt(password_with_pad, cryptkey, lnm_session_key);
340

341
	return rc;
342 343 344
}
#endif /* CIFS_WEAK_PW_HASH */

345 346 347 348
/* Build a proper attribute value/target info pairs blob.
 * Fill in netbios and dns domain name and workstation name
 * and client time (total five av pairs and + one end of fields indicator.
 * Allocate domain name which gets freed when session struct is deallocated.
349 350
 */
static int
351
build_avpair_blob(struct cifs_ses *ses, const struct nls_table *nls_cp)
352
{
353
	unsigned int dlen;
354
	unsigned int size = 2 * sizeof(struct ntlmssp2_name);
355 356
	char *defdmname = "WORKGROUP";
	unsigned char *blobptr;
357 358
	struct ntlmssp2_name *attrptr;

359 360 361 362 363 364 365 366
	if (!ses->domainName) {
		ses->domainName = kstrdup(defdmname, GFP_KERNEL);
		if (!ses->domainName)
			return -ENOMEM;
	}

	dlen = strlen(ses->domainName);

367 368 369 370 371
	/*
	 * The length of this blob is two times the size of a
	 * structure (av pair) which holds name/size
	 * ( for NTLMSSP_AV_NB_DOMAIN_NAME followed by NTLMSSP_AV_EOL ) +
	 * unicode length of a netbios domain name
372
	 */
373
	ses->auth_key.len = size + 2 * dlen;
374 375 376
	ses->auth_key.response = kzalloc(ses->auth_key.len, GFP_KERNEL);
	if (!ses->auth_key.response) {
		ses->auth_key.len = 0;
377 378 379
		cERROR(1, "Challenge target info allocation failure");
		return -ENOMEM;
	}
380

381
	blobptr = ses->auth_key.response;
382 383
	attrptr = (struct ntlmssp2_name *) blobptr;

384 385 386 387
	/*
	 * As defined in MS-NTLM 3.3.2, just this av pair field
	 * is sufficient as part of the temp
	 */
388 389 390 391 392
	attrptr->type = cpu_to_le16(NTLMSSP_AV_NB_DOMAIN_NAME);
	attrptr->length = cpu_to_le16(2 * dlen);
	blobptr = (unsigned char *)attrptr + sizeof(struct ntlmssp2_name);
	cifs_strtoUCS((__le16 *)blobptr, ses->domainName, dlen, nls_cp);

393 394 395 396 397 398 399 400 401 402 403 404 405 406
	return 0;
}

/* Server has provided av pairs/target info in the type 2 challenge
 * packet and we have plucked it and stored within smb session.
 * We parse that blob here to find netbios domain name to be used
 * as part of ntlmv2 authentication (in Target String), if not already
 * specified on the command line.
 * If this function returns without any error but without fetching
 * domain name, authentication may fail against some server but
 * may not fail against other (those who are not very particular
 * about target string i.e. for some, just user name might suffice.
 */
static int
407
find_domain_name(struct cifs_ses *ses, const struct nls_table *nls_cp)
408 409 410 411 412 413 414 415
{
	unsigned int attrsize;
	unsigned int type;
	unsigned int onesize = sizeof(struct ntlmssp2_name);
	unsigned char *blobptr;
	unsigned char *blobend;
	struct ntlmssp2_name *attrptr;

416
	if (!ses->auth_key.len || !ses->auth_key.response)
417 418
		return 0;

419 420
	blobptr = ses->auth_key.response;
	blobend = blobptr + ses->auth_key.len;
421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441

	while (blobptr + onesize < blobend) {
		attrptr = (struct ntlmssp2_name *) blobptr;
		type = le16_to_cpu(attrptr->type);
		if (type == NTLMSSP_AV_EOL)
			break;
		blobptr += 2; /* advance attr type */
		attrsize = le16_to_cpu(attrptr->length);
		blobptr += 2; /* advance attr size */
		if (blobptr + attrsize > blobend)
			break;
		if (type == NTLMSSP_AV_NB_DOMAIN_NAME) {
			if (!attrsize)
				break;
			if (!ses->domainName) {
				ses->domainName =
					kmalloc(attrsize + 1, GFP_KERNEL);
				if (!ses->domainName)
						return -ENOMEM;
				cifs_from_ucs2(ses->domainName,
					(__le16 *)blobptr, attrsize, attrsize,
442
					nls_cp, false);
443 444 445 446 447 448 449 450 451
				break;
			}
		}
		blobptr += attrsize; /* advance attr  value */
	}

	return 0;
}

452
static int calc_ntlmv2_hash(struct cifs_ses *ses, char *ntlmv2_hash,
453
			    const struct nls_table *nls_cp)
S
Steve French 已提交
454 455 456
{
	int rc = 0;
	int len;
457
	char nt_hash[CIFS_NTHASH_SIZE];
458 459
	wchar_t *user;
	wchar_t *domain;
460
	wchar_t *server;
S
Steve French 已提交
461

462 463 464 465
	if (!ses->server->secmech.sdeschmacmd5) {
		cERROR(1, "calc_ntlmv2_hash: can't generate ntlmv2 hash\n");
		return -1;
	}
466

467 468
	/* calculate md4 hash of password */
	E_md4hash(ses->password, nt_hash);
469

470
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5, nt_hash,
471
				CIFS_NTHASH_SIZE);
472 473 474 475
	if (rc) {
		cERROR(1, "%s: Could not set NT Hash as a key", __func__);
		return rc;
	}
476 477 478 479 480 481

	rc = crypto_shash_init(&ses->server->secmech.sdeschmacmd5->shash);
	if (rc) {
		cERROR(1, "calc_ntlmv2_hash: could not init hmacmd5\n");
		return rc;
	}
S
Steve French 已提交
482

483 484
	/* convert ses->user_name to unicode and uppercase */
	len = strlen(ses->user_name);
S
Steve French 已提交
485
	user = kmalloc(2 + (len * 2), GFP_KERNEL);
486 487 488
	if (user == NULL) {
		cERROR(1, "calc_ntlmv2_hash: user mem alloc failure\n");
		rc = -ENOMEM;
489
		return rc;
490
	}
491
	len = cifs_strtoUCS((__le16 *)user, ses->user_name, len, nls_cp);
S
Steve French 已提交
492
	UniStrupr(user);
493

494
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
495
				(char *)user, 2 * len);
496 497 498 499 500
	kfree(user);
	if (rc) {
		cERROR(1, "%s: Could not update with user\n", __func__);
		return rc;
	}
S
Steve French 已提交
501 502

	/* convert ses->domainName to unicode and uppercase */
503
	if (ses->domainName) {
S
Steve French 已提交
504
		len = strlen(ses->domainName);
S
Steve French 已提交
505

506
		domain = kmalloc(2 + (len * 2), GFP_KERNEL);
507 508 509
		if (domain == NULL) {
			cERROR(1, "calc_ntlmv2_hash: domain mem alloc failure");
			rc = -ENOMEM;
510
			return rc;
511
		}
512 513
		len = cifs_strtoUCS((__le16 *)domain, ses->domainName, len,
					nls_cp);
514
		rc =
515 516
		crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
					(char *)domain, 2 * len);
S
Steve French 已提交
517
		kfree(domain);
518 519 520 521 522
		if (rc) {
			cERROR(1, "%s: Could not update with domain\n",
								__func__);
			return rc;
		}
523 524 525 526 527 528 529
	} else if (ses->serverName) {
		len = strlen(ses->serverName);

		server = kmalloc(2 + (len * 2), GFP_KERNEL);
		if (server == NULL) {
			cERROR(1, "calc_ntlmv2_hash: server mem alloc failure");
			rc = -ENOMEM;
530
			return rc;
531 532 533
		}
		len = cifs_strtoUCS((__le16 *)server, ses->serverName, len,
					nls_cp);
534
		rc =
535 536 537
		crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
					(char *)server, 2 * len);
		kfree(server);
538 539 540 541 542
		if (rc) {
			cERROR(1, "%s: Could not update with server\n",
								__func__);
			return rc;
		}
S
Steve French 已提交
543
	}
544 545

	rc = crypto_shash_final(&ses->server->secmech.sdeschmacmd5->shash,
546
					ntlmv2_hash);
547 548
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
549 550 551 552 553

	return rc;
}

static int
554
CalcNTLMv2_response(const struct cifs_ses *ses, char *ntlmv2_hash)
555 556 557 558 559 560 561 562 563
{
	int rc;
	unsigned int offset = CIFS_SESS_KEY_SIZE + 8;

	if (!ses->server->secmech.sdeschmacmd5) {
		cERROR(1, "calc_ntlmv2_hash: can't generate ntlmv2 hash\n");
		return -1;
	}

564
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5,
565
				ntlmv2_hash, CIFS_HMAC_MD5_HASH_SIZE);
566 567 568 569
	if (rc) {
		cERROR(1, "%s: Could not set NTLMV2 Hash as a key", __func__);
		return rc;
	}
570 571 572 573 574 575 576

	rc = crypto_shash_init(&ses->server->secmech.sdeschmacmd5->shash);
	if (rc) {
		cERROR(1, "CalcNTLMv2_response: could not init hmacmd5");
		return rc;
	}

577 578
	if (ses->server->secType == RawNTLMSSP)
		memcpy(ses->auth_key.response + offset,
579
			ses->ntlmssp->cryptkey, CIFS_SERVER_CHALLENGE_SIZE);
580 581 582
	else
		memcpy(ses->auth_key.response + offset,
			ses->server->cryptkey, CIFS_SERVER_CHALLENGE_SIZE);
583
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
584
		ses->auth_key.response + offset, ses->auth_key.len - offset);
585 586 587 588
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		return rc;
	}
589 590 591

	rc = crypto_shash_final(&ses->server->secmech.sdeschmacmd5->shash,
		ses->auth_key.response + CIFS_SESS_KEY_SIZE);
592 593
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
594 595 596 597

	return rc;
}

598

599
int
600
setup_ntlmv2_rsp(struct cifs_ses *ses, const struct nls_table *nls_cp)
601
{
602
	int rc;
603
	int baselen;
604
	unsigned int tilen;
605
	struct ntlmv2_resp *buf;
606 607
	char ntlmv2_hash[16];
	unsigned char *tiblob = NULL; /* target info blob */
S
Steve French 已提交
608

609 610
	if (ses->server->secType == RawNTLMSSP) {
		if (!ses->domainName) {
611
			rc = find_domain_name(ses, nls_cp);
612 613 614 615 616 617
			if (rc) {
				cERROR(1, "error %d finding domain name", rc);
				goto setup_ntlmv2_rsp_ret;
			}
		}
	} else {
618
		rc = build_avpair_blob(ses, nls_cp);
619 620
		if (rc) {
			cERROR(1, "error %d building av pair blob", rc);
621
			goto setup_ntlmv2_rsp_ret;
622 623
		}
	}
S
Steve French 已提交
624

625
	baselen = CIFS_SESS_KEY_SIZE + sizeof(struct ntlmv2_resp);
626 627 628 629
	tilen = ses->auth_key.len;
	tiblob = ses->auth_key.response;

	ses->auth_key.response = kmalloc(baselen + tilen, GFP_KERNEL);
630 631
	if (!ses->auth_key.response) {
		rc = ENOMEM;
632
		ses->auth_key.len = 0;
633 634 635
		cERROR(1, "%s: Can't allocate auth blob", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
636
	ses->auth_key.len += baselen;
637 638 639 640 641 642 643 644 645

	buf = (struct ntlmv2_resp *)
			(ses->auth_key.response + CIFS_SESS_KEY_SIZE);
	buf->blob_signature = cpu_to_le32(0x00000101);
	buf->reserved = 0;
	buf->time = cpu_to_le64(cifs_UnixTimeToNT(CURRENT_TIME));
	get_random_bytes(&buf->client_chal, sizeof(buf->client_chal));
	buf->reserved2 = 0;

646
	memcpy(ses->auth_key.response + baselen, tiblob, tilen);
647

648
	/* calculate ntlmv2_hash */
649
	rc = calc_ntlmv2_hash(ses, ntlmv2_hash, nls_cp);
650
	if (rc) {
651
		cERROR(1, "could not get v2 hash rc %d", rc);
652 653
		goto setup_ntlmv2_rsp_ret;
	}
654 655

	/* calculate first part of the client response (CR1) */
656
	rc = CalcNTLMv2_response(ses, ntlmv2_hash);
657 658 659 660
	if (rc) {
		cERROR(1, "Could not calculate CR1  rc: %d", rc);
		goto setup_ntlmv2_rsp_ret;
	}
661

662
	/* now calculate the session key for NTLMv2 */
663
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5,
664
		ntlmv2_hash, CIFS_HMAC_MD5_HASH_SIZE);
665 666 667 668
	if (rc) {
		cERROR(1, "%s: Could not set NTLMV2 Hash as a key", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
669 670 671 672 673 674 675

	rc = crypto_shash_init(&ses->server->secmech.sdeschmacmd5->shash);
	if (rc) {
		cERROR(1, "%s: Could not init hmacmd5\n", __func__);
		goto setup_ntlmv2_rsp_ret;
	}

676
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
677 678
		ses->auth_key.response + CIFS_SESS_KEY_SIZE,
		CIFS_HMAC_MD5_HASH_SIZE);
679 680 681 682
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
683 684 685

	rc = crypto_shash_final(&ses->server->secmech.sdeschmacmd5->shash,
		ses->auth_key.response);
686 687
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
688 689

setup_ntlmv2_rsp_ret:
690
	kfree(tiblob);
691 692

	return rc;
S
Steve French 已提交
693 694
}

695
int
696
calc_seckey(struct cifs_ses *ses)
697 698 699 700 701 702 703 704 705 706
{
	int rc;
	struct crypto_blkcipher *tfm_arc4;
	struct scatterlist sgin, sgout;
	struct blkcipher_desc desc;
	unsigned char sec_key[CIFS_SESS_KEY_SIZE]; /* a nonce */

	get_random_bytes(sec_key, CIFS_SESS_KEY_SIZE);

	tfm_arc4 = crypto_alloc_blkcipher("ecb(arc4)", 0, CRYPTO_ALG_ASYNC);
707 708
	if (IS_ERR(tfm_arc4)) {
		rc = PTR_ERR(tfm_arc4);
709
		cERROR(1, "could not allocate crypto API arc4\n");
710
		return rc;
711 712 713 714
	}

	desc.tfm = tfm_arc4;

715
	rc = crypto_blkcipher_setkey(tfm_arc4, ses->auth_key.response,
716
					CIFS_SESS_KEY_SIZE);
717 718 719 720
	if (rc) {
		cERROR(1, "%s: Could not set response as a key", __func__);
		return rc;
	}
721 722

	sg_init_one(&sgin, sec_key, CIFS_SESS_KEY_SIZE);
723
	sg_init_one(&sgout, ses->ntlmssp->ciphertext, CIFS_CPHTXT_SIZE);
724 725 726 727 728 729 730 731 732 733 734 735 736 737 738

	rc = crypto_blkcipher_encrypt(&desc, &sgout, &sgin, CIFS_CPHTXT_SIZE);
	if (rc) {
		cERROR(1, "could not encrypt session key rc: %d\n", rc);
		crypto_free_blkcipher(tfm_arc4);
		return rc;
	}

	/* make secondary_key/nonce as session key */
	memcpy(ses->auth_key.response, sec_key, CIFS_SESS_KEY_SIZE);
	/* and make len as that of session key only */
	ses->auth_key.len = CIFS_SESS_KEY_SIZE;

	crypto_free_blkcipher(tfm_arc4);

739
	return rc;
740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762
}

void
cifs_crypto_shash_release(struct TCP_Server_Info *server)
{
	if (server->secmech.md5)
		crypto_free_shash(server->secmech.md5);

	if (server->secmech.hmacmd5)
		crypto_free_shash(server->secmech.hmacmd5);

	kfree(server->secmech.sdeschmacmd5);

	kfree(server->secmech.sdescmd5);
}

int
cifs_crypto_shash_allocate(struct TCP_Server_Info *server)
{
	int rc;
	unsigned int size;

	server->secmech.hmacmd5 = crypto_alloc_shash("hmac(md5)", 0, 0);
763
	if (IS_ERR(server->secmech.hmacmd5)) {
764 765 766 767 768
		cERROR(1, "could not allocate crypto hmacmd5\n");
		return PTR_ERR(server->secmech.hmacmd5);
	}

	server->secmech.md5 = crypto_alloc_shash("md5", 0, 0);
769
	if (IS_ERR(server->secmech.md5)) {
770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810
		cERROR(1, "could not allocate crypto md5\n");
		rc = PTR_ERR(server->secmech.md5);
		goto crypto_allocate_md5_fail;
	}

	size = sizeof(struct shash_desc) +
			crypto_shash_descsize(server->secmech.hmacmd5);
	server->secmech.sdeschmacmd5 = kmalloc(size, GFP_KERNEL);
	if (!server->secmech.sdeschmacmd5) {
		cERROR(1, "cifs_crypto_shash_allocate: can't alloc hmacmd5\n");
		rc = -ENOMEM;
		goto crypto_allocate_hmacmd5_sdesc_fail;
	}
	server->secmech.sdeschmacmd5->shash.tfm = server->secmech.hmacmd5;
	server->secmech.sdeschmacmd5->shash.flags = 0x0;


	size = sizeof(struct shash_desc) +
			crypto_shash_descsize(server->secmech.md5);
	server->secmech.sdescmd5 = kmalloc(size, GFP_KERNEL);
	if (!server->secmech.sdescmd5) {
		cERROR(1, "cifs_crypto_shash_allocate: can't alloc md5\n");
		rc = -ENOMEM;
		goto crypto_allocate_md5_sdesc_fail;
	}
	server->secmech.sdescmd5->shash.tfm = server->secmech.md5;
	server->secmech.sdescmd5->shash.flags = 0x0;

	return 0;

crypto_allocate_md5_sdesc_fail:
	kfree(server->secmech.sdeschmacmd5);

crypto_allocate_hmacmd5_sdesc_fail:
	crypto_free_shash(server->secmech.md5);

crypto_allocate_md5_fail:
	crypto_free_shash(server->secmech.hmacmd5);

	return rc;
}