2016-10-26 21:19:56 +02:00
|
|
|
extern crate event;
|
2016-10-21 01:49:54 +02:00
|
|
|
extern crate netutils;
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
extern crate syscall;
|
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
use event::EventQueue;
|
|
|
|
use netutils::{getcfg, n16, Ipv4Addr, MacAddr, Ipv4, EthernetII, EthernetIIHeader, Arp, Tcp};
|
|
|
|
use std::cell::RefCell;
|
|
|
|
use std::collections::{BTreeMap, VecDeque};
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
use std::fs::File;
|
2016-10-26 21:19:56 +02:00
|
|
|
use std::io::{self, Read, Write};
|
|
|
|
use std::os::unix::io::FromRawFd;
|
|
|
|
use std::rc::Rc;
|
2016-11-11 04:02:51 +01:00
|
|
|
use std::{slice, str};
|
2016-10-26 21:19:56 +02:00
|
|
|
use syscall::data::Packet;
|
2016-10-31 17:49:00 +01:00
|
|
|
use syscall::error::{Error, Result, EACCES, EADDRNOTAVAIL, EBADF, EIO, EINVAL, ENOENT, EWOULDBLOCK};
|
2016-10-26 21:19:56 +02:00
|
|
|
use syscall::flag::{EVENT_READ, O_NONBLOCK};
|
|
|
|
use syscall::scheme::SchemeMut;
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
struct Interface {
|
|
|
|
mac: MacAddr,
|
|
|
|
ip: Ipv4Addr,
|
|
|
|
router: Ipv4Addr,
|
|
|
|
subnet: Ipv4Addr,
|
|
|
|
arp_file: File,
|
|
|
|
ip_file: File,
|
|
|
|
arp: BTreeMap<Ipv4Addr, MacAddr>,
|
|
|
|
rarp: BTreeMap<MacAddr, Ipv4Addr>,
|
|
|
|
}
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
impl Interface {
|
|
|
|
fn new(arp_fd: usize, ip_fd: usize) -> Self {
|
|
|
|
Interface {
|
|
|
|
mac: MacAddr::from_str(&getcfg("mac").unwrap()),
|
|
|
|
ip: Ipv4Addr::from_str(&getcfg("ip").unwrap()),
|
|
|
|
router: Ipv4Addr::from_str(&getcfg("ip_router").unwrap()),
|
|
|
|
subnet: Ipv4Addr::from_str(&getcfg("ip_subnet").unwrap()),
|
|
|
|
arp_file: unsafe { File::from_raw_fd(arp_fd) },
|
|
|
|
ip_file: unsafe { File::from_raw_fd(ip_fd) },
|
|
|
|
arp: BTreeMap::new(),
|
|
|
|
rarp: BTreeMap::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
struct Handle {
|
|
|
|
proto: u8,
|
|
|
|
flags: usize,
|
|
|
|
events: usize,
|
|
|
|
data: VecDeque<Vec<u8>>,
|
|
|
|
todo: VecDeque<Packet>,
|
|
|
|
}
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
struct Ipd {
|
|
|
|
scheme_file: File,
|
|
|
|
interfaces: Vec<Interface>,
|
|
|
|
next_id: usize,
|
|
|
|
handles: BTreeMap<usize, Handle>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ipd {
|
|
|
|
fn new(scheme_file: File) -> Self {
|
|
|
|
Ipd {
|
|
|
|
scheme_file: scheme_file,
|
|
|
|
interfaces: Vec::new(),
|
|
|
|
next_id: 1,
|
|
|
|
handles: BTreeMap::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn scheme_event(&mut self) -> io::Result<()> {
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
loop {
|
|
|
|
let mut packet = Packet::default();
|
2016-10-26 21:19:56 +02:00
|
|
|
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)?;
|
|
|
|
}
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
}
|
2016-10-26 21:19:56 +02:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn arp_event(&mut self, if_id: usize) -> io::Result<()> {
|
|
|
|
if let Some(mut interface) = self.interfaces.get_mut(if_id) {
|
|
|
|
loop {
|
|
|
|
let mut bytes = [0; 65536];
|
|
|
|
let count = interface.arp_file.read(&mut bytes)?;
|
|
|
|
if count == 0 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if let Some(frame) = EthernetII::from_bytes(&bytes[.. count]) {
|
|
|
|
if let Some(packet) = Arp::from_bytes(&frame.data) {
|
|
|
|
if packet.header.oper.get() == 1 {
|
|
|
|
if packet.header.dst_ip == interface.ip {
|
|
|
|
let mut response = Arp {
|
|
|
|
header: packet.header,
|
|
|
|
data: packet.data.clone(),
|
|
|
|
};
|
|
|
|
response.header.oper.set(2);
|
|
|
|
response.header.dst_mac = packet.header.src_mac;
|
|
|
|
response.header.dst_ip = packet.header.src_ip;
|
|
|
|
response.header.src_mac = interface.mac;
|
|
|
|
response.header.src_ip = interface.ip;
|
|
|
|
|
|
|
|
let mut response_frame = EthernetII {
|
|
|
|
header: frame.header,
|
|
|
|
data: response.to_bytes()
|
|
|
|
};
|
|
|
|
|
|
|
|
response_frame.header.dst = response_frame.header.src;
|
|
|
|
response_frame.header.src = interface.mac;
|
|
|
|
|
|
|
|
interface.arp_file.write(&response_frame.to_bytes())?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ip_event(&mut self, if_id: usize) -> io::Result<()> {
|
|
|
|
if let Some(mut interface) = self.interfaces.get_mut(if_id) {
|
|
|
|
loop {
|
|
|
|
let mut bytes = [0; 65536];
|
|
|
|
let count = interface.ip_file.read(&mut bytes)?;
|
|
|
|
if count == 0 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if let Some(frame) = EthernetII::from_bytes(&bytes[.. count]) {
|
|
|
|
if let Some(ip) = Ipv4::from_bytes(&frame.data) {
|
|
|
|
if ip.header.dst == interface.ip || ip.header.dst == Ipv4Addr::BROADCAST {
|
|
|
|
//TODO: Handle ping here
|
|
|
|
for (id, handle) in self.handles.iter_mut() {
|
|
|
|
if ip.header.proto == handle.proto {
|
|
|
|
handle.data.push_back(frame.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(())
|
|
|
|
}
|
Orbital (#16)
* Port previous ethernet scheme
* Add ipd
* Fix initfs rebuilds, use QEMU user networking addresses in ipd
* Add tcp/udp, netutils, dns, and network config
* Add fsync to network driver
* Add dns, router, subnet by default
* Fix e1000 driver. Make ethernet and IP non-blocking to avoid deadlocks
* Add orbital server, WIP
* Add futex
* Add orbutils and orbital
* Update libstd, orbutils, and orbital
Move ANSI key encoding to vesad
* Add orbital assets
* Update orbital
* Update to add login manager
* Add blocking primitives, block for most things except waitpid, update orbital
* Wait in waitpid and IRQ, improvements for other waits
* Fevent in root scheme
* WIP: Switch to using fevent
* Reorganize
* Event based e1000d driver
* Superuser-only access to some network schemes, display, and disk
* Superuser root and irq schemes
* Fix orbital
2016-10-14 01:21:42 +02:00
|
|
|
}
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
impl SchemeMut for Ipd {
|
|
|
|
fn open(&mut self, url: &[u8], flags: usize, uid: u32, _gid: u32) -> Result<usize> {
|
|
|
|
if uid == 0 {
|
|
|
|
let path = str::from_utf8(url).or(Err(Error::new(EINVAL)))?;
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
let proto = u8::from_str_radix(path, 16).or(Err(Error::new(ENOENT)))?;
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
let id = self.next_id;
|
|
|
|
self.next_id += 1;
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
self.handles.insert(id, Handle {
|
|
|
|
proto: proto,
|
|
|
|
flags: flags,
|
|
|
|
events: 0,
|
|
|
|
data: VecDeque::new(),
|
|
|
|
todo: VecDeque::new(),
|
|
|
|
});
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
Ok(id)
|
|
|
|
} else {
|
|
|
|
Err(Error::new(EACCES))
|
|
|
|
}
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
fn dup(&mut self, file: usize, _buf: &[u8]) -> Result<usize> {
|
|
|
|
let handle = {
|
|
|
|
let handle = self.handles.get(&file).ok_or(Error::new(EBADF))?;
|
|
|
|
Handle {
|
|
|
|
proto: handle.proto,
|
|
|
|
flags: handle.flags,
|
|
|
|
events: 0,
|
|
|
|
data: handle.data.clone(),
|
|
|
|
todo: VecDeque::new(),
|
|
|
|
}
|
|
|
|
};
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
let id = self.next_id;
|
|
|
|
self.next_id += 1;
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
self.handles.insert(id, handle);
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
Ok(id)
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
fn read(&mut self, file: usize, buf: &mut [u8]) -> Result<usize> {
|
|
|
|
let mut handle = self.handles.get_mut(&file).ok_or(Error::new(EBADF))?;
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
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;
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
Ok(i)
|
|
|
|
} else if handle.flags & O_NONBLOCK == O_NONBLOCK {
|
|
|
|
Ok(0)
|
|
|
|
} else {
|
|
|
|
Err(Error::new(EWOULDBLOCK))
|
|
|
|
}
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
fn write(&mut self, file: usize, buf: &[u8]) -> Result<usize> {
|
|
|
|
let handle = self.handles.get(&file).ok_or(Error::new(EBADF))?;
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
if let Some(mut ip) = Ipv4::from_bytes(buf) {
|
|
|
|
for mut interface in self.interfaces.iter_mut() {
|
|
|
|
if ip.header.src == interface.ip || ip.header.src == Ipv4Addr::NULL {
|
|
|
|
ip.header.src = interface.ip;
|
|
|
|
ip.header.proto = handle.proto;
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
if let Some(mut tcp) = Tcp::from_bytes(&ip.data) {
|
|
|
|
tcp.checksum(&ip.header.src, &ip.header.dst);
|
|
|
|
ip.data = tcp.to_bytes();
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
ip.checksum();
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
let frame = EthernetII {
|
|
|
|
header: EthernetIIHeader {
|
|
|
|
//TODO: Get real dst
|
|
|
|
dst: MacAddr::BROADCAST,
|
|
|
|
src: interface.mac,
|
|
|
|
ethertype: n16::new(0x800),
|
|
|
|
},
|
|
|
|
data: ip.to_bytes()
|
|
|
|
};
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-31 17:49:00 +01:00
|
|
|
interface.ip_file.write(&frame.to_bytes()).map_err(|err| Error::new(err.raw_os_error().unwrap_or(EIO)))?;
|
2016-10-24 22:17:12 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
return Ok(buf.len());
|
|
|
|
}
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
Err(Error::new(EADDRNOTAVAIL))
|
|
|
|
} else {
|
|
|
|
Err(Error::new(EINVAL))
|
|
|
|
}
|
|
|
|
}
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
fn fevent(&mut self, file: usize, flags: usize) -> Result<usize> {
|
|
|
|
let mut handle = self.handles.get_mut(&file).ok_or(Error::new(EBADF))?;
|
2016-10-24 23:02:57 +02:00
|
|
|
|
2016-10-26 21:19:56 +02:00
|
|
|
handle.events = flags;
|
|
|
|
|
|
|
|
Ok(file)
|
2016-10-24 22:17:12 +02:00
|
|
|
}
|
2016-10-26 21:19:56 +02:00
|
|
|
|
|
|
|
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!("ip:{:X}", handle.proto);
|
|
|
|
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)
|
2016-10-24 22:17:12 +02:00
|
|
|
}
|
2016-10-26 21:19:56 +02:00
|
|
|
|
|
|
|
fn close(&mut self, file: usize) -> Result<usize> {
|
|
|
|
let handle = self.handles.remove(&file).ok_or(Error::new(EBADF))?;
|
|
|
|
|
|
|
|
drop(handle);
|
|
|
|
|
|
|
|
Ok(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
2016-11-11 04:02:51 +01:00
|
|
|
// Daemonize
|
|
|
|
if unsafe { syscall::clone(0).unwrap() } == 0 {
|
2016-10-26 21:19:56 +02:00
|
|
|
let scheme_fd = syscall::open(":ip", syscall::O_RDWR | syscall::O_CREAT | syscall::O_NONBLOCK).expect("ipd: failed to create :ip");
|
|
|
|
let scheme_file = unsafe { File::from_raw_fd(scheme_fd) };
|
|
|
|
|
|
|
|
let ipd = Rc::new(RefCell::new(Ipd::new(scheme_file)));
|
|
|
|
|
|
|
|
let mut event_queue = EventQueue::<()>::new().expect("ipd: failed to create event queue");
|
|
|
|
|
|
|
|
//TODO: Multiple interfaces
|
|
|
|
{
|
|
|
|
let arp_fd = syscall::open("ethernet:806", syscall::O_RDWR | syscall::O_NONBLOCK).expect("ipd: failed to open ethernet:806");
|
|
|
|
let ip_fd = syscall::open("ethernet:800", syscall::O_RDWR | syscall::O_NONBLOCK).expect("ipd: failed to open ethernet:800");
|
|
|
|
let if_id = {
|
|
|
|
let mut ipd = ipd.borrow_mut();
|
|
|
|
let if_id = ipd.interfaces.len();
|
|
|
|
ipd.interfaces.push(Interface::new(arp_fd, ip_fd));
|
|
|
|
if_id
|
|
|
|
};
|
|
|
|
|
|
|
|
let arp_ipd = ipd.clone();
|
|
|
|
event_queue.add(arp_fd, move |_count: usize| -> io::Result<Option<()>> {
|
|
|
|
arp_ipd.borrow_mut().arp_event(if_id)?;
|
|
|
|
Ok(None)
|
|
|
|
}).expect("ipd: failed to listen to events on ethernet:806");
|
|
|
|
|
|
|
|
let ip_ipd = ipd.clone();
|
|
|
|
event_queue.add(ip_fd, move |_count: usize| -> io::Result<Option<()>> {
|
|
|
|
ip_ipd.borrow_mut().ip_event(if_id)?;
|
|
|
|
Ok(None)
|
|
|
|
}).expect("ipd: failed to listen to events on ethernet:800");
|
|
|
|
}
|
|
|
|
|
|
|
|
event_queue.add(scheme_fd, move |_count: usize| -> io::Result<Option<()>> {
|
|
|
|
ipd.borrow_mut().scheme_event()?;
|
|
|
|
Ok(None)
|
|
|
|
}).expect("ipd: failed to listen to events on :ip");
|
|
|
|
|
|
|
|
// Make sure that all descriptors are at EOF
|
|
|
|
event_queue.trigger_all(0).expect("ipd: failed to trigger event queue");
|
|
|
|
|
|
|
|
event_queue.run().expect("ipd: failed to run event queue");
|
2016-11-11 04:02:51 +01:00
|
|
|
}
|
2016-10-26 21:19:56 +02:00
|
|
|
}
|