nodemcu-firmware/components/modules/wifi_ap.c

267 lines
8.4 KiB
C

/*
* Copyright 2016-2021 Dius Computing Pty Ltd. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the
* distribution.
* - Neither the name of the copyright holders nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*
* @author Johny Mattsson <jmattsson@dius.com.au>
*/
#include "module.h"
#include "lauxlib.h"
#include "lextra.h"
#include "wifi_common.h"
#include "ip_fmt.h"
#include "nodemcu_esp_event.h"
#include <string.h>
#include "dhcpserver/dhcpserver_options.h"
#include "esp_netif.h"
// Note: these are documented in wifi.md, update there too if changed here!
#define DEFAULT_AP_CHANNEL 11
#define DEFAULT_AP_MAXCONNS 4
#define DEFAULT_AP_BEACON 100
static esp_netif_t *wifi_ap = NULL;
// --- Event handling ----------------------------------------------------
static void ap_staconn (lua_State *L, const void *data);
static void ap_stadisconn (lua_State *L, const void *data);
static void ap_probe_req (lua_State *L, const void *data);
static void empty_arg (lua_State *L, const void *data) {}
static const event_desc_t events[] =
{
{ "start", &WIFI_EVENT, WIFI_EVENT_AP_START, empty_arg },
{ "stop", &WIFI_EVENT, WIFI_EVENT_AP_STOP, empty_arg },
{ "sta_connected", &WIFI_EVENT, WIFI_EVENT_AP_STACONNECTED, ap_staconn},
{ "sta_disconnected", &WIFI_EVENT, WIFI_EVENT_AP_STADISCONNECTED, ap_stadisconn },
{ "probe_req", &WIFI_EVENT, WIFI_EVENT_AP_PROBEREQRECVED, ap_probe_req },
};
static int event_cb[ARRAY_LEN(events)];
static void ap_staconn (lua_State *L, const void *data)
{
const wifi_event_ap_staconnected_t *sta_connected =
(const wifi_event_ap_staconnected_t *)data;
char mac[MAC_STR_SZ];
macstr (mac, sta_connected->mac);
lua_pushstring (L, mac);
lua_setfield (L, -2, "mac");
lua_pushinteger (L, sta_connected->aid);
lua_setfield (L, -2, "id");
}
static void ap_stadisconn (lua_State *L, const void *data)
{
const wifi_event_ap_stadisconnected_t *sta_disconnected =
(const wifi_event_ap_stadisconnected_t *)data;
char mac[MAC_STR_SZ];
macstr (mac, sta_disconnected->mac);
lua_pushstring (L, mac);
lua_setfield (L, -2, "mac");
lua_pushinteger (L, sta_disconnected->aid);
lua_setfield (L, -2, "id");
}
static void ap_probe_req (lua_State *L, const void *data)
{
const wifi_event_ap_probe_req_rx_t *ap_probereqrecved =
(const wifi_event_ap_probe_req_rx_t *)data;
char str[MAC_STR_SZ];
macstr (str, ap_probereqrecved->mac);
lua_pushstring (L, str);
lua_setfield (L, -2, "from");
lua_pushinteger (L, ap_probereqrecved->rssi);
lua_setfield (L, -2, "rssi");
}
static void on_event (esp_event_base_t base, int32_t id, const void *data)
{
int idx = wifi_event_idx_by_id (events, ARRAY_LEN(events), base, id);
if (idx < 0 || event_cb[idx] == LUA_NOREF)
return;
lua_State *L = lua_getstate ();
lua_rawgeti (L, LUA_REGISTRYINDEX, event_cb[idx]);
lua_pushstring (L, events[idx].name);
lua_createtable (L, 0, 5);
events[idx].fill_cb_arg (L, data);
lua_call (L, 2, 0);
}
NODEMCU_ESP_EVENT(WIFI_EVENT, WIFI_EVENT_AP_START, on_event);
NODEMCU_ESP_EVENT(WIFI_EVENT, WIFI_EVENT_AP_STOP, on_event);
NODEMCU_ESP_EVENT(WIFI_EVENT, WIFI_EVENT_AP_STACONNECTED, on_event);
NODEMCU_ESP_EVENT(WIFI_EVENT, WIFI_EVENT_AP_STADISCONNECTED, on_event);
NODEMCU_ESP_EVENT(WIFI_EVENT, WIFI_EVENT_AP_PROBEREQRECVED, on_event);
void wifi_ap_init (void)
{
wifi_ap = esp_netif_create_default_wifi_ap();
for (unsigned i = 0; i < ARRAY_LEN(event_cb); ++i)
event_cb[i] = LUA_NOREF;
}
// --- Lua API funcs -----------------------------------------------------
static int wifi_ap_setip(lua_State *L)
{
luaL_checkanytable (L, 1);
size_t len = 0;
const char *str = NULL;
esp_netif_ip_info_t ip_info = { 0, };
lua_getfield (L, 1, "ip");
str = luaL_checklstring (L, -1, &len);
if (esp_netif_str_to_ip4(str, &ip_info.ip) != ESP_OK)
{
return luaL_error(L, "Could not parse IP address, aborting");
}
lua_getfield (L, 1, "gateway");
str = luaL_checklstring (L, -1, &len);
if (esp_netif_str_to_ip4(str, &ip_info.gw) != ESP_OK)
{
return luaL_error(L, "Could not parse Gateway address, aborting");
}
lua_getfield (L, 1, "netmask");
str = luaL_checklstring (L, -1, &len);
if (esp_netif_str_to_ip4(str, &ip_info.netmask) != ESP_OK)
{
return luaL_error(L, "Could not parse Netmask, aborting");
}
ESP_ERROR_CHECK(esp_netif_dhcps_stop(wifi_ap));
esp_netif_set_ip_info(wifi_ap, &ip_info);
esp_netif_dns_info_t dns = { .ip = { .type = ESP_IPADDR_TYPE_V4 } };
lua_getfield (L, 1, "dns");
str = luaL_optlstring(L, -1, "", &len);
if (*str)
{
if (esp_netif_str_to_ip4(str, &dns.ip.u_addr.ip4) != ESP_OK)
{
return luaL_error(L, "Could not parse Dns, aborting");
}
esp_netif_set_dns_info(wifi_ap, ESP_NETIF_DNS_MAIN, &dns);
}
ESP_ERROR_CHECK(esp_netif_dhcps_start(wifi_ap));
return 0;
}
static int wifi_ap_sethostname(lua_State *L)
{
size_t l;
esp_err_t err;
const char *hostname = luaL_checklstring(L, 1, &l);
err = esp_netif_set_hostname(wifi_ap, hostname);
if (err != ESP_OK)
return luaL_error (L, "failed to set hostname, code %d", err);
lua_pushboolean (L, err==ESP_OK);
return 1;
}
static int wifi_ap_config (lua_State *L)
{
luaL_checkanytable (L, 1);
bool save = luaL_optbool (L, 2, DEFAULT_SAVE);
wifi_config_t cfg;
memset (&cfg, 0, sizeof (cfg));
lua_getfield (L, 1, "ssid");
size_t len;
const char *str = luaL_checklstring (L, -1, &len);
if (len > sizeof (cfg.ap.ssid))
len = sizeof (cfg.ap.ssid);
strncpy ((char *)cfg.ap.ssid, str, len);
cfg.ap.ssid_len = len;
lua_getfield (L, 1, "pwd");
str = luaL_optlstring (L, -1, "", &len);
if (len > sizeof (cfg.ap.password))
len = sizeof (cfg.ap.password);
strncpy ((char *)cfg.ap.password, str, len);
lua_getfield (L, 1, "auth");
int authmode = luaL_optint (L, -1, WIFI_AUTH_WPA2_PSK);
if (authmode < 0 || authmode >= WIFI_AUTH_MAX)
return luaL_error (L, "unknown auth mode %d", authmode);
cfg.ap.authmode = authmode;
lua_getfield (L, 1, "channel");
cfg.ap.channel = luaL_optint (L, -1, DEFAULT_AP_CHANNEL);
lua_getfield (L, 1, "hidden");
cfg.ap.ssid_hidden = luaL_optbool (L, -1, false);
lua_getfield (L, 1, "max");
cfg.ap.max_connection = luaL_optint (L, -1, DEFAULT_AP_MAXCONNS);
lua_getfield (L, 1, "beacon");
cfg.ap.beacon_interval = luaL_optint (L, -1, DEFAULT_AP_BEACON);
SET_SAVE_MODE(save);
esp_err_t err = esp_wifi_set_config (WIFI_IF_AP, &cfg);
return (err == ESP_OK) ?
0 : luaL_error (L, "failed to set wifi config, code %d", err);
}
static int wifi_ap_getmac (lua_State *L)
{
return wifi_getmac(WIFI_IF_AP, L);
}
static int wifi_ap_on (lua_State *L)
{
return wifi_on (L, events, ARRAY_LEN(events), event_cb);
}
LROT_PUBLIC_BEGIN(wifi_ap)
LROT_FUNCENTRY( setip, wifi_ap_setip )
LROT_FUNCENTRY( sethostname, wifi_ap_sethostname )
LROT_FUNCENTRY( config, wifi_ap_config )
LROT_FUNCENTRY( on, wifi_ap_on )
LROT_FUNCENTRY( getmac, wifi_ap_getmac )
LROT_END(wifi_ap, NULL, 0)