2 * hostapd / EAP-TLS (RFC 2716)
3 * Copyright (c) 2004-2008, Jouni Malinen <j@w1.fi>
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
13 #include "eap_tls_common.h"
14 #include "crypto/tls.h"
17 static void eap_tls_reset(struct eap_sm *sm, void *priv);
21 struct eap_ssl_data ssl;
22 enum { START, CONTINUE, SUCCESS, FAILURE } state;
28 static const char * eap_tls_state_txt(int state)
45 static void eap_tls_state(struct eap_tls_data *data, int state)
47 wpa_printf(MSG_DEBUG, "EAP-TLS: %s -> %s",
48 eap_tls_state_txt(data->state),
49 eap_tls_state_txt(state));
54 static void * eap_tls_init(struct eap_sm *sm)
56 struct eap_tls_data *data;
58 data = os_zalloc(sizeof(*data));
63 if (eap_server_tls_ssl_init(sm, &data->ssl, 1)) {
64 wpa_printf(MSG_INFO, "EAP-TLS: Failed to initialize SSL.");
65 eap_tls_reset(sm, data);
69 data->eap_type = EAP_TYPE_TLS;
75 #ifdef EAP_SERVER_UNAUTH_TLS
76 static void * eap_unauth_tls_init(struct eap_sm *sm)
78 struct eap_tls_data *data;
80 data = os_zalloc(sizeof(*data));
85 if (eap_server_tls_ssl_init(sm, &data->ssl, 0)) {
86 wpa_printf(MSG_INFO, "EAP-TLS: Failed to initialize SSL.");
87 eap_tls_reset(sm, data);
91 data->eap_type = EAP_UNAUTH_TLS_TYPE;
94 #endif /* EAP_SERVER_UNAUTH_TLS */
98 static void * eap_wfa_unauth_tls_init(struct eap_sm *sm)
100 struct eap_tls_data *data;
102 data = os_zalloc(sizeof(*data));
107 if (eap_server_tls_ssl_init(sm, &data->ssl, 0)) {
108 wpa_printf(MSG_INFO, "EAP-TLS: Failed to initialize SSL.");
109 eap_tls_reset(sm, data);
113 data->eap_type = EAP_WFA_UNAUTH_TLS_TYPE;
116 #endif /* CONFIG_HS20 */
119 static void eap_tls_reset(struct eap_sm *sm, void *priv)
121 struct eap_tls_data *data = priv;
124 eap_server_tls_ssl_deinit(sm, &data->ssl);
129 static struct wpabuf * eap_tls_build_start(struct eap_sm *sm,
130 struct eap_tls_data *data, u8 id)
134 req = eap_tls_msg_alloc(data->eap_type, 1, EAP_CODE_REQUEST, id);
136 wpa_printf(MSG_ERROR, "EAP-TLS: Failed to allocate memory for "
138 eap_tls_state(data, FAILURE);
142 wpabuf_put_u8(req, EAP_TLS_FLAGS_START);
144 eap_tls_state(data, CONTINUE);
150 static struct wpabuf * eap_tls_buildReq(struct eap_sm *sm, void *priv, u8 id)
152 struct eap_tls_data *data = priv;
155 if (data->ssl.state == FRAG_ACK) {
156 return eap_server_tls_build_ack(id, data->eap_type, 0);
159 if (data->ssl.state == WAIT_FRAG_ACK) {
160 res = eap_server_tls_build_msg(&data->ssl, data->eap_type, 0,
162 goto check_established;
165 switch (data->state) {
167 return eap_tls_build_start(sm, data, id);
169 if (tls_connection_established(sm->ssl_ctx, data->ssl.conn))
170 data->established = 1;
173 wpa_printf(MSG_DEBUG, "EAP-TLS: %s - unexpected state %d",
174 __func__, data->state);
178 res = eap_server_tls_build_msg(&data->ssl, data->eap_type, 0, id);
181 if (data->established && data->ssl.state != WAIT_FRAG_ACK) {
182 /* TLS handshake has been completed and there are no more
183 * fragments waiting to be sent out. */
184 wpa_printf(MSG_DEBUG, "EAP-TLS: Done");
185 eap_tls_state(data, SUCCESS);
192 static Boolean eap_tls_check(struct eap_sm *sm, void *priv,
193 struct wpabuf *respData)
195 struct eap_tls_data *data = priv;
199 if (data->eap_type == EAP_UNAUTH_TLS_TYPE)
200 pos = eap_hdr_validate(EAP_VENDOR_UNAUTH_TLS,
201 EAP_VENDOR_TYPE_UNAUTH_TLS, respData,
203 else if (data->eap_type == EAP_WFA_UNAUTH_TLS_TYPE)
204 pos = eap_hdr_validate(EAP_VENDOR_WFA_NEW,
205 EAP_VENDOR_WFA_UNAUTH_TLS, respData,
208 pos = eap_hdr_validate(EAP_VENDOR_IETF, data->eap_type,
210 if (pos == NULL || len < 1) {
211 wpa_printf(MSG_INFO, "EAP-TLS: Invalid frame");
219 static void eap_tls_process_msg(struct eap_sm *sm, void *priv,
220 const struct wpabuf *respData)
222 struct eap_tls_data *data = priv;
223 if (data->state == SUCCESS && wpabuf_len(data->ssl.tls_in) == 0) {
224 wpa_printf(MSG_DEBUG, "EAP-TLS: Client acknowledged final TLS "
225 "handshake message");
228 if (eap_server_tls_phase1(sm, &data->ssl) < 0)
229 eap_tls_state(data, FAILURE);
233 static void eap_tls_process(struct eap_sm *sm, void *priv,
234 struct wpabuf *respData)
236 struct eap_tls_data *data = priv;
237 if (eap_server_tls_process(sm, &data->ssl, respData, data,
238 data->eap_type, NULL, eap_tls_process_msg) <
240 eap_tls_state(data, FAILURE);
244 static Boolean eap_tls_isDone(struct eap_sm *sm, void *priv)
246 struct eap_tls_data *data = priv;
247 return data->state == SUCCESS || data->state == FAILURE;
251 static u8 * eap_tls_getKey(struct eap_sm *sm, void *priv, size_t *len)
253 struct eap_tls_data *data = priv;
256 if (data->state != SUCCESS)
259 eapKeyData = eap_server_tls_derive_key(sm, &data->ssl,
260 "client EAP encryption",
263 *len = EAP_TLS_KEY_LEN;
264 wpa_hexdump(MSG_DEBUG, "EAP-TLS: Derived key",
265 eapKeyData, EAP_TLS_KEY_LEN);
267 wpa_printf(MSG_DEBUG, "EAP-TLS: Failed to derive key");
274 static u8 * eap_tls_get_emsk(struct eap_sm *sm, void *priv, size_t *len)
276 struct eap_tls_data *data = priv;
277 u8 *eapKeyData, *emsk;
279 if (data->state != SUCCESS)
282 eapKeyData = eap_server_tls_derive_key(sm, &data->ssl,
283 "client EAP encryption",
284 EAP_TLS_KEY_LEN + EAP_EMSK_LEN);
286 emsk = os_malloc(EAP_EMSK_LEN);
288 os_memcpy(emsk, eapKeyData + EAP_TLS_KEY_LEN,
296 wpa_hexdump(MSG_DEBUG, "EAP-TLS: Derived EMSK",
299 wpa_printf(MSG_DEBUG, "EAP-TLS: Failed to derive EMSK");
306 static Boolean eap_tls_isSuccess(struct eap_sm *sm, void *priv)
308 struct eap_tls_data *data = priv;
309 return data->state == SUCCESS;
313 int eap_server_tls_register(void)
315 struct eap_method *eap;
318 eap = eap_server_method_alloc(EAP_SERVER_METHOD_INTERFACE_VERSION,
319 EAP_VENDOR_IETF, EAP_TYPE_TLS, "TLS");
323 eap->init = eap_tls_init;
324 eap->reset = eap_tls_reset;
325 eap->buildReq = eap_tls_buildReq;
326 eap->check = eap_tls_check;
327 eap->process = eap_tls_process;
328 eap->isDone = eap_tls_isDone;
329 eap->getKey = eap_tls_getKey;
330 eap->isSuccess = eap_tls_isSuccess;
331 eap->get_emsk = eap_tls_get_emsk;
333 ret = eap_server_method_register(eap);
335 eap_server_method_free(eap);
340 #ifdef EAP_SERVER_UNAUTH_TLS
341 int eap_server_unauth_tls_register(void)
343 struct eap_method *eap;
346 eap = eap_server_method_alloc(EAP_SERVER_METHOD_INTERFACE_VERSION,
347 EAP_VENDOR_UNAUTH_TLS,
348 EAP_VENDOR_TYPE_UNAUTH_TLS,
353 eap->init = eap_unauth_tls_init;
354 eap->reset = eap_tls_reset;
355 eap->buildReq = eap_tls_buildReq;
356 eap->check = eap_tls_check;
357 eap->process = eap_tls_process;
358 eap->isDone = eap_tls_isDone;
359 eap->getKey = eap_tls_getKey;
360 eap->isSuccess = eap_tls_isSuccess;
361 eap->get_emsk = eap_tls_get_emsk;
363 ret = eap_server_method_register(eap);
365 eap_server_method_free(eap);
368 #endif /* EAP_SERVER_UNAUTH_TLS */
372 int eap_server_wfa_unauth_tls_register(void)
374 struct eap_method *eap;
377 eap = eap_server_method_alloc(EAP_SERVER_METHOD_INTERFACE_VERSION,
379 EAP_VENDOR_WFA_UNAUTH_TLS,
384 eap->init = eap_wfa_unauth_tls_init;
385 eap->reset = eap_tls_reset;
386 eap->buildReq = eap_tls_buildReq;
387 eap->check = eap_tls_check;
388 eap->process = eap_tls_process;
389 eap->isDone = eap_tls_isDone;
390 eap->getKey = eap_tls_getKey;
391 eap->isSuccess = eap_tls_isSuccess;
392 eap->get_emsk = eap_tls_get_emsk;
394 ret = eap_server_method_register(eap);
396 eap_server_method_free(eap);
399 #endif /* CONFIG_HS20 */