Redo networking (#22)

* Rewriting network functions

* Add buffer to dup
Fix non-blocking handling by triggering once on enabling events to read to EOF

* Modifications for UDP API

* Implement TCP client side

* Add active close

* Add DMAR parser

* Implement basic TCP listening. Need to improve the state machine

* Reduce debugging

* Fixes for close procedure

* Updates to fix path processing in libstd
This commit is contained in:
Jeremy Soller 2016-10-26 13:19:56 -06:00 committed by GitHub
parent 268c859fd6
commit 2491e4771e
54 changed files with 1884 additions and 1476 deletions

View file

@ -3,6 +3,7 @@ name = "udpd"
version = "0.1.0"
[dependencies]
event = { path = "../../crates/event/" }
netutils = { path = "../../programs/netutils/" }
resource_scheme = { path = "../../crates/resource_scheme/" }
rand = { git = "https://github.com/redox-os/rand.git" }
syscall = { path = "../../syscall/" }

View file

@ -1,30 +1,348 @@
#![feature(rand)]
extern crate event;
extern crate netutils;
extern crate resource_scheme;
extern crate rand;
extern crate syscall;
use rand::{Rng, OsRng};
use std::collections::{BTreeMap, VecDeque};
use std::cell::RefCell;
use std::fs::File;
use std::io::{Read, Write};
use std::thread;
use std::io::{self, Read, Write};
use std::{mem, slice, str, thread};
use std::os::unix::io::FromRawFd;
use std::rc::Rc;
use resource_scheme::ResourceScheme;
use syscall::Packet;
use event::EventQueue;
use netutils::{n16, Ipv4, Ipv4Addr, Ipv4Header, Udp, UdpHeader, Checksum};
use syscall::data::Packet;
use syscall::error::{Error, Result, EACCES, EADDRINUSE, EBADF, EINVAL, EMSGSIZE, ENOTCONN, EWOULDBLOCK};
use syscall::flag::{EVENT_READ, O_CREAT, O_RDWR, O_NONBLOCK};
use syscall::scheme::SchemeMut;
use scheme::UdpScheme;
fn parse_socket(socket: &str) -> (Ipv4Addr, u16) {
let mut socket_parts = socket.split(":");
let host = Ipv4Addr::from_str(socket_parts.next().unwrap_or(""));
let port = socket_parts.next().unwrap_or("").parse::<u16>().unwrap_or(0);
(host, port)
}
mod resource;
mod scheme;
struct Handle {
local: (Ipv4Addr, u16),
remote: (Ipv4Addr, u16),
flags: usize,
events: usize,
data: VecDeque<Vec<u8>>,
todo: VecDeque<Packet>,
}
struct Udpd {
scheme_file: File,
udp_file: File,
ports: BTreeMap<u16, usize>,
next_id: usize,
handles: BTreeMap<usize, Handle>,
rng: OsRng,
}
impl Udpd {
fn new(scheme_file: File, udp_file: File) -> Self {
Udpd {
scheme_file: scheme_file,
udp_file: udp_file,
ports: BTreeMap::new(),
next_id: 1,
handles: BTreeMap::new(),
rng: OsRng::new().expect("udpd: failed to open RNG")
}
}
fn scheme_event(&mut self) -> io::Result<()> {
loop {
let mut packet = Packet::default();
if self.scheme_file.read(&mut packet)? == 0 {
break;
}
let a = packet.a;
self.handle(&mut packet);
if packet.a == (-EWOULDBLOCK) as usize {
packet.a = a;
if let Some(mut handle) = self.handles.get_mut(&packet.b) {
handle.todo.push_back(packet);
}
} else {
self.scheme_file.write(&packet)?;
}
}
Ok(())
}
fn udp_event(&mut self) -> io::Result<()> {
loop {
let mut bytes = [0; 65536];
let count = self.udp_file.read(&mut bytes)?;
if count == 0 {
break;
}
if let Some(ip) = Ipv4::from_bytes(&bytes[.. count]) {
if let Some(udp) = Udp::from_bytes(&ip.data) {
for (id, handle) in self.handles.iter_mut() {
// Local address not set or IP dst matches or is broadcast
if (handle.local.0 == Ipv4Addr::NULL || ip.header.dst == handle.local.0 || ip.header.dst == Ipv4Addr::BROADCAST)
// Local port matches UDP dst
&& udp.header.dst.get() == handle.local.1
// Remote address not set or is broadcast, or IP src matches
&& (handle.remote.0 == Ipv4Addr::NULL || handle.remote.0 == Ipv4Addr::BROADCAST || ip.header.src == handle.remote.0)
// Remote port not set or UDP src matches
&& (handle.remote.1 == 0 || udp.header.src.get() == handle.remote.1)
{
handle.data.push_back(udp.data.clone());
while ! handle.todo.is_empty() && ! handle.data.is_empty() {
let mut packet = handle.todo.pop_front().unwrap();
let buf = unsafe { slice::from_raw_parts_mut(packet.c as *mut u8, packet.d) };
let data = handle.data.pop_front().unwrap();
let mut i = 0;
while i < buf.len() && i < data.len() {
buf[i] = data[i];
i += 1;
}
packet.a = i;
self.scheme_file.write(&packet)?;
}
if handle.events & EVENT_READ == EVENT_READ {
if let Some(data) = handle.data.get(0) {
self.scheme_file.write(&Packet {
id: 0,
pid: 0,
uid: 0,
gid: 0,
a: syscall::number::SYS_FEVENT,
b: *id,
c: EVENT_READ,
d: data.len()
})?;
}
}
}
}
}
}
}
Ok(())
}
}
impl SchemeMut for Udpd {
fn open(&mut self, url: &[u8], flags: usize, uid: u32, _gid: u32) -> Result<usize> {
let path = str::from_utf8(url).or(Err(Error::new(EINVAL)))?;
let mut parts = path.split("/");
let remote = parse_socket(parts.next().unwrap_or(""));
let mut local = parse_socket(parts.next().unwrap_or(""));
if local.1 == 0 {
local.1 = self.rng.gen_range(32768, 65535);
}
if local.1 <= 1024 && uid != 0 {
return Err(Error::new(EACCES));
}
if self.ports.contains_key(&local.1) {
return Err(Error::new(EADDRINUSE));
}
self.ports.insert(local.1, 1);
let id = self.next_id;
self.next_id += 1;
self.handles.insert(id, Handle {
local: local,
remote: remote,
flags: flags,
events: 0,
data: VecDeque::new(),
todo: VecDeque::new(),
});
Ok(id)
}
fn dup(&mut self, file: usize, buf: &[u8]) -> Result<usize> {
let mut handle = {
let handle = self.handles.get(&file).ok_or(Error::new(EBADF))?;
Handle {
local: handle.local,
remote: handle.remote,
flags: handle.flags,
events: 0,
data: handle.data.clone(),
todo: VecDeque::new(),
}
};
let path = str::from_utf8(buf).or(Err(Error::new(EINVAL)))?;
if handle.remote.0 == Ipv4Addr::NULL || handle.remote.1 == 0 {
handle.remote = parse_socket(path);
}
if let Some(mut port) = self.ports.get_mut(&handle.local.1) {
*port = *port + 1;
}
let id = self.next_id;
self.next_id += 1;
self.handles.insert(id, handle);
Ok(id)
}
fn read(&mut self, file: usize, buf: &mut [u8]) -> Result<usize> {
let mut handle = self.handles.get_mut(&file).ok_or(Error::new(EBADF))?;
if handle.remote.0 == Ipv4Addr::NULL || handle.remote.1 == 0 {
Err(Error::new(ENOTCONN))
} else if let Some(data) = handle.data.pop_front() {
let mut i = 0;
while i < buf.len() && i < data.len() {
buf[i] = data[i];
i += 1;
}
Ok(i)
} else if handle.flags & O_NONBLOCK == O_NONBLOCK {
Ok(0)
} else {
Err(Error::new(EWOULDBLOCK))
}
}
fn write(&mut self, file: usize, buf: &[u8]) -> Result<usize> {
let handle = self.handles.get(&file).ok_or(Error::new(EBADF))?;
if handle.remote.0 == Ipv4Addr::NULL || handle.remote.1 == 0 {
Err(Error::new(ENOTCONN))
} else if buf.len() >= 65507 {
Err(Error::new(EMSGSIZE))
} else {
let udp_data = buf.to_vec();
let udp = Udp {
header: UdpHeader {
src: n16::new(handle.local.1),
dst: n16::new(handle.remote.1),
len: n16::new((udp_data.len() + mem::size_of::<UdpHeader>()) as u16),
checksum: Checksum { data: 0 }
},
data: udp_data
};
let ip_data = udp.to_bytes();
let ip = Ipv4 {
header: Ipv4Header {
ver_hlen: 0x45,
services: 0,
len: n16::new((ip_data.len() + mem::size_of::<Ipv4Header>()) as u16),
id: n16::new(self.rng.gen()),
flags_fragment: n16::new(0),
ttl: 127,
proto: 0x11,
checksum: Checksum { data: 0 },
src: handle.local.0,
dst: handle.remote.0
},
options: Vec::new(),
data: ip_data
};
self.udp_file.write(&ip.to_bytes()).map_err(|err| err.into_sys()).and(Ok(buf.len()))
}
}
fn fevent(&mut self, file: usize, flags: usize) -> Result<usize> {
let mut handle = self.handles.get_mut(&file).ok_or(Error::new(EBADF))?;
handle.events = flags;
Ok(file)
}
fn fpath(&mut self, id: usize, buf: &mut [u8]) -> Result<usize> {
let handle = self.handles.get(&id).ok_or(Error::new(EBADF))?;
let path_string = format!("udp:{}:{}/{}:{}", handle.remote.0.to_string(), handle.remote.1, handle.local.0.to_string(), handle.local.1);
let path = path_string.as_bytes();
let mut i = 0;
while i < buf.len() && i < path.len() {
buf[i] = path[i];
i += 1;
}
Ok(i)
}
fn fsync(&mut self, file: usize) -> Result<usize> {
let _handle = self.handles.get(&file).ok_or(Error::new(EBADF))?;
Ok(0)
}
fn close(&mut self, file: usize) -> Result<usize> {
let handle = self.handles.remove(&file).ok_or(Error::new(EBADF))?;
let remove = if let Some(mut port) = self.ports.get_mut(&handle.local.1) {
*port = *port + 1;
*port == 0
} else {
false
};
if remove {
drop(self.ports.remove(&handle.local.1));
}
drop(handle);
Ok(0)
}
}
fn main() {
thread::spawn(move || {
let mut socket = File::create(":udp").expect("udpd: failed to create udp scheme");
let scheme = UdpScheme;
loop {
let mut packet = Packet::default();
socket.read(&mut packet).expect("udpd: failed to read events from udp scheme");
scheme.handle(&mut packet);
socket.write(&packet).expect("udpd: failed to write responses to udp scheme");
}
let scheme_fd = syscall::open(":udp", O_RDWR | O_CREAT | O_NONBLOCK).expect("udpd: failed to create :udp");
let scheme_file = unsafe { File::from_raw_fd(scheme_fd) };
let udp_fd = syscall::open("ip:11", O_RDWR | O_NONBLOCK).expect("udpd: failed to open ip:11");
let udp_file = unsafe { File::from_raw_fd(udp_fd) };
let udpd = Rc::new(RefCell::new(Udpd::new(scheme_file, udp_file)));
let mut event_queue = EventQueue::<()>::new().expect("udpd: failed to create event queue");
let udp_udpd = udpd.clone();
event_queue.add(udp_fd, move |_count: usize| -> io::Result<Option<()>> {
udp_udpd.borrow_mut().udp_event()?;
Ok(None)
}).expect("udpd: failed to listen to events on ip:11");
event_queue.add(scheme_fd, move |_count: usize| -> io::Result<Option<()>> {
udpd.borrow_mut().scheme_event()?;
Ok(None)
}).expect("udpd: failed to listen to events on :udp");
event_queue.trigger_all(0).expect("udpd: failed to trigger event queue");
event_queue.run().expect("udpd: failed to run event queue");
});
}

View file

@ -1,115 +0,0 @@
use std::{cmp, mem};
use netutils::{n16, Ipv4Addr, Checksum, Udp, UdpHeader};
use resource_scheme::Resource;
use syscall;
use syscall::error::*;
/// UDP resource
pub struct UdpResource {
pub ip: usize,
pub data: Vec<u8>,
pub host_addr: Ipv4Addr,
pub peer_addr: Ipv4Addr,
pub peer_port: u16,
pub host_port: u16,
}
impl Resource for UdpResource {
fn dup(&self) -> Result<Box<UdpResource>> {
match syscall::dup(self.ip) {
Ok(ip) => {
Ok(Box::new(UdpResource {
ip: ip,
data: self.data.clone(),
host_addr: self.host_addr,
peer_addr: self.peer_addr,
peer_port: self.peer_port,
host_port: self.host_port,
}))
}
Err(err) => Err(err),
}
}
fn path(&self, buf: &mut [u8]) -> Result<usize> {
let path_string = format!("udp:{}:{}/{}", self.peer_addr.to_string(), self.peer_port, self.host_port);
let path = path_string.as_bytes();
for (b, p) in buf.iter_mut().zip(path.iter()) {
*b = *p;
}
Ok(cmp::min(buf.len(), path.len()))
}
fn read(&mut self, buf: &mut [u8]) -> Result<usize> {
if ! self.data.is_empty() {
let mut bytes: Vec<u8> = Vec::new();
mem::swap(&mut self.data, &mut bytes);
// TODO: Allow splitting
let mut i = 0;
while i < buf.len() && i < bytes.len() {
buf[i] = bytes[i];
i += 1;
}
return Ok(i);
}
loop {
let mut bytes = [0; 65536];
let count = try!(syscall::read(self.ip, &mut bytes));
if let Some(datagram) = Udp::from_bytes(&bytes[..count]) {
if datagram.header.dst.get() == self.host_port &&
datagram.header.src.get() == self.peer_port {
// TODO: Allow splitting
let mut i = 0;
while i < buf.len() && i < datagram.data.len() {
buf[i] = datagram.data[i];
i += 1;
}
return Ok(i);
}
}
}
}
fn write(&mut self, buf: &[u8]) -> Result<usize> {
let mut udp = Udp {
header: UdpHeader {
src: n16::new(self.host_port),
dst: n16::new(self.peer_port),
len: n16::new((mem::size_of::<UdpHeader>() + buf.len()) as u16),
checksum: Checksum { data: 0 },
},
data: Vec::from(buf),
};
unsafe {
let proto = n16::new(0x11);
let datagram_len = n16::new((mem::size_of::<UdpHeader>() + udp.data.len()) as u16);
udp.header.checksum.data =
Checksum::compile(Checksum::sum((&self.host_addr as *const Ipv4Addr) as usize,
mem::size_of::<Ipv4Addr>()) +
Checksum::sum((&self.peer_addr as *const Ipv4Addr) as usize,
mem::size_of::<Ipv4Addr>()) +
Checksum::sum((&proto as *const n16) as usize,
mem::size_of::<n16>()) +
Checksum::sum((&datagram_len as *const n16) as usize,
mem::size_of::<n16>()) +
Checksum::sum((&udp.header as *const UdpHeader) as usize,
mem::size_of::<UdpHeader>()) +
Checksum::sum(udp.data.as_ptr() as usize, udp.data.len()));
}
syscall::write(self.ip, &udp.to_bytes()).and(Ok(buf.len()))
}
fn sync(&mut self) -> Result<usize> {
syscall::fsync(self.ip)
}
}

View file

@ -1,73 +0,0 @@
use std::rand;
use std::{str, u16};
use netutils::{getcfg, Ipv4Addr, Udp};
use resource_scheme::ResourceScheme;
use syscall;
use syscall::error::{Error, Result, ENOENT, EINVAL};
use syscall::flag::O_RDWR;
use resource::UdpResource;
/// UDP UdpScheme
pub struct UdpScheme;
impl ResourceScheme<UdpResource> for UdpScheme {
fn open_resource(&self, url: &[u8], _flags: usize, _uid: u32, _gid: u32) -> Result<Box<UdpResource>> {
let ip_addr = Ipv4Addr::from_str(&getcfg("ip").map_err(|err| err.into_sys())?);
let path = try!(str::from_utf8(url).or(Err(Error::new(EINVAL))));
let mut parts = path.split('/');
let remote = parts.next().unwrap_or("");
let path = parts.next().unwrap_or("");
// Check host and port vs path
if remote.is_empty() {
let host_port = path.parse::<u16>().unwrap_or(0);
if host_port > 0 {
while let Ok(ip) = syscall::open("ip:/11", O_RDWR) {
let mut bytes = [0; 65536];
if let Ok(count) = syscall::read(ip, &mut bytes) {
if let Some(datagram) = Udp::from_bytes(&bytes[..count]) {
if datagram.header.dst.get() == host_port {
let mut path = [0; 256];
if let Ok(path_count) = syscall::fpath(ip, &mut path) {
let ip_reference = unsafe { str::from_utf8_unchecked(&path[.. path_count]) }.split(':').nth(1).unwrap_or("");
let peer_addr = ip_reference.split('/').next().unwrap_or("").split(':').next().unwrap_or("");
return Ok(Box::new(UdpResource {
ip: ip,
data: datagram.data,
host_addr: ip_addr,
peer_addr: Ipv4Addr::from_str(peer_addr),
peer_port: datagram.header.src.get(),
host_port: host_port,
}));
}
}
}
}
}
}
} else {
let mut remote_parts = remote.split(':');
let peer_addr = remote_parts.next().unwrap_or("");
let peer_port = remote_parts.next().unwrap_or("").parse::<u16>().unwrap_or(0);
if peer_port > 0 {
let host_port = path.parse::<u16>().unwrap_or((rand() % 32768 + 32768) as u16);
if let Ok(ip) = syscall::open(&format!("ip:{}/11", peer_addr), O_RDWR) {
return Ok(Box::new(UdpResource {
ip: ip,
data: Vec::new(),
host_addr: ip_addr,
peer_addr: Ipv4Addr::from_str(peer_addr),
peer_port: peer_port as u16,
host_port: host_port,
}));
}
}
}
Err(Error::new(ENOENT))
}
}