1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
use embedded_nal::SocketAddr;
use heapless::{spsc::Queue, FnvIndexMap};

use crate::SocketHandle;

pub struct TcpListener<const N: usize, const L: usize> {
    handles: FnvIndexMap<SocketHandle, u16, N>,
    connections: FnvIndexMap<u16, Queue<(SocketHandle, SocketAddr), L>, N>,
}

impl<const N: usize, const L: usize> TcpListener<N, L> {
    pub fn new() -> Self {
        Self {
            handles: FnvIndexMap::new(),
            connections: FnvIndexMap::new(),
        }
    }

    pub fn bind(&mut self, handle: SocketHandle, port: u16) -> Result<(), ()> {
        if self.handles.contains_key(&handle) {
            return Err(());
        }

        self.handles.insert(handle, port).map_err(drop)?;
        self.connections.insert(port, Queue::new()).map_err(drop)?;

        Ok(())
    }

    pub fn incoming(&mut self, port: u16) -> Option<&mut Queue<(SocketHandle, SocketAddr), L>> {
        self.connections.get_mut(&port)
    }

    pub fn available(&mut self, handle: SocketHandle) -> Result<bool, ()> {
        let port = self.handles.get(&handle).ok_or(())?;
        Ok(!self.connections.get_mut(port).ok_or(())?.is_empty())
    }

    pub fn accept(&mut self, handle: SocketHandle) -> Result<(SocketHandle, SocketAddr), ()> {
        let port = self.handles.get(&handle).ok_or(())?;
        self.connections
            .get_mut(port)
            .ok_or(())?
            .dequeue()
            .ok_or(())
    }
}