Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for esp32 network driver scanning for access points #1165

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Added support for `registered_name` in `erlang:process_info/2` and `Process.info/2`
- Added `net:gethostname/0` on platforms with gethostname(3).
- Added `socket:getopt/2`
- Added `network:wifi_scan/0,1` to ESP32 network driver to scan available APs when in sta or sta+ap mode.

### Fixed
- ESP32: improved sntp sync speed from a cold boot.
Expand Down
51 changes: 48 additions & 3 deletions doc/src/network-programming-guide.md
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,9 @@ gotIp(IpInfo) ->
io:format("Got IP: ~p~n", [IpInfo]).

disconnected() ->
io:format("Disconnected from AP.~n").
io:format("Disconnected from AP, starting scan~n"),
{ok, {Num, NetworkList}} = network:wifi_scan(),
...
```

In a typical application, the network should be configured and an IP address should be acquired first, before starting clients or services that have a dependency on the network.
Expand All @@ -100,7 +102,50 @@ case network:wait_for_sta(Config, 15000) of
end
```

To obtain the signal strength (in decibels) of the connection to the associated access point use [`network:sta_rssi/0`](./apidocs/erlang/eavmlib/network.md#sta_rssi0).
### STA (or AP+STA) mode functions

Some functions are only available if the device is configured in STA or AP+STA mode.

#### `sta_rssi`

Once connected to an access point, the signal strength in decibel-milliwatts (dBm) of the connection to the associated access point may be obtained using [`network:sta_rssi/0`](./apidocs/erlang/eavmlib/network.md#sta_rssi0). The value returned as `{ok, Value}` will typically be a negative number, but in the presence of a powerful signal this can be a positive number. A level of 0 dBm corresponds to the power of 1 milliwatt. A 10 dBm decrease in level is equivalent to a ten-fold decrease in signal power.

#### `wifi_scan`

```{notice}
This function is currently only supported on the ESP32 platform.
```

After the network has been configured for STA mode and started, as long as no connection has been initiated or associated, you may scan for available access points using [`network:wifi_scan/0`](./apidocs/erlang/eavmlib/network.md#wifi_scan1) or [`network:wifi_scan/1`](./apidocs/erlang/eavmlib/network.md#wifi_scan1). Scanning for access points will temporarily inhibit other traffic on the access point network if it is in use, but should not cause any active connections to be dropped. With no options, a default 'active' scan, with a per-channel dwell time of 120ms will be used and will return network details for up to 6 access points. The return value for the scan takes the form of a tuple consisting of `{ok, Results}`, where `Results = {FoundAPs [NetworkList]}`. `FoundAPs` may be a number larger than the length of the NetworkList if more access points were discoverd than the number of results requested. The entries in the `NetworkList` take the form of `{SSID, [AP_Properties]}`. `SSID` is the name of the network, and the `AP_Properties` is a proplist with the keys `rssi` for the dBm signal strength of the access point, `authmode` value is the authentication method used by the network, `bssid` (a.k.a MAC address) of the access point, and the `channel` key for obtaining the primary channel for the network.

Example return results:
```erlang
{ok, {Num, Networks}} = network:wifi_scan(Config),
io:format("network scan found ~p networks.~n", [Num]),
lists:foreach(
fun(_Network = {SSID, [{rssi, DBm}, {authmode, Mode}, {bssid, BSSID}, {channel, Number}]}) ->
io:format(
"Network: ~p, BSSID: ~p, signal ~p dBm, Security: ~p, channel ~p~n",
[SSID, BSSID, DBm, Mode, Number]
)
end,
Networks
).
```

```{tip}
The bssid binary results can be conveniently formatted for logging using `binary:encode_hex/1`.
```

Due to network stack size limitations on the esp32 classic the maximum number of scan results that can be returned is 14. Other ESP32 family chips can return up to 20 networks in the scan results. The default scan is quite fast, and likely may not find all the available networks. Scans are quite configurable with `active` (the default) and `passive` modes. Options should take the form of a proplist. The per channel scan time can be changed with the `dwell` key, the channel dwell time can be set for up to 1500 ms. Passive scans are slower, as they always linger on each channel for the full dwell time. Passive mode can be used by simply adding `passive` to the configuration proplist. Keep in mind when choosing a dwell time that between each progressively scanned channel the device must return to the home channel for a short time (typically 30ms), but for scans with a dwell time of over 1000ms the home channel dwell time will increase to 60ms to help mitigate beacon timeout events. In some network configuration beacon timeout events may still occur, but should not lead to a dropped connection, and after the scan completes the device should receive the next beacon from the access point. The default of 6 access points in the returned `NetworkList` may be changed with the `results` key. By default hidden networks are ignored, but can be included in the results by adding `show_hidden` to the configuration.

For example to do a passive scan using and ESP32-C6, including hidden networks, using the longest allowed scan time and showing the maximum number of networks available use the following:

```erlang
{ok, Results} = network:wifi_scan([passive, {results, 20}, {dwell, 1500}, show_hidden]),
```

For convenience the default options used for `network:wifi_scan/0` may be configured along with the `sta_config()` used to start the network driver. For the corresponding configuration keys consult the [`network:scan_options()`](./apidocs/erlang/eavmlib/network.md#scan-options) type definition. You may also perform a scan without starting the network with a configuration. This will use the configuration `[{sta, [managed]}]`, which uses the default `disconnected` event callback that will always attempt to reconnect to the last network. This makes future use of `network:wifi_scan/0,1` impossible, and `network:disconnect/0` useless as the driver will immediately attempt to reconnect to the last network. This is mainly intended for quick testing purposes. For most applications that will use wifi scan results it is recommended to start the driver with a configuration that uses a custom callback function for `disconnected` events, so that the driver will remain idle and allow the use of scan results to decide if a connection should be made.

## AP mode

Expand All @@ -116,7 +161,7 @@ The `<ap-properties>` property list may contain the following entries:

If the SSID is omitted in configuration, the SSID name `atomvm-<hexmac>` will be created, where `<hexmac>` is the hexadecimal representation of the factory-assigned MAC address of the device. This name should be sufficiently unique to disambiguate it from other reachable ESP32 devices, but it may also be difficult to read or remember.

If the password is omitted, then an _open network_ will be created, and a warning will be printed to the console. Otherwise, the AP network will be started using WPA+WPA2 authentication.
If the password is omitted, then an __open network__ will be created, and a warning will be printed to the console. Otherwise, the AP network will be started using WPA+WPA2 authentication.

If the channel is omitted the default chanel for esp32 is `1`. This setting is only used while a device is operation is AP mode only. If `ap_channel` is configured, it will be temporarily changed to match the associated access point if AP + STA mode is used and the station is associated with an access point. This is a hardware limitation due to the modem radio only being able to operate on a single channel (frequency) at a time.

Expand Down
1 change: 1 addition & 0 deletions examples/erlang/esp32/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -37,3 +37,4 @@ pack_runnable(sx127x sx127x eavmlib estdlib)
pack_runnable(reformat_nvs reformat_nvs eavmlib)
pack_runnable(uartecho uartecho eavmlib estdlib)
pack_runnable(ledc_example ledc_example eavmlib estdlib)
pack_runnable(wifi_scan wifi_scan estdlib eavmlib)
70 changes: 70 additions & 0 deletions examples/erlang/esp32/wifi_scan.erl
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
%% This file is part of AtomVM.
%%
%% Copyright (c) 2023 <[email protected]>
%% All rights reserved.
%%
%% Licensed under the Apache License, Version 2.0 (the "License");
%% you may not use this file except in compliance with the License.
%% You may obtain a copy of the License at
%%
%% http://www.apache.org/licenses/LICENSE-2.0
%%
%% Unless required by applicable law or agreed to in writing, software
%% distributed under the License is distributed on an "AS IS" BASIS,
%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
%% See the License for the specific language governing permissions and
%% limitations under the License.
%%
%%
%% SPDX-License-Identifier: Apache-2.0 OR LGPL-2.1-or-later
%%

-module(wifi_scan).

-export([start/0]).

start() ->
scan_passive([show_hidden, {dwell, 1000}]),
scan_active([{dwell, 500}]).

scan_active(Config) ->
io:format(
"~nStarting active scan with configuration ~p, this may take some time depending on dwell ms used.~n~n",
[Config]
),
BeginTime = erlang:monotonic_time(millisecond),
{ok, {Num, Networks}} = network:wifi_scan(Config),
io:format("Active scan found ~p networks in ~pms.~n", [
Num, erlang:monotonic_time(millisecond) - BeginTime
]),
lists:foreach(
fun(_Network = {SSID, [{rssi, DBm}, {authmode, Mode}, {bssid, BSSID}, {channel, Number}]}) ->
io:format(
"Network: ~p, BSSID: ~p, signal ~p dBm, Security: ~p, channel ~p~n",
[SSID, binary:encode_hex(BSSID), DBm, Mode, Number]
)
end,
Networks
).

scan_passive(Config) ->
io:format(
"~nStarting passive scan with configuration: ~p, this may take some time depending on dwell ms used.~n~n",
[Config]
),
Opts = lists:flatten([passive | Config]),
BeginTime = erlang:monotonic_time(millisecond),
ScanResults = network:wifi_scan(Opts),
{ok, {Num, Networks}} = ScanResults,
io:format("Passive scan found ~p networks in ~pms.~n", [
Num, erlang:monotonic_time(millisecond) - BeginTime
]),
lists:foreach(
fun(_Network = {SSID, [{rssi, DBm}, {authmode, Mode}, {bssid, BSSID}, {channel, Number}]}) ->
io:format(
"Network: ~p, BSSID: ~p, signal ~p dBm, Security: ~p, channel ~p~n",
[SSID, binary:encode_hex(BSSID), DBm, Mode, Number]
)
end,
Networks
).
Loading
Loading