#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <stdarg.h>
#include <fcntl.h>
#include <unistd.h>
#include <stropts.h>
#include <string.h>
#include <stddef.h>
#include "wpa_impl.h"
#include "driver.h"
#define WPA_STATUS(status) (status == DLADM_STATUS_OK? 0 : -1)
int
wpa_driver_wifi_get_bssid(dladm_handle_t handle, datalink_id_t linkid,
char *bssid)
{
dladm_status_t status;
dladm_wlan_linkattr_t attr;
dladm_wlan_attr_t *wl_attrp;
status = dladm_wlan_get_linkattr(handle, linkid, &attr);
if (status != DLADM_STATUS_OK)
return (-1);
wl_attrp = &attr.la_wlan_attr;
if ((attr.la_valid & DLADM_WLAN_LINKATTR_WLAN) == 0 ||
(wl_attrp->wa_valid & DLADM_WLAN_ATTR_BSSID) == 0)
return (-1);
(void) memcpy(bssid, wl_attrp->wa_bssid.wb_bytes, DLADM_WLAN_BSSID_LEN);
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_get_bssid: " MACSTR,
MAC2STR((unsigned char *)bssid));
return (WPA_STATUS(status));
}
int
wpa_driver_wifi_get_ssid(dladm_handle_t handle, datalink_id_t linkid,
char *ssid)
{
int ret;
dladm_status_t status;
dladm_wlan_linkattr_t attr;
dladm_wlan_attr_t *wl_attrp;
status = dladm_wlan_get_linkattr(handle, linkid, &attr);
if (status != DLADM_STATUS_OK)
return (-1);
wl_attrp = &attr.la_wlan_attr;
if ((attr.la_valid & DLADM_WLAN_LINKATTR_WLAN) == 0 ||
(wl_attrp->wa_valid & DLADM_WLAN_ATTR_ESSID) == 0)
return (-1);
(void) memcpy(ssid, wl_attrp->wa_essid.we_bytes, MAX_ESSID_LENGTH);
ret = strlen(ssid);
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_get_ssid: ssid=%s len=%d",
ssid, ret);
return (ret);
}
static int
wpa_driver_wifi_set_wpa_ie(dladm_handle_t handle, datalink_id_t linkid,
uint8_t *wpa_ie, uint32_t wpa_ie_len)
{
dladm_status_t status;
wpa_printf(MSG_DEBUG, "%s", "wpa_driver_wifi_set_wpa_ie");
status = dladm_wlan_wpa_set_ie(handle, linkid, wpa_ie, wpa_ie_len);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_set_wpa(dladm_handle_t handle, datalink_id_t linkid,
boolean_t enabled)
{
dladm_status_t status;
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_set_wpa: enable=%d", enabled);
if (!enabled && wpa_driver_wifi_set_wpa_ie(handle, linkid, NULL, 0) < 0)
return (-1);
status = dladm_wlan_wpa_set_wpa(handle, linkid, enabled);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_del_key(dladm_handle_t handle, datalink_id_t linkid,
int key_idx, unsigned char *addr)
{
dladm_status_t status;
dladm_wlan_bssid_t bss;
wpa_printf(MSG_DEBUG, "%s: id=%d", "wpa_driver_wifi_del_key",
key_idx);
(void) memcpy(bss.wb_bytes, addr, DLADM_WLAN_BSSID_LEN);
status = dladm_wlan_wpa_del_key(handle, linkid, key_idx, &bss);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_set_key(dladm_handle_t handle, datalink_id_t linkid,
wpa_alg alg, unsigned char *addr, int key_idx, boolean_t set_tx,
uint8_t *seq, uint32_t seq_len, uint8_t *key, uint32_t key_len)
{
char *alg_name;
dladm_wlan_cipher_t cipher;
dladm_wlan_bssid_t bss;
dladm_status_t status;
wpa_printf(MSG_DEBUG, "%s", "wpa_driver_wifi_set_key");
if (alg == WPA_ALG_NONE)
return (wpa_driver_wifi_del_key(handle, linkid, key_idx, addr));
switch (alg) {
case WPA_ALG_WEP:
alg_name = "WEP";
cipher = DLADM_WLAN_CIPHER_WEP;
break;
case WPA_ALG_TKIP:
alg_name = "TKIP";
cipher = DLADM_WLAN_CIPHER_TKIP;
break;
case WPA_ALG_CCMP:
alg_name = "CCMP";
cipher = DLADM_WLAN_CIPHER_AES_CCM;
break;
default:
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_set_key:"
" unknown/unsupported algorithm %d", alg);
return (-1);
}
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_set_key: alg=%s key_idx=%d"
" set_tx=%d seq_len=%d seq=%d key_len=%d",
alg_name, key_idx, set_tx,
seq_len, *(uint64_t *)(uintptr_t)seq, key_len);
if (seq_len > sizeof (uint64_t)) {
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_set_key:"
" seq_len %d too big", seq_len);
return (-1);
}
(void) memcpy(bss.wb_bytes, addr, DLADM_WLAN_BSSID_LEN);
status = dladm_wlan_wpa_set_key(handle, linkid, cipher, &bss, set_tx,
*(uint64_t *)(uintptr_t)seq, key_idx, key, key_len);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_disassociate(dladm_handle_t handle, datalink_id_t linkid,
int reason_code)
{
dladm_status_t status;
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_disassociate");
status = dladm_wlan_wpa_set_mlme(handle, linkid,
DLADM_WLAN_MLME_DISASSOC, reason_code, NULL);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_associate(dladm_handle_t handle, datalink_id_t linkid,
const char *bssid, uint8_t *wpa_ie, uint32_t wpa_ie_len)
{
dladm_status_t status;
dladm_wlan_bssid_t bss;
wpa_printf(MSG_DEBUG, "wpa_driver_wifi_associate : "
MACSTR, MAC2STR(bssid));
if (wpa_driver_wifi_set_wpa_ie(handle, linkid, wpa_ie, wpa_ie_len) < 0)
return (-1);
(void) memcpy(bss.wb_bytes, bssid, DLADM_WLAN_BSSID_LEN);
status = dladm_wlan_wpa_set_mlme(handle, linkid, DLADM_WLAN_MLME_ASSOC,
0, &bss);
return (WPA_STATUS(status));
}
static int
wpa_driver_wifi_scan(dladm_handle_t handle, datalink_id_t linkid)
{
dladm_status_t status;
wpa_printf(MSG_DEBUG, "%s", "wpa_driver_wifi_scan");
(void) wpa_driver_wifi_disassociate(handle, linkid,
DLADM_WLAN_REASON_DISASSOC_LEAVING);
status = dladm_wlan_scan(handle, linkid, NULL, NULL);
wpa_printf(MSG_DEBUG, "%s: return", "wpa_driver_wifi_scan");
return (WPA_STATUS(status));
}
int
wpa_driver_wifi_get_scan_results(dladm_handle_t handle, datalink_id_t linkid,
dladm_wlan_ess_t *results, uint32_t max_size)
{
uint_t ret;
wpa_printf(MSG_DEBUG, "%s: max size=%d\n",
"wpa_driver_wifi_get_scan_results", max_size);
if (dladm_wlan_wpa_get_sr(handle, linkid, results, max_size, &ret)
!= DLADM_STATUS_OK) {
return (-1);
}
return (ret);
}
struct wpa_driver_ops wpa_driver_wifi_ops = {
wpa_driver_wifi_get_bssid,
wpa_driver_wifi_get_ssid,
wpa_driver_wifi_set_wpa,
wpa_driver_wifi_set_key,
wpa_driver_wifi_scan,
wpa_driver_wifi_get_scan_results,
wpa_driver_wifi_disassociate,
wpa_driver_wifi_associate
};