cifsencrypt.c 23.4 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 95 96 97 98
	if (!server->session_estab) {
		strncpy(cifs_pdu->Signature.SecuritySignature, "BSRSPYL", 8);
		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 192 193 194 195
	if (!server->session_estab) {
		strncpy(cifs_pdu->Signature.SecuritySignature, "BSRSPYL", 8);
		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 354 355 356 357 358
	unsigned int dlen;
	unsigned int wlen;
	unsigned int size = 6 * sizeof(struct ntlmssp2_name);
	__le64  curtime;
	char *defdmname = "WORKGROUP";
	unsigned char *blobptr;
359 360
	struct ntlmssp2_name *attrptr;

361 362 363 364 365 366 367 368 369 370 371 372 373 374 375
	if (!ses->domainName) {
		ses->domainName = kstrdup(defdmname, GFP_KERNEL);
		if (!ses->domainName)
			return -ENOMEM;
	}

	dlen = strlen(ses->domainName);
	wlen = strlen(ses->server->hostname);

	/* The length of this blob is a size which is
	 * six times the size of a structure which holds name/size +
	 * two times the unicode length of a domain name +
	 * two times the unicode length of a server name +
	 * size of a timestamp (which is 8 bytes).
	 */
376 377 378 379
	ses->auth_key.len = size + 2 * (2 * dlen) + 2 * (2 * wlen) + 8;
	ses->auth_key.response = kzalloc(ses->auth_key.len, GFP_KERNEL);
	if (!ses->auth_key.response) {
		ses->auth_key.len = 0;
380 381 382
		cERROR(1, "Challenge target info allocation failure");
		return -ENOMEM;
	}
383

384
	blobptr = ses->auth_key.response;
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423
	attrptr = (struct ntlmssp2_name *) blobptr;

	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);

	blobptr += 2 * dlen;
	attrptr = (struct ntlmssp2_name *) blobptr;

	attrptr->type = cpu_to_le16(NTLMSSP_AV_NB_COMPUTER_NAME);
	attrptr->length = cpu_to_le16(2 * wlen);
	blobptr = (unsigned char *)attrptr + sizeof(struct ntlmssp2_name);
	cifs_strtoUCS((__le16 *)blobptr, ses->server->hostname, wlen, nls_cp);

	blobptr += 2 * wlen;
	attrptr = (struct ntlmssp2_name *) blobptr;

	attrptr->type = cpu_to_le16(NTLMSSP_AV_DNS_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);

	blobptr += 2 * dlen;
	attrptr = (struct ntlmssp2_name *) blobptr;

	attrptr->type = cpu_to_le16(NTLMSSP_AV_DNS_COMPUTER_NAME);
	attrptr->length = cpu_to_le16(2 * wlen);
	blobptr = (unsigned char *)attrptr + sizeof(struct ntlmssp2_name);
	cifs_strtoUCS((__le16 *)blobptr, ses->server->hostname, wlen, nls_cp);

	blobptr += 2 * wlen;
	attrptr = (struct ntlmssp2_name *) blobptr;

	attrptr->type = cpu_to_le16(NTLMSSP_AV_TIMESTAMP);
	attrptr->length = cpu_to_le16(sizeof(__le64));
	blobptr = (unsigned char *)attrptr + sizeof(struct ntlmssp2_name);
	curtime = cpu_to_le64(cifs_UnixTimeToNT(CURRENT_TIME));
	memcpy(blobptr, &curtime, sizeof(__le64));
424 425 426 427 428 429 430 431 432 433 434 435 436 437 438

	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
439
find_domain_name(struct cifs_ses *ses, const struct nls_table *nls_cp)
440 441 442 443 444 445 446 447
{
	unsigned int attrsize;
	unsigned int type;
	unsigned int onesize = sizeof(struct ntlmssp2_name);
	unsigned char *blobptr;
	unsigned char *blobend;
	struct ntlmssp2_name *attrptr;

448
	if (!ses->auth_key.len || !ses->auth_key.response)
449 450
		return 0;

451 452
	blobptr = ses->auth_key.response;
	blobend = blobptr + ses->auth_key.len;
453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473

	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,
474
					nls_cp, false);
475 476 477 478 479 480 481 482 483
				break;
			}
		}
		blobptr += attrsize; /* advance attr  value */
	}

	return 0;
}

484
static int calc_ntlmv2_hash(struct cifs_ses *ses, char *ntlmv2_hash,
485
			    const struct nls_table *nls_cp)
