139beb93cSSam Leffler /* 239beb93cSSam Leffler * EAP peer method: EAP-GTC (RFC 3748) 339beb93cSSam Leffler * Copyright (c) 2004-2006, Jouni Malinen <j@w1.fi> 439beb93cSSam Leffler * 539beb93cSSam Leffler * This program is free software; you can redistribute it and/or modify 639beb93cSSam Leffler * it under the terms of the GNU General Public License version 2 as 739beb93cSSam Leffler * published by the Free Software Foundation. 839beb93cSSam Leffler * 939beb93cSSam Leffler * Alternatively, this software may be distributed under the terms of BSD 1039beb93cSSam Leffler * license. 1139beb93cSSam Leffler * 1239beb93cSSam Leffler * See README and COPYING for more details. 1339beb93cSSam Leffler */ 1439beb93cSSam Leffler 1539beb93cSSam Leffler #include "includes.h" 1639beb93cSSam Leffler 1739beb93cSSam Leffler #include "common.h" 1839beb93cSSam Leffler #include "eap_i.h" 1939beb93cSSam Leffler 2039beb93cSSam Leffler 2139beb93cSSam Leffler struct eap_gtc_data { 2239beb93cSSam Leffler int prefix; 2339beb93cSSam Leffler }; 2439beb93cSSam Leffler 2539beb93cSSam Leffler 2639beb93cSSam Leffler static void * eap_gtc_init(struct eap_sm *sm) 2739beb93cSSam Leffler { 2839beb93cSSam Leffler struct eap_gtc_data *data; 2939beb93cSSam Leffler data = os_zalloc(sizeof(*data)); 3039beb93cSSam Leffler if (data == NULL) 3139beb93cSSam Leffler return NULL; 3239beb93cSSam Leffler 3339beb93cSSam Leffler if (sm->m && sm->m->vendor == EAP_VENDOR_IETF && 3439beb93cSSam Leffler sm->m->method == EAP_TYPE_FAST) { 3539beb93cSSam Leffler wpa_printf(MSG_DEBUG, "EAP-GTC: EAP-FAST tunnel - use prefix " 3639beb93cSSam Leffler "with challenge/response"); 3739beb93cSSam Leffler data->prefix = 1; 3839beb93cSSam Leffler } 3939beb93cSSam Leffler return data; 4039beb93cSSam Leffler } 4139beb93cSSam Leffler 4239beb93cSSam Leffler 4339beb93cSSam Leffler static void eap_gtc_deinit(struct eap_sm *sm, void *priv) 4439beb93cSSam Leffler { 4539beb93cSSam Leffler struct eap_gtc_data *data = priv; 4639beb93cSSam Leffler os_free(data); 4739beb93cSSam Leffler } 4839beb93cSSam Leffler 4939beb93cSSam Leffler 5039beb93cSSam Leffler static struct wpabuf * eap_gtc_process(struct eap_sm *sm, void *priv, 5139beb93cSSam Leffler struct eap_method_ret *ret, 5239beb93cSSam Leffler const struct wpabuf *reqData) 5339beb93cSSam Leffler { 5439beb93cSSam Leffler struct eap_gtc_data *data = priv; 5539beb93cSSam Leffler struct wpabuf *resp; 5639beb93cSSam Leffler const u8 *pos, *password, *identity; 5739beb93cSSam Leffler size_t password_len, identity_len, len, plen; 5839beb93cSSam Leffler int otp; 5939beb93cSSam Leffler u8 id; 6039beb93cSSam Leffler 6139beb93cSSam Leffler pos = eap_hdr_validate(EAP_VENDOR_IETF, EAP_TYPE_GTC, reqData, &len); 6239beb93cSSam Leffler if (pos == NULL) { 6339beb93cSSam Leffler ret->ignore = TRUE; 6439beb93cSSam Leffler return NULL; 6539beb93cSSam Leffler } 6639beb93cSSam Leffler id = eap_get_id(reqData); 6739beb93cSSam Leffler 6839beb93cSSam Leffler wpa_hexdump_ascii(MSG_MSGDUMP, "EAP-GTC: Request message", pos, len); 6939beb93cSSam Leffler if (data->prefix && 7039beb93cSSam Leffler (len < 10 || os_memcmp(pos, "CHALLENGE=", 10) != 0)) { 7139beb93cSSam Leffler wpa_printf(MSG_DEBUG, "EAP-GTC: Challenge did not start with " 7239beb93cSSam Leffler "expected prefix"); 7339beb93cSSam Leffler 7439beb93cSSam Leffler /* Send an empty response in order to allow tunneled 7539beb93cSSam Leffler * acknowledgement of the failure. This will also cover the 7639beb93cSSam Leffler * error case which seems to use EAP-MSCHAPv2 like error 7739beb93cSSam Leffler * reporting with EAP-GTC inside EAP-FAST tunnel. */ 7839beb93cSSam Leffler resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_GTC, 7939beb93cSSam Leffler 0, EAP_CODE_RESPONSE, id); 8039beb93cSSam Leffler return resp; 8139beb93cSSam Leffler } 8239beb93cSSam Leffler 8339beb93cSSam Leffler password = eap_get_config_otp(sm, &password_len); 8439beb93cSSam Leffler if (password) 8539beb93cSSam Leffler otp = 1; 8639beb93cSSam Leffler else { 8739beb93cSSam Leffler password = eap_get_config_password(sm, &password_len); 8839beb93cSSam Leffler otp = 0; 8939beb93cSSam Leffler } 9039beb93cSSam Leffler 9139beb93cSSam Leffler if (password == NULL) { 9239beb93cSSam Leffler wpa_printf(MSG_INFO, "EAP-GTC: Password not configured"); 9339beb93cSSam Leffler eap_sm_request_otp(sm, (const char *) pos, len); 9439beb93cSSam Leffler ret->ignore = TRUE; 9539beb93cSSam Leffler return NULL; 9639beb93cSSam Leffler } 9739beb93cSSam Leffler 9839beb93cSSam Leffler ret->ignore = FALSE; 9939beb93cSSam Leffler 10039beb93cSSam Leffler ret->methodState = data->prefix ? METHOD_MAY_CONT : METHOD_DONE; 10139beb93cSSam Leffler ret->decision = DECISION_COND_SUCC; 10239beb93cSSam Leffler ret->allowNotifications = FALSE; 10339beb93cSSam Leffler 10439beb93cSSam Leffler plen = password_len; 10539beb93cSSam Leffler identity = eap_get_config_identity(sm, &identity_len); 10639beb93cSSam Leffler if (identity == NULL) 10739beb93cSSam Leffler return NULL; 10839beb93cSSam Leffler if (data->prefix) 10939beb93cSSam Leffler plen += 9 + identity_len + 1; 11039beb93cSSam Leffler resp = eap_msg_alloc(EAP_VENDOR_IETF, EAP_TYPE_GTC, plen, 11139beb93cSSam Leffler EAP_CODE_RESPONSE, id); 11239beb93cSSam Leffler if (resp == NULL) 11339beb93cSSam Leffler return NULL; 11439beb93cSSam Leffler if (data->prefix) { 11539beb93cSSam Leffler wpabuf_put_data(resp, "RESPONSE=", 9); 11639beb93cSSam Leffler wpabuf_put_data(resp, identity, identity_len); 11739beb93cSSam Leffler wpabuf_put_u8(resp, '\0'); 11839beb93cSSam Leffler } 11939beb93cSSam Leffler wpabuf_put_data(resp, password, password_len); 12039beb93cSSam Leffler wpa_hexdump_ascii_key(MSG_MSGDUMP, "EAP-GTC: Response", 12139beb93cSSam Leffler wpabuf_head_u8(resp) + sizeof(struct eap_hdr) + 12239beb93cSSam Leffler 1, plen); 12339beb93cSSam Leffler 12439beb93cSSam Leffler if (otp) { 12539beb93cSSam Leffler wpa_printf(MSG_DEBUG, "EAP-GTC: Forgetting used password"); 12639beb93cSSam Leffler eap_clear_config_otp(sm); 12739beb93cSSam Leffler } 12839beb93cSSam Leffler 12939beb93cSSam Leffler return resp; 13039beb93cSSam Leffler } 13139beb93cSSam Leffler 13239beb93cSSam Leffler 13339beb93cSSam Leffler int eap_peer_gtc_register(void) 13439beb93cSSam Leffler { 13539beb93cSSam Leffler struct eap_method *eap; 13639beb93cSSam Leffler int ret; 13739beb93cSSam Leffler 13839beb93cSSam Leffler eap = eap_peer_method_alloc(EAP_PEER_METHOD_INTERFACE_VERSION, 13939beb93cSSam Leffler EAP_VENDOR_IETF, EAP_TYPE_GTC, "GTC"); 14039beb93cSSam Leffler if (eap == NULL) 14139beb93cSSam Leffler return -1; 14239beb93cSSam Leffler 14339beb93cSSam Leffler eap->init = eap_gtc_init; 14439beb93cSSam Leffler eap->deinit = eap_gtc_deinit; 14539beb93cSSam Leffler eap->process = eap_gtc_process; 14639beb93cSSam Leffler 14739beb93cSSam Leffler ret = eap_peer_method_register(eap); 14839beb93cSSam Leffler if (ret) 14939beb93cSSam Leffler eap_peer_method_free(eap); 15039beb93cSSam Leffler return ret; 15139beb93cSSam Leffler } 152