eap_mschapv2.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895
  1. /*
  2. * EAP peer method: EAP-MSCHAPV2 (draft-kamath-pppext-eap-mschapv2-00.txt)
  3. * Copyright (c) 2004-2008, Jouni Malinen <j@w1.fi>
  4. *
  5. * This software may be distributed under the terms of the BSD license.
  6. * See README for more details.
  7. *
  8. * This file implements EAP peer part of EAP-MSCHAPV2 method (EAP type 26).
  9. * draft-kamath-pppext-eap-mschapv2-00.txt defines the Microsoft EAP CHAP
  10. * Extensions Protocol, Version 2, for mutual authentication and key
  11. * derivation. This encapsulates MS-CHAP-v2 protocol which is defined in
  12. * RFC 2759. Use of EAP-MSCHAPV2 derived keys with MPPE cipher is described in
  13. * RFC 3079.
  14. */
  15. #include "includes.h"
  16. #include "common.h"
  17. #include "crypto/ms_funcs.h"
  18. #include "crypto/random.h"
  19. #include "common/wpa_ctrl.h"
  20. #include "mschapv2.h"
  21. #include "eap_i.h"
  22. #include "eap_config.h"
  23. #ifdef _MSC_VER
  24. #pragma pack(push, 1)
  25. #endif /* _MSC_VER */
  26. struct eap_mschapv2_hdr {
  27. u8 op_code; /* MSCHAPV2_OP_* */
  28. u8 mschapv2_id; /* usually same as EAP identifier; must be changed
  29. * for challenges, but not for success/failure */
  30. u8 ms_length[2]; /* Note: misaligned; length - 5 */
  31. /* followed by data */
  32. } STRUCT_PACKED;
  33. /* Response Data field */
  34. struct ms_response {
  35. u8 peer_challenge[MSCHAPV2_CHAL_LEN];
  36. u8 reserved[8];
  37. u8 nt_response[MSCHAPV2_NT_RESPONSE_LEN];
  38. u8 flags;
  39. } STRUCT_PACKED;
  40. /* Change-Password Data field */
  41. struct ms_change_password {
  42. u8 encr_password[516];
  43. u8 encr_hash[16];
  44. u8 peer_challenge[MSCHAPV2_CHAL_LEN];
  45. u8 reserved[8];
  46. u8 nt_response[MSCHAPV2_NT_RESPONSE_LEN];
  47. u8 flags[2];
  48. } STRUCT_PACKED;
  49. #ifdef _MSC_VER
  50. #pragma pack(pop)
  51. #endif /* _MSC_VER */
  52. #define MSCHAPV2_OP_CHALLENGE 1
  53. #define MSCHAPV2_OP_RESPONSE 2
  54. #define MSCHAPV2_OP_SUCCESS 3
  55. #define MSCHAPV2_OP_FAILURE 4
  56. #define MSCHAPV2_OP_CHANGE_PASSWORD 7
  57. #define ERROR_RESTRICTED_LOGON_HOURS 646
  58. #define ERROR_ACCT_DISABLED 647
  59. #define ERROR_PASSWD_EXPIRED 648
  60. #define ERROR_NO_DIALIN_PERMISSION 649
  61. #define ERROR_AUTHENTICATION_FAILURE 691
  62. #define ERROR_CHANGING_PASSWORD 709
  63. #define PASSWD_CHANGE_CHAL_LEN 16
  64. #define MSCHAPV2_KEY_LEN 16
  65. struct eap_mschapv2_data {
  66. u8 auth_response[MSCHAPV2_AUTH_RESPONSE_LEN];
  67. int auth_response_valid;
  68. int prev_error;
  69. u8 passwd_change_challenge[PASSWD_CHANGE_CHAL_LEN];
  70. int passwd_change_challenge_valid;
  71. int passwd_change_version;
  72. /* Optional challenge values generated in EAP-FAST Phase 1 negotiation
  73. */
  74. u8 *peer_challenge;
  75. u8 *auth_challenge;
  76. int phase2;
  77. u8 master_key[MSCHAPV2_MASTER_KEY_LEN];
  78. int master_key_valid;
  79. int success;
  80. struct wpabuf *prev_challenge;
  81. };
  82. static void eap_mschapv2_deinit(struct eap_sm *sm, void *priv);
  83. static void * eap_mschapv2_init(struct eap_sm *sm)
  84. {
  85. struct eap_mschapv2_data *data;
  86. data = os_zalloc(sizeof(*data));
  87. if (data == NULL)
  88. return NULL;
  89. if (sm->peer_challenge) {
  90. data->peer_challenge = os_memdup(sm->peer_challenge,
  91. MSCHAPV2_CHAL_LEN);
  92. if (data->peer_challenge == NULL) {
  93. eap_mschapv2_deinit(sm, data);
  94. return NULL;
  95. }
  96. }
  97. if (sm->auth_challenge) {
  98. data->auth_challenge = os_memdup(sm->auth_challenge,
  99. MSCHAPV2_CHAL_LEN);
  100. if (data->auth_challenge == NULL) {
  101. eap_mschapv2_deinit(sm, data);
  102. return NULL;
  103. }
  104. }
  105. data->phase2 = sm->init_phase2;
  106. return data;
  107. }
  108. static void eap_mschapv2_deinit(struct eap_sm *sm, void *priv)
  109. {
  110. struct eap_mschapv2_data *data = priv;
  111. os_free(data->peer_challenge);
  112. os_free(data->auth_challenge);
  113. wpabuf_free(data->prev_challenge);
  114. bin_clear_free(data, sizeof(*data));
  115. }
  116. static struct wpabuf * eap_mschapv2_challenge_reply(
  117. struct eap_sm *sm, struct eap_mschapv2_data *data, u8 id,
  118. u8 mschapv2_id, const u8 *auth_challenge)
  119. {
  120. struct wpabuf *resp;
  121. struct eap_mschapv2_hdr *ms;
  122. u8 *peer_challenge;
  123. int ms_len;
  124. struct ms_response *r;
  125. size_t identity_len, password_len;
  126. const u8 *identity, *password;
  127. int pwhash;
  128. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Generating Challenge Response");
  129. identity = eap_get_config_identity(sm, &identity_len);
  130. password = eap_get_config_password2(sm, &password_len, &pwhash);
  131. if (identity == NULL || password == NULL)
  132. return NULL;
  133. ms_len = sizeof(*ms) + 1 + sizeof(*r) + identity_len;
  134. resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2, ms_len,
  135. EAP_CODE_RESPONSE, id);
  136. if (resp == NULL)
  137. return NULL;
  138. ms = wpabuf_put(resp, sizeof(*ms));
  139. ms->op_code = MSCHAPV2_OP_RESPONSE;
  140. ms->mschapv2_id = mschapv2_id;
  141. if (data->prev_error) {
  142. /*
  143. * TODO: this does not seem to be enough when processing two
  144. * or more failure messages. IAS did not increment mschapv2_id
  145. * in its own packets, but it seemed to expect the peer to
  146. * increment this for all packets(?).
  147. */
  148. ms->mschapv2_id++;
  149. }
  150. WPA_PUT_BE16(ms->ms_length, ms_len);
  151. wpabuf_put_u8(resp, sizeof(*r)); /* Value-Size */
  152. /* Response */
  153. r = wpabuf_put(resp, sizeof(*r));
  154. peer_challenge = r->peer_challenge;
  155. if (data->peer_challenge) {
  156. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: peer_challenge generated "
  157. "in Phase 1");
  158. peer_challenge = data->peer_challenge;
  159. os_memset(r->peer_challenge, 0, MSCHAPV2_CHAL_LEN);
  160. } else if (random_get_bytes(peer_challenge, MSCHAPV2_CHAL_LEN)) {
  161. wpabuf_free(resp);
  162. return NULL;
  163. }
  164. os_memset(r->reserved, 0, 8);
  165. if (data->auth_challenge) {
  166. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: auth_challenge generated "
  167. "in Phase 1");
  168. auth_challenge = data->auth_challenge;
  169. }
  170. if (mschapv2_derive_response(identity, identity_len, password,
  171. password_len, pwhash, auth_challenge,
  172. peer_challenge, r->nt_response,
  173. data->auth_response, data->master_key)) {
  174. wpa_printf(MSG_ERROR, "EAP-MSCHAPV2: Failed to derive "
  175. "response");
  176. wpabuf_free(resp);
  177. return NULL;
  178. }
  179. data->auth_response_valid = 1;
  180. data->master_key_valid = 1;
  181. r->flags = 0; /* reserved, must be zero */
  182. wpabuf_put_data(resp, identity, identity_len);
  183. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: TX identifier %d mschapv2_id %d "
  184. "(response)", id, ms->mschapv2_id);
  185. return resp;
  186. }
  187. /**
  188. * eap_mschapv2_process - Process an EAP-MSCHAPv2 challenge message
  189. * @sm: Pointer to EAP state machine allocated with eap_peer_sm_init()
  190. * @data: Pointer to private EAP method data from eap_mschapv2_init()
  191. * @ret: Return values from EAP request validation and processing
  192. * @req: Pointer to EAP-MSCHAPv2 header from the request
  193. * @req_len: Length of the EAP-MSCHAPv2 data
  194. * @id: EAP identifier used in the request
  195. * Returns: Pointer to allocated EAP response packet (eapRespData) or %NULL if
  196. * no reply available
  197. */
  198. static struct wpabuf * eap_mschapv2_challenge(
  199. struct eap_sm *sm, struct eap_mschapv2_data *data,
  200. struct eap_method_ret *ret, const struct eap_mschapv2_hdr *req,
  201. size_t req_len, u8 id)
  202. {
  203. size_t len, challenge_len;
  204. const u8 *pos, *challenge;
  205. if (eap_get_config_identity(sm, &len) == NULL ||
  206. eap_get_config_password(sm, &len) == NULL)
  207. return NULL;
  208. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Received challenge");
  209. if (req_len < sizeof(*req) + 1) {
  210. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Too short challenge data "
  211. "(len %lu)", (unsigned long) req_len);
  212. ret->ignore = TRUE;
  213. return NULL;
  214. }
  215. pos = (const u8 *) (req + 1);
  216. challenge_len = *pos++;
  217. len = req_len - sizeof(*req) - 1;
  218. if (challenge_len != MSCHAPV2_CHAL_LEN) {
  219. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Invalid challenge length "
  220. "%lu", (unsigned long) challenge_len);
  221. ret->ignore = TRUE;
  222. return NULL;
  223. }
  224. if (len < challenge_len) {
  225. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Too short challenge"
  226. " packet: len=%lu challenge_len=%lu",
  227. (unsigned long) len, (unsigned long) challenge_len);
  228. ret->ignore = TRUE;
  229. return NULL;
  230. }
  231. if (data->passwd_change_challenge_valid) {
  232. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Using challenge from the "
  233. "failure message");
  234. challenge = data->passwd_change_challenge;
  235. } else
  236. challenge = pos;
  237. pos += challenge_len;
  238. len -= challenge_len;
  239. wpa_hexdump_ascii(MSG_DEBUG, "EAP-MSCHAPV2: Authentication Servername",
  240. pos, len);
  241. ret->ignore = FALSE;
  242. ret->methodState = METHOD_MAY_CONT;
  243. ret->decision = DECISION_FAIL;
  244. ret->allowNotifications = TRUE;
  245. return eap_mschapv2_challenge_reply(sm, data, id, req->mschapv2_id,
  246. challenge);
  247. }
  248. static void eap_mschapv2_password_changed(struct eap_sm *sm,
  249. struct eap_mschapv2_data *data)
  250. {
  251. struct eap_peer_config *config = eap_get_config(sm);
  252. if (config && config->new_password) {
  253. wpa_msg(sm->msg_ctx, MSG_INFO,
  254. WPA_EVENT_PASSWORD_CHANGED
  255. "EAP-MSCHAPV2: Password changed successfully");
  256. data->prev_error = 0;
  257. bin_clear_free(config->password, config->password_len);
  258. if (config->flags & EAP_CONFIG_FLAGS_EXT_PASSWORD) {
  259. /* TODO: update external storage */
  260. } else if (config->flags & EAP_CONFIG_FLAGS_PASSWORD_NTHASH) {
  261. config->password = os_malloc(16);
  262. config->password_len = 16;
  263. if (config->password &&
  264. nt_password_hash(config->new_password,
  265. config->new_password_len,
  266. config->password)) {
  267. bin_clear_free(config->password,
  268. config->password_len);
  269. config->password = NULL;
  270. config->password_len = 0;
  271. }
  272. bin_clear_free(config->new_password,
  273. config->new_password_len);
  274. } else {
  275. config->password = config->new_password;
  276. config->password_len = config->new_password_len;
  277. }
  278. config->new_password = NULL;
  279. config->new_password_len = 0;
  280. }
  281. }
  282. /**
  283. * eap_mschapv2_process - Process an EAP-MSCHAPv2 success message
  284. * @sm: Pointer to EAP state machine allocated with eap_peer_sm_init()
  285. * @data: Pointer to private EAP method data from eap_mschapv2_init()
  286. * @ret: Return values from EAP request validation and processing
  287. * @req: Pointer to EAP-MSCHAPv2 header from the request
  288. * @req_len: Length of the EAP-MSCHAPv2 data
  289. * @id: EAP identifier used in th erequest
  290. * Returns: Pointer to allocated EAP response packet (eapRespData) or %NULL if
  291. * no reply available
  292. */
  293. static struct wpabuf * eap_mschapv2_success(struct eap_sm *sm,
  294. struct eap_mschapv2_data *data,
  295. struct eap_method_ret *ret,
  296. const struct eap_mschapv2_hdr *req,
  297. size_t req_len, u8 id)
  298. {
  299. struct wpabuf *resp;
  300. const u8 *pos;
  301. size_t len;
  302. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Received success");
  303. len = req_len - sizeof(*req);
  304. pos = (const u8 *) (req + 1);
  305. if (!data->auth_response_valid ||
  306. mschapv2_verify_auth_response(data->auth_response, pos, len)) {
  307. wpa_printf(MSG_WARNING, "EAP-MSCHAPV2: Invalid authenticator "
  308. "response in success request");
  309. ret->methodState = METHOD_DONE;
  310. ret->decision = DECISION_FAIL;
  311. return NULL;
  312. }
  313. pos += 2 + 2 * MSCHAPV2_AUTH_RESPONSE_LEN;
  314. len -= 2 + 2 * MSCHAPV2_AUTH_RESPONSE_LEN;
  315. while (len > 0 && *pos == ' ') {
  316. pos++;
  317. len--;
  318. }
  319. wpa_hexdump_ascii(MSG_DEBUG, "EAP-MSCHAPV2: Success message",
  320. pos, len);
  321. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Authentication succeeded");
  322. /* Note: Only op_code of the EAP-MSCHAPV2 header is included in success
  323. * message. */
  324. resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2, 1,
  325. EAP_CODE_RESPONSE, id);
  326. if (resp == NULL) {
  327. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Failed to allocate "
  328. "buffer for success response");
  329. ret->ignore = TRUE;
  330. return NULL;
  331. }
  332. wpabuf_put_u8(resp, MSCHAPV2_OP_SUCCESS); /* op_code */
  333. ret->methodState = METHOD_DONE;
  334. ret->decision = DECISION_UNCOND_SUCC;
  335. ret->allowNotifications = FALSE;
  336. data->success = 1;
  337. if (data->prev_error == ERROR_PASSWD_EXPIRED)
  338. eap_mschapv2_password_changed(sm, data);
  339. return resp;
  340. }
  341. static int eap_mschapv2_failure_txt(struct eap_sm *sm,
  342. struct eap_mschapv2_data *data, char *txt)
  343. {
  344. char *pos, *msg = "";
  345. int retry = 1;
  346. struct eap_peer_config *config = eap_get_config(sm);
  347. /* For example:
  348. * E=691 R=1 C=<32 octets hex challenge> V=3 M=Authentication Failure
  349. */
  350. pos = txt;
  351. if (pos && os_strncmp(pos, "E=", 2) == 0) {
  352. pos += 2;
  353. data->prev_error = atoi(pos);
  354. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: error %d",
  355. data->prev_error);
  356. pos = os_strchr(pos, ' ');
  357. if (pos)
  358. pos++;
  359. }
  360. if (pos && os_strncmp(pos, "R=", 2) == 0) {
  361. pos += 2;
  362. retry = atoi(pos);
  363. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: retry is %sallowed",
  364. retry == 1 ? "" : "not ");
  365. pos = os_strchr(pos, ' ');
  366. if (pos)
  367. pos++;
  368. }
  369. if (pos && os_strncmp(pos, "C=", 2) == 0) {
  370. int hex_len;
  371. pos += 2;
  372. hex_len = os_strchr(pos, ' ') - (char *) pos;
  373. if (hex_len == PASSWD_CHANGE_CHAL_LEN * 2) {
  374. if (hexstr2bin(pos, data->passwd_change_challenge,
  375. PASSWD_CHANGE_CHAL_LEN)) {
  376. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: invalid "
  377. "failure challenge");
  378. } else {
  379. wpa_hexdump(MSG_DEBUG, "EAP-MSCHAPV2: failure "
  380. "challenge",
  381. data->passwd_change_challenge,
  382. PASSWD_CHANGE_CHAL_LEN);
  383. data->passwd_change_challenge_valid = 1;
  384. }
  385. } else {
  386. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: invalid failure "
  387. "challenge len %d", hex_len);
  388. }
  389. pos = os_strchr(pos, ' ');
  390. if (pos)
  391. pos++;
  392. } else {
  393. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: required challenge field "
  394. "was not present in failure message");
  395. }
  396. if (pos && os_strncmp(pos, "V=", 2) == 0) {
  397. pos += 2;
  398. data->passwd_change_version = atoi(pos);
  399. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: password changing "
  400. "protocol version %d", data->passwd_change_version);
  401. pos = os_strchr(pos, ' ');
  402. if (pos)
  403. pos++;
  404. }
  405. if (pos && os_strncmp(pos, "M=", 2) == 0) {
  406. pos += 2;
  407. msg = pos;
  408. }
  409. if (data->prev_error == ERROR_AUTHENTICATION_FAILURE && retry &&
  410. config && config->phase2 &&
  411. os_strstr(config->phase2, "mschapv2_retry=0")) {
  412. wpa_printf(MSG_DEBUG,
  413. "EAP-MSCHAPV2: mark password retry disabled based on local configuration");
  414. retry = 0;
  415. }
  416. wpa_msg(sm->msg_ctx, MSG_WARNING,
  417. "EAP-MSCHAPV2: failure message: '%s' (retry %sallowed, error "
  418. "%d)",
  419. msg, retry == 1 ? "" : "not ", data->prev_error);
  420. if (data->prev_error == ERROR_PASSWD_EXPIRED &&
  421. data->passwd_change_version == 3 && config) {
  422. if (config->new_password == NULL) {
  423. wpa_msg(sm->msg_ctx, MSG_INFO,
  424. "EAP-MSCHAPV2: Password expired - password "
  425. "change required");
  426. eap_sm_request_new_password(sm);
  427. }
  428. } else if (retry == 1 && config) {
  429. /* TODO: could prevent the current password from being used
  430. * again at least for some period of time */
  431. if (!config->mschapv2_retry)
  432. eap_sm_request_identity(sm);
  433. eap_sm_request_password(sm);
  434. config->mschapv2_retry = 1;
  435. } else if (config) {
  436. /* TODO: prevent retries using same username/password */
  437. config->mschapv2_retry = 0;
  438. }
  439. return retry == 1;
  440. }
  441. static struct wpabuf * eap_mschapv2_change_password(
  442. struct eap_sm *sm, struct eap_mschapv2_data *data,
  443. struct eap_method_ret *ret, const struct eap_mschapv2_hdr *req, u8 id)
  444. {
  445. #ifdef CONFIG_NO_RC4
  446. wpa_printf(MSG_ERROR,
  447. "EAP-MSCHAPV2: RC4 not support in the build - cannot change password");
  448. return NULL;
  449. #else /* CONFIG_NO_RC4 */
  450. struct wpabuf *resp;
  451. int ms_len;
  452. const u8 *username, *password, *new_password;
  453. size_t username_len, password_len, new_password_len;
  454. struct eap_mschapv2_hdr *ms;
  455. struct ms_change_password *cp;
  456. u8 password_hash[16], password_hash_hash[16];
  457. int pwhash;
  458. username = eap_get_config_identity(sm, &username_len);
  459. password = eap_get_config_password2(sm, &password_len, &pwhash);
  460. new_password = eap_get_config_new_password(sm, &new_password_len);
  461. if (username == NULL || password == NULL || new_password == NULL)
  462. return NULL;
  463. username = mschapv2_remove_domain(username, &username_len);
  464. ret->ignore = FALSE;
  465. ret->methodState = METHOD_MAY_CONT;
  466. ret->decision = DECISION_COND_SUCC;
  467. ret->allowNotifications = TRUE;
  468. ms_len = sizeof(*ms) + sizeof(*cp);
  469. resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2, ms_len,
  470. EAP_CODE_RESPONSE, id);
  471. if (resp == NULL)
  472. return NULL;
  473. ms = wpabuf_put(resp, sizeof(*ms));
  474. ms->op_code = MSCHAPV2_OP_CHANGE_PASSWORD;
  475. ms->mschapv2_id = req->mschapv2_id + 1;
  476. WPA_PUT_BE16(ms->ms_length, ms_len);
  477. cp = wpabuf_put(resp, sizeof(*cp));
  478. /* Encrypted-Password */
  479. if (pwhash) {
  480. if (encrypt_pw_block_with_password_hash(
  481. new_password, new_password_len,
  482. password, cp->encr_password))
  483. goto fail;
  484. } else {
  485. if (new_password_encrypted_with_old_nt_password_hash(
  486. new_password, new_password_len,
  487. password, password_len, cp->encr_password))
  488. goto fail;
  489. }
  490. /* Encrypted-Hash */
  491. if (pwhash) {
  492. u8 new_password_hash[16];
  493. if (nt_password_hash(new_password, new_password_len,
  494. new_password_hash) ||
  495. nt_password_hash_encrypted_with_block(password,
  496. new_password_hash,
  497. cp->encr_hash))
  498. goto fail;
  499. } else {
  500. if (old_nt_password_hash_encrypted_with_new_nt_password_hash(
  501. new_password, new_password_len,
  502. password, password_len, cp->encr_hash))
  503. goto fail;
  504. }
  505. /* Peer-Challenge */
  506. if (random_get_bytes(cp->peer_challenge, MSCHAPV2_CHAL_LEN))
  507. goto fail;
  508. /* Reserved, must be zero */
  509. os_memset(cp->reserved, 0, 8);
  510. /* NT-Response */
  511. wpa_hexdump(MSG_DEBUG, "EAP-MSCHAPV2: auth_challenge",
  512. data->passwd_change_challenge, PASSWD_CHANGE_CHAL_LEN);
  513. wpa_hexdump(MSG_DEBUG, "EAP-MSCHAPV2: peer_challenge",
  514. cp->peer_challenge, MSCHAPV2_CHAL_LEN);
  515. wpa_hexdump_ascii(MSG_DEBUG, "EAP-MSCHAPV2: username",
  516. username, username_len);
  517. wpa_hexdump_ascii_key(MSG_DEBUG, "EAP-MSCHAPV2: new password",
  518. new_password, new_password_len);
  519. generate_nt_response(data->passwd_change_challenge, cp->peer_challenge,
  520. username, username_len,
  521. new_password, new_password_len,
  522. cp->nt_response);
  523. wpa_hexdump(MSG_DEBUG, "EAP-MSCHAPV2: NT-Response",
  524. cp->nt_response, MSCHAPV2_NT_RESPONSE_LEN);
  525. /* Authenticator response is not really needed yet, but calculate it
  526. * here so that challenges need not be saved. */
  527. generate_authenticator_response(new_password, new_password_len,
  528. cp->peer_challenge,
  529. data->passwd_change_challenge,
  530. username, username_len,
  531. cp->nt_response, data->auth_response);
  532. data->auth_response_valid = 1;
  533. /* Likewise, generate master_key here since we have the needed data
  534. * available. */
  535. if (nt_password_hash(new_password, new_password_len, password_hash) ||
  536. hash_nt_password_hash(password_hash, password_hash_hash) ||
  537. get_master_key(password_hash_hash, cp->nt_response,
  538. data->master_key)) {
  539. data->auth_response_valid = 0;
  540. goto fail;
  541. }
  542. data->master_key_valid = 1;
  543. /* Flags */
  544. os_memset(cp->flags, 0, 2);
  545. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: TX identifier %d mschapv2_id %d "
  546. "(change pw)", id, ms->mschapv2_id);
  547. return resp;
  548. fail:
  549. wpabuf_free(resp);
  550. return NULL;
  551. #endif /* CONFIG_NO_RC4 */
  552. }
  553. /**
  554. * eap_mschapv2_process - Process an EAP-MSCHAPv2 failure message
  555. * @sm: Pointer to EAP state machine allocated with eap_peer_sm_init()
  556. * @data: Pointer to private EAP method data from eap_mschapv2_init()
  557. * @ret: Return values from EAP request validation and processing
  558. * @req: Pointer to EAP-MSCHAPv2 header from the request
  559. * @req_len: Length of the EAP-MSCHAPv2 data
  560. * @id: EAP identifier used in th erequest
  561. * Returns: Pointer to allocated EAP response packet (eapRespData) or %NULL if
  562. * no reply available
  563. */
  564. static struct wpabuf * eap_mschapv2_failure(struct eap_sm *sm,
  565. struct eap_mschapv2_data *data,
  566. struct eap_method_ret *ret,
  567. const struct eap_mschapv2_hdr *req,
  568. size_t req_len, u8 id)
  569. {
  570. struct wpabuf *resp;
  571. const u8 *msdata = (const u8 *) (req + 1);
  572. char *buf;
  573. size_t len = req_len - sizeof(*req);
  574. int retry = 0;
  575. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Received failure");
  576. wpa_hexdump_ascii(MSG_DEBUG, "EAP-MSCHAPV2: Failure data",
  577. msdata, len);
  578. /*
  579. * eap_mschapv2_failure_txt() expects a nul terminated string, so we
  580. * must allocate a large enough temporary buffer to create that since
  581. * the received message does not include nul termination.
  582. */
  583. buf = dup_binstr(msdata, len);
  584. if (buf) {
  585. retry = eap_mschapv2_failure_txt(sm, data, buf);
  586. os_free(buf);
  587. }
  588. ret->ignore = FALSE;
  589. ret->methodState = METHOD_DONE;
  590. ret->decision = DECISION_FAIL;
  591. ret->allowNotifications = FALSE;
  592. if (data->prev_error == ERROR_PASSWD_EXPIRED &&
  593. data->passwd_change_version == 3) {
  594. struct eap_peer_config *config = eap_get_config(sm);
  595. if (config && config->new_password)
  596. return eap_mschapv2_change_password(sm, data, ret, req,
  597. id);
  598. if (config && config->pending_req_new_password)
  599. return NULL;
  600. } else if (retry && data->prev_error == ERROR_AUTHENTICATION_FAILURE) {
  601. /* TODO: could try to retry authentication, e.g, after having
  602. * changed the username/password. In this case, EAP MS-CHAP-v2
  603. * Failure Response would not be sent here. */
  604. return NULL;
  605. }
  606. /* Note: Only op_code of the EAP-MSCHAPV2 header is included in failure
  607. * message. */
  608. resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2, 1,
  609. EAP_CODE_RESPONSE, id);
  610. if (resp == NULL)
  611. return NULL;
  612. wpabuf_put_u8(resp, MSCHAPV2_OP_FAILURE); /* op_code */
  613. return resp;
  614. }
  615. static int eap_mschapv2_check_config(struct eap_sm *sm)
  616. {
  617. size_t len;
  618. if (eap_get_config_identity(sm, &len) == NULL) {
  619. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Identity not configured");
  620. eap_sm_request_identity(sm);
  621. return -1;
  622. }
  623. if (eap_get_config_password(sm, &len) == NULL) {
  624. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Password not configured");
  625. eap_sm_request_password(sm);
  626. return -1;
  627. }
  628. return 0;
  629. }
  630. static int eap_mschapv2_check_mslen(struct eap_sm *sm, size_t len,
  631. const struct eap_mschapv2_hdr *ms)
  632. {
  633. size_t ms_len = WPA_GET_BE16(ms->ms_length);
  634. if (ms_len == len)
  635. return 0;
  636. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Invalid header: len=%lu "
  637. "ms_len=%lu", (unsigned long) len, (unsigned long) ms_len);
  638. if (sm->workaround) {
  639. /* Some authentication servers use invalid ms_len,
  640. * ignore it for interoperability. */
  641. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: workaround, ignore"
  642. " invalid ms_len %lu (len %lu)",
  643. (unsigned long) ms_len,
  644. (unsigned long) len);
  645. return 0;
  646. }
  647. return -1;
  648. }
  649. static void eap_mschapv2_copy_challenge(struct eap_mschapv2_data *data,
  650. const struct wpabuf *reqData)
  651. {
  652. /*
  653. * Store a copy of the challenge message, so that it can be processed
  654. * again in case retry is allowed after a possible failure.
  655. */
  656. wpabuf_free(data->prev_challenge);
  657. data->prev_challenge = wpabuf_dup(reqData);
  658. }
  659. /**
  660. * eap_mschapv2_process - Process an EAP-MSCHAPv2 request
  661. * @sm: Pointer to EAP state machine allocated with eap_peer_sm_init()
  662. * @priv: Pointer to private EAP method data from eap_mschapv2_init()
  663. * @ret: Return values from EAP request validation and processing
  664. * @reqData: EAP request to be processed (eapReqData)
  665. * Returns: Pointer to allocated EAP response packet (eapRespData) or %NULL if
  666. * no reply available
  667. */
  668. static struct wpabuf * eap_mschapv2_process(struct eap_sm *sm, void *priv,
  669. struct eap_method_ret *ret,
  670. const struct wpabuf *reqData)
  671. {
  672. struct eap_mschapv2_data *data = priv;
  673. struct eap_peer_config *config = eap_get_config(sm);
  674. const struct eap_mschapv2_hdr *ms;
  675. int using_prev_challenge = 0;
  676. const u8 *pos;
  677. size_t len;
  678. u8 id;
  679. if (eap_mschapv2_check_config(sm)) {
  680. ret->ignore = TRUE;
  681. return NULL;
  682. }
  683. if (config->mschapv2_retry && data->prev_challenge &&
  684. data->prev_error == ERROR_AUTHENTICATION_FAILURE) {
  685. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: Replacing pending packet "
  686. "with the previous challenge");
  687. reqData = data->prev_challenge;
  688. using_prev_challenge = 1;
  689. config->mschapv2_retry = 0;
  690. }
  691. pos = eap_hdr_validate(EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2, reqData,
  692. &len);
  693. if (pos == NULL || len < sizeof(*ms) + 1) {
  694. ret->ignore = TRUE;
  695. return NULL;
  696. }
  697. ms = (const struct eap_mschapv2_hdr *) pos;
  698. if (eap_mschapv2_check_mslen(sm, len, ms)) {
  699. ret->ignore = TRUE;
  700. return NULL;
  701. }
  702. id = eap_get_id(reqData);
  703. wpa_printf(MSG_DEBUG, "EAP-MSCHAPV2: RX identifier %d mschapv2_id %d",
  704. id, ms->mschapv2_id);
  705. switch (ms->op_code) {
  706. case MSCHAPV2_OP_CHALLENGE:
  707. if (!using_prev_challenge)
  708. eap_mschapv2_copy_challenge(data, reqData);
  709. return eap_mschapv2_challenge(sm, data, ret, ms, len, id);
  710. case MSCHAPV2_OP_SUCCESS:
  711. return eap_mschapv2_success(sm, data, ret, ms, len, id);
  712. case MSCHAPV2_OP_FAILURE:
  713. return eap_mschapv2_failure(sm, data, ret, ms, len, id);
  714. default:
  715. wpa_printf(MSG_INFO, "EAP-MSCHAPV2: Unknown op %d - ignored",
  716. ms->op_code);
  717. ret->ignore = TRUE;
  718. return NULL;
  719. }
  720. }
  721. static Boolean eap_mschapv2_isKeyAvailable(struct eap_sm *sm, void *priv)
  722. {
  723. struct eap_mschapv2_data *data = priv;
  724. return data->success && data->master_key_valid;
  725. }
  726. static u8 * eap_mschapv2_getKey(struct eap_sm *sm, void *priv, size_t *len)
  727. {
  728. struct eap_mschapv2_data *data = priv;
  729. u8 *key;
  730. int key_len;
  731. if (!data->master_key_valid || !data->success)
  732. return NULL;
  733. key_len = 2 * MSCHAPV2_KEY_LEN;
  734. key = os_malloc(key_len);
  735. if (key == NULL)
  736. return NULL;
  737. /* MSK = server MS-MPPE-Recv-Key | MS-MPPE-Send-Key, i.e.,
  738. * peer MS-MPPE-Send-Key | MS-MPPE-Recv-Key */
  739. get_asymetric_start_key(data->master_key, key, MSCHAPV2_KEY_LEN, 1, 0);
  740. get_asymetric_start_key(data->master_key, key + MSCHAPV2_KEY_LEN,
  741. MSCHAPV2_KEY_LEN, 0, 0);
  742. wpa_hexdump_key(MSG_DEBUG, "EAP-MSCHAPV2: Derived key",
  743. key, key_len);
  744. *len = key_len;
  745. return key;
  746. }
  747. /**
  748. * eap_peer_mschapv2_register - Register EAP-MSCHAPv2 peer method
  749. * Returns: 0 on success, -1 on failure
  750. *
  751. * This function is used to register EAP-MSCHAPv2 peer method into the EAP
  752. * method list.
  753. */
  754. int eap_peer_mschapv2_register(void)
  755. {
  756. struct eap_method *eap;
  757. eap = eap_peer_method_alloc(EAP_PEER_METHOD_INTERFACE_VERSION,
  758. EAP_VENDOR_IETF, EAP_TYPE_MSCHAPV2,
  759. "MSCHAPV2");
  760. if (eap == NULL)
  761. return -1;
  762. eap->init = eap_mschapv2_init;
  763. eap->deinit = eap_mschapv2_deinit;
  764. eap->process = eap_mschapv2_process;
  765. eap->isKeyAvailable = eap_mschapv2_isKeyAvailable;
  766. eap->getKey = eap_mschapv2_getKey;
  767. return eap_peer_method_register(eap);
  768. }