S
Steve French 已提交
486 487 488
{
	int rc = 0;
	int len;
489
	char nt_hash[CIFS_NTHASH_SIZE];
490 491
	wchar_t *user;
	wchar_t *domain;
492
	wchar_t *server;
S
Steve French 已提交
493

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

499 500
	/* calculate md4 hash of password */
	E_md4hash(ses->password, nt_hash);
501

502
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5, nt_hash,
503
				CIFS_NTHASH_SIZE);
504 505 506 507
	if (rc) {
		cERROR(1, "%s: Could not set NT Hash as a key", __func__);
		return rc;
	}
508 509 510 511 512 513

	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 已提交
514

515 516
	/* convert ses->user_name to unicode and uppercase */
	len = strlen(ses->user_name);
S
Steve French 已提交
517
	user = kmalloc(2 + (len * 2), GFP_KERNEL);
518 519 520
	if (user == NULL) {
		cERROR(1, "calc_ntlmv2_hash: user mem alloc failure\n");
		rc = -ENOMEM;
521
		return rc;
522
	}
523
	len = cifs_strtoUCS((__le16 *)user, ses->user_name, len, nls_cp);
S
Steve French 已提交
524
	UniStrupr(user);
525

526
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
527
				(char *)user, 2 * len);
528 529 530 531 532
	kfree(user);
	if (rc) {
		cERROR(1, "%s: Could not update with user\n", __func__);
		return rc;
	}
S
Steve French 已提交
533 534

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

538
		domain = kmalloc(2 + (len * 2), GFP_KERNEL);
539 540 541
		if (domain == NULL) {
			cERROR(1, "calc_ntlmv2_hash: domain mem alloc failure");
			rc = -ENOMEM;
542
			return rc;
543
		}
544 545
		len = cifs_strtoUCS((__le16 *)domain, ses->domainName, len,
					nls_cp);
546
		rc =
547 548
		crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
					(char *)domain, 2 * len);
S
Steve French 已提交
549
		kfree(domain);
550 551 552 553 554
		if (rc) {
			cERROR(1, "%s: Could not update with domain\n",
								__func__);
			return rc;
		}
555 556 557 558 559 560 561
	} 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;
562
			return rc;
563 564 565
		}
		len = cifs_strtoUCS((__le16 *)server, ses->serverName, len,
					nls_cp);
566
		rc =
567 568 569
		crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
					(char *)server, 2 * len);
		kfree(server);
570 571 572 573 574
		if (rc) {
			cERROR(1, "%s: Could not update with server\n",
								__func__);
			return rc;
		}
S
Steve French 已提交
575
	}
576 577

	rc = crypto_shash_final(&ses->server->secmech.sdeschmacmd5->shash,
578
					ntlmv2_hash);
579 580
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
581 582 583 584 585

	return rc;
}

static int
586
CalcNTLMv2_response(const struct cifs_ses *ses, char *ntlmv2_hash)
587 588 589 590 591 592 593 594 595
{
	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;
	}

596
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5,
597
				ntlmv2_hash, CIFS_HMAC_MD5_HASH_SIZE);
598 599 600 601
	if (rc) {
		cERROR(1, "%s: Could not set NTLMV2 Hash as a key", __func__);
		return rc;
	}
602 603 604 605 606 607 608

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

609 610
	if (ses->server->secType == RawNTLMSSP)
		memcpy(ses->auth_key.response + offset,
611
			ses->ntlmssp->cryptkey, CIFS_SERVER_CHALLENGE_SIZE);
612 613 614
	else
		memcpy(ses->auth_key.response + offset,
			ses->server->cryptkey, CIFS_SERVER_CHALLENGE_SIZE);
615
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
616
		ses->auth_key.response + offset, ses->auth_key.len - offset);
617 618 619 620
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		return rc;
	}
621 622 623

	rc = crypto_shash_final(&ses->server->secmech.sdeschmacmd5->shash,
		ses->auth_key.response + CIFS_SESS_KEY_SIZE);
624 625
	if (rc)
		cERROR(1, "%s: Could not generate md5 hash\n", __func__);
626 627 628 629

	return rc;
}

630

