123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 |
- use core::str::{self, FromStr};
- use core::mem;
- use core::ptr;
- use libc::{c_int, c_char, size_t, c_void, ssize_t};
- use ::types::{socklen_t, in_addr, sockaddr, sockaddr_in};
- use syscall::{self, O_RDWR};
- use syscall::error::{Error, EPROTOTYPE, EPROTONOSUPPORT, EAFNOSUPPORT, EINVAL, EOPNOTSUPP, ENOBUFS, ENOSPC};
- use core::slice;
- use byteorder::{BigEndian, ByteOrder};
- pub const AF_INET: c_int = 2;
- pub const SOCK_STREAM: c_int = 1;
- pub const SOCK_DGRAM: c_int = 2;
- static mut NTOA_ADDR: [c_char; 16] = [0; 16];
- libc_fn!(unsafe inet_aton(cp: *const c_char, inp: *mut in_addr) -> c_int {
- // TODO: octal/hex
- inet_pton(AF_INET, cp, inp as *mut c_void)
- });
- libc_fn!(unsafe inet_ntoa(addr: in_addr) -> *const c_char {
- inet_ntop(AF_INET, &addr as *const in_addr as *const c_void, NTOA_ADDR.as_mut_ptr(), 16)
- });
- libc_fn!(unsafe inet_pton(domain: c_int, src: *const c_char, dest: *mut c_void) -> Result<c_int> {
- if domain != AF_INET {
- Err(Error::new(EAFNOSUPPORT))
- } else {
- let s_addr = &mut ((*(dest as *mut in_addr)).s_addr);
- let mut octets = str::from_utf8_unchecked(::cstr_to_slice(src)).split('.');
- for i in 0..4 {
- if let Some(n) = octets.next().and_then(|x| u8::from_str(x).ok()) {
- s_addr[i] = n;
- } else {
- return Ok(0);
- }
- }
- if octets.next() == None {
- Ok(1) // Success
- } else {
- Ok(0)
- }
- }
- });
- libc_fn!(unsafe inet_ntop(domain: c_int, src: *const c_void, dest: *mut c_char, size: socklen_t) -> Result<*const c_char> {
- if domain != AF_INET {
- Err(Error::new(EAFNOSUPPORT))
- } else if size < 16 {
- Err(Error::new(ENOSPC))
- } else {
- let s_addr = (&*(src as *const in_addr)).s_addr;
- let addr = format!("{}.{}.{}.{}\0", s_addr[0], s_addr[1], s_addr[2], s_addr[3]);
- ptr::copy(addr.as_ptr() as *const c_char, dest, addr.len());
- Ok(dest)
- }
- });
- libc_fn!(unsafe socket(domain: c_int, type_: c_int, protocol: c_int) -> Result<c_int> {
- if domain != AF_INET {
- Err(Error::new(EAFNOSUPPORT))
- } else if protocol != 0 {
- Err(Error::new(EPROTONOSUPPORT))
- } else {
- let fd = match type_ {
- SOCK_STREAM => syscall::open("tcp:", O_RDWR),
- SOCK_DGRAM => syscall::open("udp:", O_RDWR),
- _ => Err(Error::new(EPROTOTYPE))
- }?;
- Ok(fd as c_int)
- }
- });
- libc_fn!(unsafe connect(socket: c_int, address: *const sockaddr, _address_len: socklen_t) -> Result<c_int> {
- // XXX with UDP, should recieve messages only from that peer after this
- // XXX Check address_len
- if (*address).sa_family as i32 != AF_INET {
- return Err(Error::new(EINVAL))
- };
- let address = &*(address as *const sockaddr_in);
- let s_addr =address.sin_addr.s_addr;
- let path = format!("{}.{}.{}.{}:{}", s_addr[0], s_addr[1], s_addr[2], s_addr[3], ntohs(address.sin_port));
- let fd = syscall::dup(socket as usize, path.as_bytes())?;
- let ret = syscall::dup2(fd, socket as usize, &vec![]);
- let _ = syscall::close(fd);
- ret?;
- Ok(0)
- });
- libc_fn!(unsafe bind(socket: c_int, address: *const sockaddr, _address_len: socklen_t) -> Result<c_int> {
- // XXX Check address_len
- if (*address).sa_family as i32 != AF_INET {
- return Err(Error::new(EINVAL))
- };
- let address = &*(address as *const sockaddr_in);
- let s_addr =address.sin_addr.s_addr;
- let path = format!("/{}.{}.{}.{}:{}", s_addr[0], s_addr[1], s_addr[2], s_addr[3], ntohs(address.sin_port));
- let fd = syscall::dup(socket as usize, path.as_bytes())?;
- let ret = syscall::dup2(fd, socket as usize, &vec![]);
- let _ = syscall::close(fd);
- ret?;
- Ok(0)
- });
- libc_fn!(unsafe listen(_socket: c_int, _backlog: c_int) -> Result<c_int> {
- // TODO
- Ok(0)
- });
- libc_fn!(unsafe recv(socket: c_int, buffer: *mut c_void, length: size_t, flags: c_int) -> Result<ssize_t> {
- // XXX flags
- if flags != 0 {
- Err(Error::new(EOPNOTSUPP))
- } else {
- let buf = slice::from_raw_parts_mut(buffer as *mut u8, length);
- Ok(syscall::read(socket as usize, buf)? as ssize_t)
- }
- });
- libc_fn!(unsafe send(socket: c_int, buffer: *const c_void, length: size_t, flags: c_int) -> Result<ssize_t> {
- if flags != 0 {
- Err(Error::new(EOPNOTSUPP))
- } else {
- let buf = slice::from_raw_parts(buffer as *const u8, length);
- Ok(syscall::write(socket as usize, buf)? as ssize_t)
- }
- });
- libc_fn!(unsafe recvfrom(socket: c_int, buffer: *mut c_void, length: size_t, flags: c_int, _address: *const sockaddr, _address_len: *const socklen_t) -> Result<ssize_t> {
- let fd = syscall::dup(socket as usize, b"listen")?;
- let mut path = [0; 4096];
- syscall::fpath(socket as usize, &mut path)?;
- // XXX process path and write to address
- let ret = recv(socket, buffer, length, flags);
- syscall::close(fd)?;
- Ok(ret)
- });
- libc_fn!(unsafe sendto(socket: c_int, message: *const c_void, length: size_t, flags: c_int, dest_addr: *const sockaddr, _dest_len: socklen_t) -> Result<ssize_t> {
- // XXX test dest_len
- if (*dest_addr).sa_family as i32 == AF_INET {
- let addr = &*(dest_addr as *const sockaddr_in);
- let s_addr = addr.sin_addr.s_addr;
- let url = format!("{}.{}.{}.{}:{}", s_addr[0], s_addr[1], s_addr[2], s_addr[3], ntohs(addr.sin_port));
- let fd = syscall::dup(socket as usize, url.as_bytes())?;
- let ret = send(fd as c_int, message, length, flags);
- syscall::close(fd)?;
- Ok(ret)
- } else {
- Err(Error::new(EOPNOTSUPP))
- }
- });
- libc_fn!(unsafe getpeername(socket: c_int, address: *mut sockaddr, address_len: *mut socklen_t) -> Result<c_int> {
- // XXX will need to be changed for other sockaddr types
- if *address_len < mem::size_of::<sockaddr_in>() {
- return Err(Error::new(ENOBUFS));
- }
- *address_len = mem::size_of::<sockaddr_in>();
- let addr = &mut *(address as *mut sockaddr_in);
- addr.sin_family = AF_INET as u16;
- let mut path = [0; 4096];
- syscall::fpath(socket as usize, &mut path)?;
- let start;
- let sep;
- let end;
- {
- let mut iter = path.iter();
- start = iter.position(|x| *x == b':').ok_or(Error::new(EINVAL))? + 1;
- sep = start + iter.position(|x| *x == b':').ok_or(Error::new(EINVAL))?;
- end = sep + 1 + iter.position(|x| *x == b'/').ok_or(Error::new(EINVAL))?;
- }
- path[sep] = b'\0';
- if inet_aton(&path[start] as *const u8 as *const c_char, &mut addr.sin_addr) == 1 {
- if let Ok(port) = u16::from_str(str::from_utf8_unchecked(&path[sep+1..end])) {
- addr.sin_port = htons(port);
- Ok(0)
- } else {
- Err(Error::new(EINVAL))
- }
- } else {
- Err(Error::new(EINVAL)) // ?
- }
- });
- libc_fn!(unsafe getsockname(socket: c_int, address: *mut sockaddr, address_len: *mut socklen_t) -> Result<c_int> {
- // XXX will need to be changed for other sockaddr types
- if *address_len < mem::size_of::<sockaddr_in>() {
- return Err(Error::new(ENOBUFS));
- }
- *address_len = mem::size_of::<sockaddr_in>();
- let addr = &mut *(address as *mut sockaddr_in);
- addr.sin_family = AF_INET as u16;
- let mut path = [0; 4096];
- syscall::fpath(socket as usize, &mut path)?;
- let start;
- let sep;
- let end;
- {
- let mut iter = path.iter();
- start = iter.position(|x| *x == b'/').ok_or(Error::new(EINVAL))? + 1;
- sep = start + iter.position(|x| *x == b':').ok_or(Error::new(EINVAL))?;
- end = sep + 1 + iter.position(|x| *x == b'\0').ok_or(Error::new(EINVAL))?;
- }
- path[sep] = b'\0';
- if inet_aton(&path[start] as *const u8 as *const c_char, &mut addr.sin_addr) == 1 {
- if let Ok(port) = u16::from_str(str::from_utf8_unchecked(&path[sep+1..end])) {
- addr.sin_port = htons(port);
- Ok(0)
- } else {
- Err(Error::new(EINVAL))
- }
- } else {
- Err(Error::new(EINVAL)) // ?
- }
- });
- libc_fn!(htonl(hostlong: u32) -> [u8; 4] {
- let mut netlong = [0; 4];
- BigEndian::write_u32(&mut netlong, hostlong);
- netlong
- });
- libc_fn!(htons(hostshort: u16) -> [u8; 2] {
- let mut netshort = [0; 2];
- BigEndian::write_u16(&mut netshort, hostshort);
- netshort
- });
- libc_fn!(ntohl(netlong: [u8; 4]) -> u32 {
- BigEndian::read_u32(&netlong)
- });
- libc_fn!(ntohs(netshort: [u8; 2]) -> u16 {
- BigEndian::read_u16(&netshort)
- });
- libc_fn!(setsockopt(socket: c_int, level: c_int, option_name: c_int, option_value: *const c_void, option_len: socklen_t) -> Result<c_int> {
- syscall::write(2, format!("unimplemented: setsockopt({}, {}, {}, {:?}, {})\n",
- socket, level, option_name, option_value, option_len).as_bytes()).unwrap();
- Err(Error::new(syscall::ENOSYS))
- });
|