initial commit :p
This commit is contained in:
commit
ce65897102
14
.cargo/config.toml
Normal file
14
.cargo/config.toml
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
[target.xtensa-esp32-none-elf]
|
||||||
|
runner = "espflash flash --monitor"
|
||||||
|
|
||||||
|
[env]
|
||||||
|
ESP_LOG = "TRACE"
|
||||||
|
ESP_LOGLEVEL = "TRACE"
|
||||||
|
|
||||||
|
[build]
|
||||||
|
rustflags = ["-C", "link-arg=-nostartfiles"]
|
||||||
|
|
||||||
|
target = "xtensa-esp32-none-elf"
|
||||||
|
|
||||||
|
[unstable]
|
||||||
|
build-std = ["alloc", "core"]
|
17
.gitignore
vendored
Normal file
17
.gitignore
vendored
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# Generated by Cargo
|
||||||
|
# will have compiled files and executables
|
||||||
|
debug/
|
||||||
|
target/
|
||||||
|
|
||||||
|
# These are backup files generated by rustfmt
|
||||||
|
**/*.rs.bk
|
||||||
|
|
||||||
|
# MSVC Windows builds of rustc generate these, which store debugging information
|
||||||
|
*.pdb
|
||||||
|
|
||||||
|
# RustRover
|
||||||
|
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||||
|
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||||
|
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||||
|
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||||
|
#.idea/
|
17
.lazy.lua
Normal file
17
.lazy.lua
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
return {
|
||||||
|
"williamboman/mason-lspconfig.nvim",
|
||||||
|
config = function()
|
||||||
|
local nvim_lsp = require("lspconfig")
|
||||||
|
|
||||||
|
nvim_lsp.rust_analyzer.setup({
|
||||||
|
settings = {
|
||||||
|
["rust-analyzer"] = {
|
||||||
|
cargo = {
|
||||||
|
allTargets = false,
|
||||||
|
target = "xtensa-esp32-none-elf",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
end,
|
||||||
|
}
|
73
Cargo.toml
Normal file
73
Cargo.toml
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
[package]
|
||||||
|
name = "evil_wifi"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[[bin]]
|
||||||
|
name = "evil_wifi"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
esp-backtrace = { version = "0.14.2", features = [
|
||||||
|
"esp32",
|
||||||
|
"exception-handler",
|
||||||
|
"panic-handler",
|
||||||
|
"println",
|
||||||
|
] }
|
||||||
|
|
||||||
|
esp-hal = { version = "0.22.0", features = ["esp32"] }
|
||||||
|
esp-println = { version = "0.12.0", features = ["esp32", "log"] }
|
||||||
|
log = { version = "0.4.21" }
|
||||||
|
esp-alloc = { version = "0.5.0" }
|
||||||
|
embedded-io = "0.6.1"
|
||||||
|
|
||||||
|
embedded-io-async = "0.6.1"
|
||||||
|
embassy-net = { version = "0.4.0", features = [
|
||||||
|
"tcp",
|
||||||
|
"udp",
|
||||||
|
"dhcpv4",
|
||||||
|
"medium-ethernet",
|
||||||
|
] }
|
||||||
|
|
||||||
|
esp-wifi = { version = "0.11.0", default-features = false, features = [
|
||||||
|
"esp32",
|
||||||
|
"utils",
|
||||||
|
"wifi",
|
||||||
|
"esp-alloc",
|
||||||
|
"log",
|
||||||
|
] }
|
||||||
|
heapless = { version = "0.8.0", default-features = false }
|
||||||
|
smoltcp = { version = "0.11.0", default-features = false, features = [
|
||||||
|
"medium-ethernet",
|
||||||
|
"proto-dhcpv4",
|
||||||
|
"proto-igmp",
|
||||||
|
"proto-ipv4",
|
||||||
|
"socket-dhcpv4",
|
||||||
|
"socket-icmp",
|
||||||
|
"socket-raw",
|
||||||
|
"socket-tcp",
|
||||||
|
"socket-udp",
|
||||||
|
] }
|
||||||
|
embassy-executor = { version = "0.6.0", features = ["nightly"] }
|
||||||
|
embassy-time = { version = "0.3.1", features = ["generic-queue-8"] }
|
||||||
|
esp-hal-embassy = { version = "0.5.0", features = ["esp32"] }
|
||||||
|
static_cell = { version = "2.1.0", features = ["nightly"] }
|
||||||
|
edge-nal = "0.3.0"
|
||||||
|
edge-dhcp = "0.3.0"
|
||||||
|
edge-mdns = "0.3.1"
|
||||||
|
edge-nal-embassy = "0.3.0"
|
||||||
|
edge-captive = "0.3.0"
|
||||||
|
ufmt = "0.2.0"
|
||||||
|
|
||||||
|
[profile.dev]
|
||||||
|
# Rust debug is too slow.
|
||||||
|
# For debug builds always builds with some optimization
|
||||||
|
opt-level = "s"
|
||||||
|
|
||||||
|
[profile.release]
|
||||||
|
codegen-units = 1 # LLVM can perform better optimizations using a single thread
|
||||||
|
debug = 2
|
||||||
|
debug-assertions = false
|
||||||
|
incremental = false
|
||||||
|
lto = 'fat'
|
||||||
|
opt-level = 's'
|
||||||
|
overflow-checks = false
|
3
build.rs
Normal file
3
build.rs
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
fn main() {
|
||||||
|
println!("cargo:rustc-link-arg-bins=-Tlinkall.x");
|
||||||
|
}
|
2
rust-toolchain.toml
Normal file
2
rust-toolchain.toml
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
[toolchain]
|
||||||
|
channel = "esp"
|
34
src/captive_portal.rs
Normal file
34
src/captive_portal.rs
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
use embassy_net::Stack;
|
||||||
|
use esp_wifi::wifi::{WifiApDevice, WifiDevice};
|
||||||
|
|
||||||
|
#[embassy_executor::task]
|
||||||
|
pub async fn captive_task(stack: &'static Stack<WifiDevice<'static, WifiApDevice>>) -> ! {
|
||||||
|
use edge_captive::io::run;
|
||||||
|
use edge_nal_embassy::{Udp, UdpBuffers};
|
||||||
|
|
||||||
|
use core::net::{Ipv4Addr, SocketAddrV4};
|
||||||
|
|
||||||
|
let mut tx_buf = [0; 1500];
|
||||||
|
let mut rx_buf = [0; 1500];
|
||||||
|
|
||||||
|
let buffers = UdpBuffers::<2, 1024, 1024, 10>::new();
|
||||||
|
let unbound_socket = Udp::new(stack, &buffers);
|
||||||
|
|
||||||
|
log::info!("Running Captive Portal DNS on UDP port 53...");
|
||||||
|
|
||||||
|
loop {
|
||||||
|
match run(
|
||||||
|
&unbound_socket,
|
||||||
|
core::net::SocketAddr::V4(SocketAddrV4::new(Ipv4Addr::UNSPECIFIED, 53)),
|
||||||
|
&mut tx_buf,
|
||||||
|
&mut rx_buf,
|
||||||
|
core::net::Ipv4Addr::new(192, 168, 2, 1),
|
||||||
|
core::time::Duration::from_secs(60),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
Ok(()) => (),
|
||||||
|
Err(e) => log::warn!("{e:?}"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
54
src/dhcp.rs
Normal file
54
src/dhcp.rs
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
use core::str::FromStr;
|
||||||
|
|
||||||
|
use embassy_net::Stack;
|
||||||
|
use embassy_time::{Duration, Timer};
|
||||||
|
use esp_wifi::wifi::{WifiApDevice, WifiDevice};
|
||||||
|
|
||||||
|
#[embassy_executor::task]
|
||||||
|
pub async fn run_dhcp(
|
||||||
|
stack: &'static Stack<WifiDevice<'static, WifiApDevice>>,
|
||||||
|
gw_ip_addr: &'static str,
|
||||||
|
) {
|
||||||
|
use core::net::{Ipv4Addr, SocketAddrV4};
|
||||||
|
|
||||||
|
use edge_dhcp::{
|
||||||
|
io::{self, DEFAULT_SERVER_PORT},
|
||||||
|
server::{Server, ServerOptions},
|
||||||
|
};
|
||||||
|
use edge_nal::UdpBind;
|
||||||
|
use edge_nal_embassy::{Udp, UdpBuffers};
|
||||||
|
|
||||||
|
let ip = Ipv4Addr::from_str(gw_ip_addr).expect("dhcp task failed to parse gw ip");
|
||||||
|
|
||||||
|
let mut buf = [0u8; 1500];
|
||||||
|
|
||||||
|
let mut gw_buf = [Ipv4Addr::UNSPECIFIED];
|
||||||
|
|
||||||
|
let buffers = UdpBuffers::<3, 1024, 1024, 10>::new();
|
||||||
|
let unbound_socket = Udp::new(stack, &buffers);
|
||||||
|
let mut bound_socket = unbound_socket
|
||||||
|
.bind(core::net::SocketAddr::V4(SocketAddrV4::new(
|
||||||
|
Ipv4Addr::UNSPECIFIED,
|
||||||
|
DEFAULT_SERVER_PORT,
|
||||||
|
)))
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let mut opts = ServerOptions::new(ip, Some(&mut gw_buf));
|
||||||
|
|
||||||
|
let dns = [Ipv4Addr::from_str(&gw_ip_addr).unwrap()];
|
||||||
|
|
||||||
|
opts.dns = &dns;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
_ = io::server::run(
|
||||||
|
&mut Server::<64>::new(ip),
|
||||||
|
&opts,
|
||||||
|
&mut bound_socket,
|
||||||
|
&mut buf,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.inspect_err(|e| log::warn!("DHCP server error: {e:?}"));
|
||||||
|
Timer::after(Duration::from_millis(500)).await;
|
||||||
|
}
|
||||||
|
}
|
99
src/http.rs
Normal file
99
src/http.rs
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
use embassy_net::{tcp::TcpSocket, IpListenEndpoint, Stack};
|
||||||
|
use embassy_time::{with_timeout, Duration, Timer};
|
||||||
|
use esp_wifi::wifi::{WifiApDevice, WifiDevice};
|
||||||
|
|
||||||
|
#[embassy_executor::task]
|
||||||
|
pub async fn run_http(stack: &'static Stack<WifiDevice<'static, WifiApDevice>>) {
|
||||||
|
let mut rx_buffer = [0; 1536];
|
||||||
|
let mut tx_buffer = [0; 1536];
|
||||||
|
let mut connected_clients: u32 = 0;
|
||||||
|
// I would like a nicer solution for this
|
||||||
|
let mut buf: heapless::String<256> = heapless::String::new();
|
||||||
|
|
||||||
|
let mut socket = TcpSocket::new(&stack, &mut rx_buffer, &mut tx_buffer);
|
||||||
|
socket.set_timeout(Some(embassy_time::Duration::from_secs(1)));
|
||||||
|
'connection: loop {
|
||||||
|
log::info!("Wait for connection on HTTP");
|
||||||
|
let r = socket
|
||||||
|
.accept(IpListenEndpoint {
|
||||||
|
addr: None,
|
||||||
|
port: 80,
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
connected_clients += 1;
|
||||||
|
|
||||||
|
if let Err(e) = r {
|
||||||
|
log::error!("connect error: {:?}", e);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut buffer = [0u8; 1024];
|
||||||
|
let mut pos = 0;
|
||||||
|
'read: loop {
|
||||||
|
log::info!("Waiting for new data!");
|
||||||
|
buf.clear();
|
||||||
|
match with_timeout(Duration::from_secs(1), socket.read(&mut buffer)).await {
|
||||||
|
Ok(Ok(0)) => {
|
||||||
|
log::info!("read EOF");
|
||||||
|
break 'read;
|
||||||
|
}
|
||||||
|
Ok(Ok(len)) => {
|
||||||
|
let to_print =
|
||||||
|
unsafe { core::str::from_utf8_unchecked(&buffer[..(pos + len)]) };
|
||||||
|
|
||||||
|
if to_print.contains("\r\n\r\n") {
|
||||||
|
log::info!("Got HTML data probably: {}", to_print);
|
||||||
|
log::info!("HTML data over");
|
||||||
|
break 'read;
|
||||||
|
}
|
||||||
|
|
||||||
|
pos += len;
|
||||||
|
}
|
||||||
|
Ok(Err(e)) => {
|
||||||
|
log::info!("read error: {:?}", e);
|
||||||
|
continue 'connection;
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
log::info!(
|
||||||
|
"Timeout: {e:?} socket state: {:?}, aborting socket and fuck it",
|
||||||
|
socket.state()
|
||||||
|
);
|
||||||
|
socket.abort();
|
||||||
|
continue 'connection;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
log::info!("im outta reading phase");
|
||||||
|
|
||||||
|
{
|
||||||
|
use core::fmt::Write;
|
||||||
|
match write!(buf, "HTTP/1.0 200 OK\r\n\r\n<html><body style=\"font-size: 45px; text-align: center;\"><p style=\"paddin:200px\">Fra ma che cazzo ti connetti... coglione... \nSi sono connessi {} coglioni fin ora. Bravo!!</p></body></html>\r\n", connected_clients) {
|
||||||
|
Ok(()) => log::debug!("Formatted message succesfully"),
|
||||||
|
Err(e) => log::error!("{}", e),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
log::debug!("Our turn to write");
|
||||||
|
|
||||||
|
let r = {
|
||||||
|
use embedded_io_async::Write;
|
||||||
|
socket.write_all(&buf.as_bytes()).await
|
||||||
|
};
|
||||||
|
if let Err(e) = r {
|
||||||
|
log::error!("write error: {:?}", e);
|
||||||
|
}
|
||||||
|
log::debug!("Just wrote a bunch of stuff");
|
||||||
|
|
||||||
|
let r = socket.flush().await;
|
||||||
|
if let Err(e) = r {
|
||||||
|
log::error!("flush error: {:?}", e);
|
||||||
|
// log::info!("wrote {} bytes:{}", buf.len(), buf)
|
||||||
|
}
|
||||||
|
log::debug!("Just flushed the stuff");
|
||||||
|
Timer::after(Duration::from_millis(300)).await;
|
||||||
|
socket.close();
|
||||||
|
Timer::after(Duration::from_millis(300)).await;
|
||||||
|
|
||||||
|
socket.abort();
|
||||||
|
}
|
||||||
|
}
|
2
src/lib.rs
Normal file
2
src/lib.rs
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
#![no_std]
|
||||||
|
#![feature(impl_trait_in_assoc_type)]
|
156
src/main.rs
Normal file
156
src/main.rs
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
#![no_std]
|
||||||
|
#![no_main]
|
||||||
|
#![feature(impl_trait_in_assoc_type)]
|
||||||
|
|
||||||
|
mod captive_portal;
|
||||||
|
mod dhcp;
|
||||||
|
mod http;
|
||||||
|
|
||||||
|
use core::str::FromStr;
|
||||||
|
|
||||||
|
use captive_portal::captive_task;
|
||||||
|
use dhcp::run_dhcp;
|
||||||
|
use embassy_executor::Spawner;
|
||||||
|
use embassy_net::{Ipv4Address, Ipv4Cidr, Stack, StackResources, StaticConfigV4};
|
||||||
|
use embassy_time::{Duration, Timer};
|
||||||
|
use esp_alloc as _;
|
||||||
|
use esp_backtrace as _;
|
||||||
|
use esp_hal::{prelude::*, rng::Rng, timer::timg::TimerGroup};
|
||||||
|
use esp_wifi::{
|
||||||
|
init,
|
||||||
|
wifi::{
|
||||||
|
AccessPointConfiguration, Configuration, WifiApDevice, WifiController, WifiDevice,
|
||||||
|
WifiEvent, WifiState,
|
||||||
|
},
|
||||||
|
EspWifiController,
|
||||||
|
};
|
||||||
|
|
||||||
|
// When you are okay with using a nightly compiler it's better to use https://docs.rs/static_cell/2.1.0/static_cell/macro.make_static.html
|
||||||
|
macro_rules! mk_static {
|
||||||
|
($t:ty,$val:expr) => {{
|
||||||
|
static STATIC_CELL: static_cell::StaticCell<$t> = static_cell::StaticCell::new();
|
||||||
|
#[deny(unused_attributes)]
|
||||||
|
let x = STATIC_CELL.uninit().write(($val));
|
||||||
|
x
|
||||||
|
}};
|
||||||
|
}
|
||||||
|
|
||||||
|
const GW_IP_ADDR_ENV: &'static str = "192.168.2.1";
|
||||||
|
|
||||||
|
#[esp_hal_embassy::main]
|
||||||
|
async fn main(spawner: Spawner) -> ! {
|
||||||
|
// esp_println::logger::init_logger_from_env();
|
||||||
|
esp_println::logger::init_logger(log::LevelFilter::Trace);
|
||||||
|
|
||||||
|
log::info!("Loggin enabled, max level: {:?}", log::max_level());
|
||||||
|
log::error!("Error!");
|
||||||
|
log::warn!("Warn!");
|
||||||
|
log::info!("Info!");
|
||||||
|
log::debug!("Debug!");
|
||||||
|
log::trace!("Trace!");
|
||||||
|
let mut config = esp_hal::Config::default();
|
||||||
|
config.cpu_clock = CpuClock::max();
|
||||||
|
let peripherals = esp_hal::init(config);
|
||||||
|
|
||||||
|
esp_alloc::heap_allocator!(78 * 1024);
|
||||||
|
|
||||||
|
let timg0 = TimerGroup::new(peripherals.TIMG0);
|
||||||
|
let mut rng = Rng::new(peripherals.RNG);
|
||||||
|
|
||||||
|
let init = &*mk_static!(
|
||||||
|
EspWifiController<'static>,
|
||||||
|
init(timg0.timer0, rng.clone(), peripherals.RADIO_CLK).unwrap()
|
||||||
|
);
|
||||||
|
|
||||||
|
let wifi = peripherals.WIFI;
|
||||||
|
let (wifi_interface, controller) =
|
||||||
|
esp_wifi::wifi::new_with_mode(&init, wifi, WifiApDevice).unwrap();
|
||||||
|
|
||||||
|
let timg1 = TimerGroup::new(peripherals.TIMG1);
|
||||||
|
esp_hal_embassy::init(timg1.timer0);
|
||||||
|
|
||||||
|
let gw_ip_addr_str = GW_IP_ADDR_ENV;
|
||||||
|
let gw_ip_addr = Ipv4Address::from_str(gw_ip_addr_str).expect("failed to parse gateway ip");
|
||||||
|
|
||||||
|
let mut dns_servers: heapless::Vec<_, 3> = heapless::Vec::new();
|
||||||
|
dns_servers
|
||||||
|
.push(Ipv4Address::from_str(GW_IP_ADDR_ENV).unwrap())
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
log::info!("Configured dns: {:?}", dns_servers);
|
||||||
|
|
||||||
|
let config = embassy_net::Config::ipv4_static(StaticConfigV4 {
|
||||||
|
address: Ipv4Cidr::new(gw_ip_addr, 24),
|
||||||
|
gateway: Some(gw_ip_addr),
|
||||||
|
dns_servers,
|
||||||
|
});
|
||||||
|
|
||||||
|
let seed = (rng.random() as u64) << 32 | rng.random() as u64;
|
||||||
|
|
||||||
|
// Init network stack
|
||||||
|
let stack = &*mk_static!(
|
||||||
|
Stack<WifiDevice<'_, WifiApDevice>>,
|
||||||
|
Stack::new(
|
||||||
|
wifi_interface,
|
||||||
|
config,
|
||||||
|
mk_static!(StackResources<4>, StackResources::<4>::new()),
|
||||||
|
seed
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
spawner.spawn(connection(controller)).ok();
|
||||||
|
spawner.spawn(net_task(&stack)).ok();
|
||||||
|
spawner.spawn(run_dhcp(&stack, gw_ip_addr_str)).ok();
|
||||||
|
spawner.spawn(captive_task(&stack)).ok();
|
||||||
|
spawner.spawn(http::run_http(&stack)).ok();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
if stack.is_link_up() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
Timer::after(Duration::from_millis(500)).await;
|
||||||
|
}
|
||||||
|
|
||||||
|
while !stack.is_config_up() {
|
||||||
|
Timer::after(Duration::from_millis(100)).await
|
||||||
|
}
|
||||||
|
stack
|
||||||
|
.config_v4()
|
||||||
|
.inspect(|c| log::info!("ipv4 config: {c:?}"));
|
||||||
|
|
||||||
|
loop {
|
||||||
|
Timer::after(Duration::from_secs(10)).await;
|
||||||
|
log::info!("All going well :)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[embassy_executor::task]
|
||||||
|
async fn connection(mut controller: WifiController<'static>) {
|
||||||
|
log::info!("start connection task");
|
||||||
|
log::info!("Device capabilities: {:?}", controller.capabilities());
|
||||||
|
loop {
|
||||||
|
match esp_wifi::wifi::wifi_state() {
|
||||||
|
WifiState::ApStarted => {
|
||||||
|
// wait until we're no longer connected
|
||||||
|
controller.wait_for_event(WifiEvent::ApStop).await;
|
||||||
|
Timer::after(Duration::from_millis(5000)).await
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
if !matches!(controller.is_started(), Ok(true)) {
|
||||||
|
let client_config = Configuration::AccessPoint(AccessPointConfiguration {
|
||||||
|
ssid: "NON CONNETTERTI DIO CANE".try_into().unwrap(),
|
||||||
|
..Default::default()
|
||||||
|
});
|
||||||
|
controller.set_configuration(&client_config).unwrap();
|
||||||
|
log::info!("Starting wifi");
|
||||||
|
controller.start_async().await.unwrap();
|
||||||
|
log::info!("Wifi started!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[embassy_executor::task]
|
||||||
|
async fn net_task(stack: &'static Stack<WifiDevice<'static, WifiApDevice>>) -> ! {
|
||||||
|
stack.run().await
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user