/* * 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 */ #include "module.h" #include "lauxlib.h" #include "lextra.h" #include "wifi_common.h" #include "ip_fmt.h" #include "nodemcu_esp_event.h" #include #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)