631
int
632
setup_ntlmv2_rsp(struct cifs_ses *ses, const struct nls_table *nls_cp)
633
{
634
	int rc;
635
	int baselen;
636
	unsigned int tilen;
637
	struct ntlmv2_resp *buf;
638 639
	char ntlmv2_hash[16];
	unsigned char *tiblob = NULL; /* target info blob */
S
Steve French 已提交
640

641 642
	if (ses->server->secType == RawNTLMSSP) {
		if (!ses->domainName) {
643
			rc = find_domain_name(ses, nls_cp);
644 645 646 647 648 649
			if (rc) {
				cERROR(1, "error %d finding domain name", rc);
				goto setup_ntlmv2_rsp_ret;
			}
		}
	} else {
650
		rc = build_avpair_blob(ses, nls_cp);
651 652
		if (rc) {
			cERROR(1, "error %d building av pair blob", rc);
653
			goto setup_ntlmv2_rsp_ret;
654 655
		}
	}
S
Steve French 已提交
656

657
	baselen = CIFS_SESS_KEY_SIZE + sizeof(struct ntlmv2_resp);
658 659 660 661
	tilen = ses->auth_key.len;
	tiblob = ses->auth_key.response;

	ses->auth_key.response = kmalloc(baselen + tilen, GFP_KERNEL);
662 663
	if (!ses->auth_key.response) {
		rc = ENOMEM;
664
		ses->auth_key.len = 0;
665 666 667
		cERROR(1, "%s: Can't allocate auth blob", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
668
	ses->auth_key.len += baselen;
669 670 671 672 673 674 675 676 677

	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;

678
	memcpy(ses->auth_key.response + baselen, tiblob, tilen);
679

680
	/* calculate ntlmv2_hash */
681
	rc = calc_ntlmv2_hash(ses, ntlmv2_hash, nls_cp);
682
	if (rc) {
683
		cERROR(1, "could not get v2 hash rc %d", rc);
684 685
		goto setup_ntlmv2_rsp_ret;
	}
686 687

	/* calculate first part of the client response (CR1) */
688
	rc = CalcNTLMv2_response(ses, ntlmv2_hash);
689 690 691 692
	if (rc) {
		cERROR(1, "Could not calculate CR1  rc: %d", rc);
		goto setup_ntlmv2_rsp_ret;
	}
693

694
	/* now calculate the session key for NTLMv2 */
695
	rc = crypto_shash_setkey(ses->server->secmech.hmacmd5,
696
		ntlmv2_hash, CIFS_HMAC_MD5_HASH_SIZE);
697 698 699 700
	if (rc) {
		cERROR(1, "%s: Could not set NTLMV2 Hash as a key", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
701 702 703 704 705 706 707

	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;
	}

708
	rc = crypto_shash_update(&ses->server->secmech.sdeschmacmd5->shash,
709 710
		ses->auth_key.response + CIFS_SESS_KEY_SIZE,
		CIFS_HMAC_MD5_HASH_SIZE);
711 712 713 714
	if (rc) {
		cERROR(1, "%s: Could not update with response\n", __func__);
		goto setup_ntlmv2_rsp_ret;
	}
715 716 717

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

setup_ntlmv2_rsp_ret:
722
	kfree(tiblob);
723 724

	return rc;
S
Steve French 已提交
725 726
}

727
int
728
calc_seckey(struct cifs_ses *ses)
729 730 731 732 733 734 735 736 737 738
{
	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);
739 740
	if (IS_ERR(tfm_arc4)) {
		rc = PTR_ERR(tfm_arc4);
741
		cERROR(1, "could not allocate crypto API arc4\n");
742
		return rc;
743 744 745 746
	}

	desc.tfm = tfm_arc4;

747
	rc = crypto_blkcipher_setkey(tfm_arc4, ses->auth_key.response,
748
					CIFS_SESS_KEY_SIZE);
749 750 751 752
	if (rc) {
		cERROR(1, "%s: Could not set response as a key", __func__);
		return rc;
	}
753 754

	sg_init_one(&sgin, sec_key, CIFS_SESS_KEY_SIZE);
755
	sg_init_one(&sgout, ses->ntlmssp->ciphertext, CIFS_CPHTXT_SIZE);
756 757 758 759 760 761 762 763 764 765 766 767 768 769 770

	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);

771
	return rc;
772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794
}

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);
795
	if (IS_ERR(server->secmech.hmacmd5)) {
796 797 798 799 800
		cERROR(1, "could not allocate crypto hmacmd5\n");
		return PTR_ERR(server->secmech.hmacmd5);
	}

	server->secmech.md5 = crypto_alloc_shash("md5", 0, 0);
801
	if (IS_ERR(server->secmech.md5)) {
802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842
		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;
}