From d3f6985ee91677380da5d558133e4fbcfade5bbd Mon Sep 17 00:00:00 2001 From: jD91mZM2 <me@krake.one> Date: Sun, 8 Jul 2018 08:44:23 +0200 Subject: [PATCH] Add a few things necessary for openssl (not all) --- Cargo.lock | 9 ++ Cargo.toml | 1 + include/sys/param.h | 34 +++++++ src/ctype/src/lib.rs | 2 +- src/lib.rs | 1 + src/platform/src/lib.rs | 17 ++++ src/platform/src/linux/mod.rs | 24 ++++- src/platform/src/redox/mod.rs | 114 ++++++++++++++++++++++- src/stdio/src/printf.rs | 6 +- src/stdlib/src/lib.rs | 2 +- src/string/src/lib.rs | 15 ++- src/strings/Cargo.toml | 11 +++ src/strings/build.rs | 11 +++ src/strings/cbindgen.toml | 7 ++ src/strings/src/lib.rs | 136 ++++++++++++++++++++++++++++ src/sys_socket/src/lib.rs | 29 ++++-- tests/.gitignore | 1 + tests/Makefile | 3 +- tests/expected/string/strchr.stdout | 2 +- tests/expected/strings.stderr | 0 tests/expected/strings.stdout | 0 tests/string/strchr.c | 2 +- tests/strings.c | 30 ++++++ 23 files changed, 427 insertions(+), 30 deletions(-) create mode 100644 include/sys/param.h create mode 100644 src/strings/Cargo.toml create mode 100644 src/strings/build.rs create mode 100644 src/strings/cbindgen.toml create mode 100644 src/strings/src/lib.rs create mode 100644 tests/expected/strings.stderr create mode 100644 tests/expected/strings.stdout create mode 100644 tests/strings.c diff --git a/Cargo.lock b/Cargo.lock index a719fd585..bffba2450 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -304,6 +304,7 @@ dependencies = [ "stdio 0.1.0", "stdlib 0.1.0", "string 0.1.0", + "strings 0.1.0", "sys_mman 0.1.0", "sys_resource 0.1.0", "sys_socket 0.1.0", @@ -446,6 +447,14 @@ dependencies = [ "platform 0.1.0", ] +[[package]] +name = "strings" +version = "0.1.0" +dependencies = [ + "cbindgen 0.5.2", + "platform 0.1.0", +] + [[package]] name = "strsim" version = "0.7.0" diff --git a/Cargo.toml b/Cargo.toml index 951472027..db0493f13 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -30,6 +30,7 @@ signal = { path = "src/signal" } stdio = { path = "src/stdio" } stdlib = { path = "src/stdlib" } string = { path = "src/string" } +strings = { path = "src/strings" } sys_mman = { path = "src/sys_mman" } sys_resource = { path = "src/sys_resource" } sys_socket = { path = "src/sys_socket" } diff --git a/include/sys/param.h b/include/sys/param.h new file mode 100644 index 000000000..68ff8c172 --- /dev/null +++ b/include/sys/param.h @@ -0,0 +1,34 @@ +#ifndef _SYS_PARAM_H +#define _SYS_PARAM_H + +#define MIN(a,b) (((a) < (b)) ? (a) : (b)) +#define MAX(a,b) (((a) > (b)) ? (a) : (b)) + +#define __bitop(array, index, op) ((array)[(index) / 8] op (1 << (index) % 8)) +#define setbit(array, index) __bitop(array, index, |=) +#define clrbit(array, index) __bitop(array, index, &= ~) +#define isset(array, index) __bitop(array, index, &) +#define isclr(array, index) !isset(array, index) + +#define howmany(bits, size) (((bits) + (size) - 1) / (size)) +#define roundup(bits, size) (howmany(bits, size) * (size)) +#define powerof2(n) !(((n) - 1) & (n)) + +// Shamelessly copied from musl. +// Tweak as needed. +#define MAXSYMLINKS 20 +#define MAXHOSTNAMELEN 64 +#define MAXNAMLEN 255 +#define MAXPATHLEN 4096 +#define NBBY 8 +#define NGROUPS 32 +#define CANBSIZ 255 +#define NOFILE 256 +#define NCARGS 131072 +#define DEV_BSIZE 512 +#define NOGROUP (-1) + +#include <sys/resource.h> +#include <limits.h> + +#endif diff --git a/src/ctype/src/lib.rs b/src/ctype/src/lib.rs index bcbc443f5..a9f602043 100644 --- a/src/ctype/src/lib.rs +++ b/src/ctype/src/lib.rs @@ -18,7 +18,7 @@ pub extern "C" fn isalpha(c: c_int) -> c_int { #[no_mangle] pub extern "C" fn isascii(c: c_int) -> c_int { - (!(c & !0x7f)) as c_int + ((c & !0x7f) == 0) as c_int } #[no_mangle] diff --git a/src/lib.rs b/src/lib.rs index a03b7a456..e1da46a8d 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -20,6 +20,7 @@ pub extern crate signal; pub extern crate stdio; pub extern crate stdlib; pub extern crate string; +pub extern crate strings; pub extern crate sys_mman; pub extern crate sys_resource; pub extern crate sys_socket; diff --git a/src/platform/src/lib.rs b/src/platform/src/lib.rs index 5d95b235f..e9b3bb8b8 100644 --- a/src/platform/src/lib.rs +++ b/src/platform/src/lib.rs @@ -4,6 +4,7 @@ #![feature(allocator_api)] //TODO #![feature(thread_local)] +#[cfg_attr(target_os = "redox", macro_use)] extern crate alloc; #[cfg(all(not(feature = "no_std"), target_os = "linux"))] @@ -43,6 +44,22 @@ use types::*; #[global_allocator] static ALLOCATOR: Allocator = Allocator; +pub const AF_INET: c_int = 2; +pub const SOCK_STREAM: c_int = 1; +pub const SOCK_DGRAM: c_int = 2; +pub const SOCK_NONBLOCK: c_int = 0o4000; +pub const SOCK_CLOEXEC: c_int = 0o2000000; + +pub type in_addr_t = [u8; 4]; +pub type in_port_t = u16; +pub type sa_family_t = u16; +pub type socklen_t = u32; + +pub struct sockaddr { + pub sa_family: sa_family_t, + pub data: [c_char; 14] +} + //TODO #[thread_local] #[allow(non_upper_case_globals)] #[no_mangle] diff --git a/src/platform/src/linux/mod.rs b/src/platform/src/linux/mod.rs index 9a8e783e6..89bbb2150 100644 --- a/src/platform/src/linux/mod.rs +++ b/src/platform/src/linux/mod.rs @@ -1,6 +1,6 @@ use core::ptr; -use errno; +use ::*; use types::*; const AT_FDCWD: c_int = -100; @@ -19,6 +19,10 @@ pub fn e(sys: usize) -> usize { } } +pub unsafe fn bind(socket: c_int, address: *const sockaddr, address_len: socklen_t) -> c_int { + e(syscall!(BIND, socket, address, address_len)) as c_int +} + pub fn brk(addr: *mut c_void) -> *mut c_void { unsafe { syscall!(BRK, addr) as *mut c_void } } @@ -39,6 +43,10 @@ pub fn close(fildes: c_int) -> c_int { e(unsafe { syscall!(CLOSE, fildes) }) as c_int } +pub unsafe fn connect(socket: c_int, address: *const sockaddr, address_len: socklen_t) -> c_int { + e(syscall!(CONNECT, socket, address, address_len)) as c_int +} + pub fn dup(fildes: c_int) -> c_int { e(unsafe { syscall!(DUP, fildes) }) as c_int } @@ -171,6 +179,11 @@ pub fn read(fildes: c_int, buf: &mut [u8]) -> ssize_t { e(unsafe { syscall!(READ, fildes, buf.as_mut_ptr(), buf.len()) }) as ssize_t } +pub unsafe fn recvfrom(socket: c_int, buf: *mut c_void, len: size_t, flags: c_int, + address: *mut sockaddr, address_len: *mut socklen_t) -> ssize_t { + e(syscall!(RECVFROM, socket, buf, len, flags, address, address_len)) as ssize_t +} + pub fn rename(old: *const c_char, new: *const c_char) -> c_int { e(unsafe { syscall!(RENAMEAT, AT_FDCWD, old, AT_FDCWD, new) }) as c_int } @@ -179,6 +192,11 @@ pub fn rmdir(path: *const c_char) -> c_int { e(unsafe { syscall!(UNLINKAT, AT_FDCWD, path, AT_REMOVEDIR) }) as c_int } +pub unsafe fn sendto(socket: c_int, buf: *const c_void, len: size_t, flags: c_int, + dest_addr: *const sockaddr, dest_len: socklen_t) -> ssize_t { + e(syscall!(SENDTO, socket, buf, len, flags, dest_addr, dest_len)) as ssize_t +} + pub fn setpgid(pid: pid_t, pgid: pid_t) -> c_int { e(unsafe { syscall!(SETPGID, pid, pgid) }) as c_int } @@ -195,6 +213,10 @@ pub fn stat(file: *const c_char, buf: *mut stat) -> c_int { e(unsafe { syscall!(NEWFSTATAT, AT_FDCWD, file, buf, 0) }) as c_int } +pub fn socket(domain: c_int, kind: c_int, protocol: c_int) -> c_int { + e(unsafe { syscall!(SOCKET, domain, kind, protocol) }) as c_int +} + pub fn uname(utsname: usize) -> c_int { e(unsafe { syscall!(UNAME, utsname, 0) }) as c_int } diff --git a/src/platform/src/redox/mod.rs b/src/platform/src/redox/mod.rs index a6db080e0..517105db8 100644 --- a/src/platform/src/redox/mod.rs +++ b/src/platform/src/redox/mod.rs @@ -1,3 +1,4 @@ +use alloc; use core::mem; use core::ptr; use core::slice; @@ -6,10 +7,16 @@ use syscall::data::Stat as redox_stat; use syscall::data::TimeSpec as redox_timespec; use syscall::flag::*; -use c_str; -use errno; +use ::*; use types::*; +#[repr(C)] +struct SockData { + port: in_port_t, + addr: in_addr_t, + _pad: [c_char; 8] +} + pub fn e(sys: Result<usize, syscall::Error>) -> usize { match sys { Ok(ok) => ok, @@ -22,6 +29,45 @@ pub fn e(sys: Result<usize, syscall::Error>) -> usize { } } +macro_rules! bind_or_connect { + (bind $path:expr) => { + concat!("/", $path) + }; + (connect $path:expr) => { + $path + }; + ($mode:ident $socket:expr, $address:expr, $address_len:expr) => {{ + if (*$address).sa_family as c_int != AF_INET { + errno = syscall::EAFNOSUPPORT; + return -1; + } + if ($address_len as usize) < mem::size_of::<sockaddr>() { + errno = syscall::EINVAL; + return -1; + } + let data: &SockData = mem::transmute(&(*$address).data); + let addr = &data.addr; + let port = in_port_t::from_be(data.port); // This is transmuted from bytes in BigEndian order + let path = format!(bind_or_connect!($mode "{}.{}.{}.{}:{}"), addr[0], addr[1], addr[2], addr[3], port); + + // Duplicate the socket, and then duplicate the copy back to the original fd + let fd = e(syscall::dup($socket as usize, path.as_bytes())); + if (fd as c_int) < 0 { + return -1; + } + let result = syscall::dup2(fd, $socket as usize, &[]); + let _ = syscall::close(fd); + if (e(result) as c_int) < 0 { + return -1; + } + 0 + }} +} + +pub unsafe fn bind(socket: c_int, address: *const sockaddr, address_len: socklen_t) -> c_int { + bind_or_connect!(bind socket, address, address_len) +} + pub fn brk(addr: *mut c_void) -> *mut c_void { unsafe { syscall::brk(addr as usize).unwrap_or(0) as *mut c_void } } @@ -59,6 +105,10 @@ pub fn close(fd: c_int) -> c_int { e(syscall::close(fd as usize)) as c_int } +pub unsafe fn connect(socket: c_int, address: *const sockaddr, address_len: socklen_t) -> c_int { + bind_or_connect!(connect socket, address, address_len) +} + pub fn dup(fd: c_int) -> c_int { e(syscall::dup(fd as usize, &[])) as c_int } @@ -331,6 +381,24 @@ pub fn read(fd: c_int, buf: &mut [u8]) -> ssize_t { e(syscall::read(fd as usize, buf)) as ssize_t } +pub unsafe fn recvfrom(socket: c_int, buf: *mut c_void, len: size_t, flags: c_int, + address: *mut sockaddr, address_len: *mut socklen_t) -> ssize_t { + if flags != 0 { + errno = syscall::EOPNOTSUPP; + return -1; + } + let data = slice::from_raw_parts_mut( + &mut (*address).data as *mut _ as *mut u8, + (*address).data.len() + ); + let pathlen = e(syscall::fpath(socket as usize, data)); + if pathlen < 0 { + return -1; + } + *address_len = pathlen as socklen_t; + read(socket, slice::from_raw_parts_mut(buf as *mut u8, len)) +} + pub fn rename(oldpath: *const c_char, newpath: *const c_char) -> c_int { let (oldpath, newpath) = unsafe { (c_str(oldpath), c_str(newpath)) }; match syscall::open(oldpath, O_WRONLY) { @@ -348,6 +416,16 @@ pub fn rmdir(path: *const c_char) -> c_int { e(syscall::rmdir(path)) as c_int } +pub unsafe fn sendto(socket: c_int, buf: *const c_void, len: size_t, flags: c_int, + _dest_addr: *const sockaddr, _dest_len: socklen_t) -> ssize_t { + // TODO: Use dest_addr and dest_len + if flags != 0 { + errno = syscall::EOPNOTSUPP; + return -1; + } + write(socket, slice::from_raw_parts(buf as *const u8, len)) +} + pub fn setpgid(pid: pid_t, pgid: pid_t) -> c_int { e(syscall::setpgid(pid as usize, pgid as usize)) as c_int } @@ -372,6 +450,38 @@ pub fn stat(path: *const c_char, buf: *mut stat) -> c_int { } } +pub unsafe fn socket(domain: c_int, mut kind: c_int, protocol: c_int) -> c_int { + if domain != AF_INET { + errno = syscall::EAFNOSUPPORT; + return -1; + } + if protocol != 0 { + errno = syscall::EPROTONOSUPPORT; + return -1; + } + + let mut flags = O_RDWR; + if kind & SOCK_NONBLOCK == SOCK_NONBLOCK { + kind &= !SOCK_NONBLOCK; + flags |= O_NONBLOCK; + } + if kind & SOCK_CLOEXEC == SOCK_CLOEXEC { + kind &= !SOCK_CLOEXEC; + flags |= O_CLOEXEC; + } + + // The tcp: and udp: schemes allow using no path, + // and later specifying one using `dup`. + match kind { + SOCK_STREAM => e(syscall::open("tcp:", flags)) as c_int, + SOCK_DGRAM => e(syscall::open("udp:", flags)) as c_int, + _ => { + errno = syscall::EPROTOTYPE; + -1 + } + } +} + pub fn unlink(path: *const c_char) -> c_int { let path = unsafe { c_str(path) }; e(syscall::unlink(path)) as c_int diff --git a/src/stdio/src/printf.rs b/src/stdio/src/printf.rs index b7a7bbfc4..174c603f8 100644 --- a/src/stdio/src/printf.rs +++ b/src/stdio/src/printf.rs @@ -55,13 +55,11 @@ pub unsafe fn printf<W: Write>(mut w: W, format: *const c_char, mut ap: VaList) w.write_fmt(format_args!("0x{:x}", a)) } 's' => { - let a = ap.get::<usize>(); + let a = ap.get::<*const c_char>(); found_percent = false; - w.write_str(str::from_utf8_unchecked(platform::c_str( - a as *const c_char, - ))) + w.write_str(str::from_utf8_unchecked(platform::c_str(a))) } 'u' => { let a = ap.get::<c_uint>(); diff --git a/src/stdlib/src/lib.rs b/src/stdlib/src/lib.rs index 0bc5eb012..1afe2638e 100644 --- a/src/stdlib/src/lib.rs +++ b/src/stdlib/src/lib.rs @@ -229,7 +229,7 @@ pub extern "C" fn erand(xsubi: [c_ushort; 3]) -> c_double { } #[no_mangle] -pub unsafe extern "C" fn exit(status: c_int) -> ! { +pub unsafe extern "C" fn exit(status: c_int) { for i in (0..ATEXIT_FUNCS.len()).rev() { if let Some(func) = ATEXIT_FUNCS[i] { (func)(); diff --git a/src/string/src/lib.rs b/src/string/src/lib.rs index 0caeb823a..0c627a861 100644 --- a/src/string/src/lib.rs +++ b/src/string/src/lib.rs @@ -126,14 +126,13 @@ pub unsafe extern "C" fn strcat(s1: *mut c_char, s2: *const c_char) -> *mut c_ch } #[no_mangle] -pub unsafe extern "C" fn strchr(s: *const c_char, c: c_int) -> *mut c_char { - let c = c as i8; - let mut i = 0; - while *s.offset(i) != 0 { - if *s.offset(i) == c { - return s.offset(i) as *mut c_char; +pub unsafe extern "C" fn strchr(mut s: *const c_char, c: c_int) -> *mut c_char { + let c = c as c_char; + while *s != 0 { + if *s == c { + return s as *mut c_char; } - i += 1; + s = s.offset(1); } ptr::null_mut() } @@ -276,7 +275,7 @@ pub unsafe extern "C" fn strncpy(s1: *mut c_char, s2: *const c_char, n: usize) - } // if length of s2 < n, pad s1 with zeroes - for _ in cmp::min(n, s2_len)..n { + while idx < s2_len { *s1.offset(idx as isize) = 0; idx += 1; } diff --git a/src/strings/Cargo.toml b/src/strings/Cargo.toml new file mode 100644 index 000000000..24e75867a --- /dev/null +++ b/src/strings/Cargo.toml @@ -0,0 +1,11 @@ +[package] +name = "strings" +version = "0.1.0" +authors = ["jD91mZM2 <me@krake.one>"] +build = "build.rs" + +[build-dependencies] +cbindgen = { path = "../../cbindgen" } + +[dependencies] +platform = { path = "../platform" } diff --git a/src/strings/build.rs b/src/strings/build.rs new file mode 100644 index 000000000..574237034 --- /dev/null +++ b/src/strings/build.rs @@ -0,0 +1,11 @@ +extern crate cbindgen; + +use std::{env, fs}; + +fn main() { + let crate_dir = env::var("CARGO_MANIFEST_DIR").expect("CARGO_MANIFEST_DIR not set"); + fs::create_dir_all("../../target/include").expect("failed to create include directory"); + cbindgen::generate(crate_dir) + .expect("failed to generate bindings") + .write_to_file("../../target/include/strings.h"); +} diff --git a/src/strings/cbindgen.toml b/src/strings/cbindgen.toml new file mode 100644 index 000000000..d1c7851d7 --- /dev/null +++ b/src/strings/cbindgen.toml @@ -0,0 +1,7 @@ +sys_includes = ["stddef.h", "stdint.h"] +include_guard = "_STRINGS_H" +language = "C" +style = "Tag" + +[enum] +prefix_with_name = true diff --git a/src/strings/src/lib.rs b/src/strings/src/lib.rs new file mode 100644 index 000000000..98ac565c0 --- /dev/null +++ b/src/strings/src/lib.rs @@ -0,0 +1,136 @@ +//! strings implementation for Redox, following http://pubs.opengroup.org/onlinepubs/7908799/xsh/strings.h.html +#![no_std] +#![feature(alloc)] + +extern crate alloc; +extern crate platform; + +use alloc::Vec; +use core::ptr; +use platform::types::*; + +#[no_mangle] +pub unsafe extern "C" fn bcmp(mut first: *const c_void, mut second: *const c_void, n: size_t) -> c_int { + let first = first as *const c_char; + let second = second as *const c_char; + + for i in 0..n as isize { + if *first.offset(i) != *second.offset(i) { + return -1; + } + } + 0 +} + +#[no_mangle] +pub unsafe extern "C" fn bcopy(src: *const c_void, dst: *mut c_void, n: size_t) { + let src = src as *mut c_char; + let dst = dst as *mut c_char; + + let mut tmp = Vec::with_capacity(n); + for i in 0..n as isize { + tmp.push(*src.offset(i)); + } + for (i, val) in tmp.into_iter().enumerate() { + *dst.offset(i as isize) = val; + } +} + +#[no_mangle] +pub unsafe extern "C" fn bzero(src: *mut c_void, n: size_t) { + let src = src as *mut c_char; + + for i in 0..n as isize { + *src.offset(i) = 0; + } +} + +#[no_mangle] +pub extern "C" fn ffs(mut i: c_int) -> c_int { + if i == 0 { + return 0; + } + let mut n = 1; + while i & 1 == 0 { + i >>= 1; + n += 1; + } + n +} + +#[no_mangle] +pub unsafe extern "C" fn index(mut s: *const c_char, c: c_int) -> *mut c_char { + while *s != 0 { + if *s == c as c_char { + // Input is const but output is mutable. WHY C, WHY DO THIS? + return s as *mut c_char; + } + s = s.offset(1); + } + ptr::null_mut() +} + +#[no_mangle] +pub unsafe extern "C" fn rindex(mut s: *const c_char, c: c_int) -> *mut c_char { + let original = s; + while *s != 0 { + s = s.offset(1); + } + + while s != original { + s = s.offset(-1); + if *s == c as c_char { + // Input is const but output is mutable. WHY C, WHY DO THIS? + return s as *mut c_char; + } + } + ptr::null_mut() +} + +#[no_mangle] +pub unsafe extern "C" fn strcasecmp(mut first: *const c_char, mut second: *const c_char) -> c_int { + while *first != 0 && *second != 0 { + let mut i = *first; + let mut j = *second; + + if i >= b'A' as c_char && i <= b'Z' as c_char { + i += (b'a' - b'A') as c_char; + } + if j >= b'A' as c_char && j <= b'Z' as c_char { + j += (b'a' - b'A') as c_char; + } + + if i != j { + return -1; + } + + first = first.offset(1); + second = first.offset(1); + } + // Both strings ended at the same time too + (*first == *second) as c_int +} +#[no_mangle] +pub unsafe extern "C" fn strncasecmp(mut first: *const c_char, mut second: *const c_char, mut n: size_t) -> c_int { + while *first != 0 && *second != 0 && n > 0 { + let mut i = *first; + let mut j = *second; + + if i >= b'A' as c_char && i <= b'Z' as c_char { + i += (b'a' - b'A') as c_char; + } + if j >= b'A' as c_char && j <= b'Z' as c_char { + j += (b'a' - b'A') as c_char; + } + + if i != j { + return -1; + } + + first = first.offset(1); + second = first.offset(1); + n -= 1; + } + // Both strings ended at the same time too + (n == 0 || *first == *second) as c_int +} diff --git a/src/sys_socket/src/lib.rs b/src/sys_socket/src/lib.rs index c7c0b5f2b..20ff30f16 100644 --- a/src/sys_socket/src/lib.rs +++ b/src/sys_socket/src/lib.rs @@ -5,17 +5,26 @@ extern crate platform; +use core::ptr; use platform::types::*; +pub type in_addr_t = [u8; 4]; +pub type in_port_t = u16; pub type sa_family_t = u16; pub type socklen_t = u32; #[repr(C)] pub struct sockaddr { pub sa_family: sa_family_t, - pub sa_data: [c_char; 14], + data: [c_char; 14] } +pub const AF_INET: c_int = 2; +pub const SOCK_STREAM: c_int = 1; +pub const SOCK_DGRAM: c_int = 2; +pub const SOCK_NONBLOCK: c_int = 0o4000; +pub const SOCK_CLOEXEC: c_int = 0o2000000; + #[no_mangle] pub unsafe extern "C" fn accept( socket: c_int, @@ -31,7 +40,7 @@ pub unsafe extern "C" fn bind( address: *const sockaddr, address_len: socklen_t, ) -> c_int { - unimplemented!(); + platform::bind(socket, address as *const platform::sockaddr, address_len) } #[no_mangle] @@ -40,7 +49,7 @@ pub unsafe extern "C" fn connect( address: *const sockaddr, address_len: socklen_t, ) -> c_int { - unimplemented!(); + platform::connect(socket, address as *const platform::sockaddr, address_len) } #[no_mangle] @@ -84,7 +93,7 @@ pub unsafe extern "C" fn recv( length: size_t, flags: c_int, ) -> ssize_t { - unimplemented!(); + recvfrom(socket, buffer, length, flags, ptr::null_mut(), ptr::null_mut()) } #[no_mangle] @@ -96,7 +105,7 @@ pub unsafe extern "C" fn recvfrom( address: *mut sockaddr, address_len: *mut socklen_t, ) -> ssize_t { - unimplemented!(); + platform::recvfrom(socket, buffer, length, flags, address as *mut platform::sockaddr, address_len) } #[no_mangle] @@ -106,11 +115,11 @@ pub unsafe extern "C" fn send( length: size_t, flags: c_int, ) -> ssize_t { - unimplemented!(); + sendto(socket, message, length, flags, ptr::null(), 0) } #[no_mangle] -pub unsafe extern "C" fn sento( +pub unsafe extern "C" fn sendto( socket: c_int, message: *const c_void, length: size_t, @@ -118,7 +127,7 @@ pub unsafe extern "C" fn sento( dest_addr: *const sockaddr, dest_len: socklen_t, ) -> ssize_t { - unimplemented!(); + platform::sendto(socket, message, length, flags, dest_addr as *const platform::sockaddr, dest_len) } #[no_mangle] @@ -138,8 +147,8 @@ pub unsafe extern "C" fn shutdown(socket: c_int, how: c_int) -> c_int { } #[no_mangle] -pub unsafe extern "C" fn socket(domain: c_int, _type: c_int, protocol: c_int) -> c_int { - unimplemented!(); +pub unsafe extern "C" fn socket(domain: c_int, kind: c_int, protocol: c_int) -> c_int { + platform::socket(domain, kind, protocol) } #[no_mangle] diff --git a/tests/.gitignore b/tests/.gitignore index e2242ca6a..afb0d2f8c 100644 --- a/tests/.gitignore +++ b/tests/.gitignore @@ -32,6 +32,7 @@ /setjmp /sleep /sprintf +/strings /stdlib/a64l /stdlib/bsearch /stdlib/mktemp diff --git a/tests/Makefile b/tests/Makefile index 3588a66a5..0957b27f2 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -26,6 +26,7 @@ EXPECT_BINS=\ setjmp \ sleep \ sprintf \ + strings \ stdio/fwrite \ stdio/all \ stdio/freopen \ @@ -116,4 +117,4 @@ TAILLIBS=\ ../target/openlibm/libopenlibm.a %: %.c $(HEADLIBS) $(TAILLIBS) - gcc -fno-stack-protector -Wall -g $(CFLAGS) $(HEADLIBS) "$<" $(TAILLIBS) -o "$@" + gcc -fno-builtin -fno-stack-protector -Wall -g $(CFLAGS) $(HEADLIBS) "$<" $(TAILLIBS) -o "$@" diff --git a/tests/expected/string/strchr.stdout b/tests/expected/string/strchr.stdout index 5a72f0ea5..02c9cb955 100644 --- a/tests/expected/string/strchr.stdout +++ b/tests/expected/string/strchr.stdout @@ -1,3 +1,3 @@ ello ld - +1 diff --git a/tests/expected/strings.stderr b/tests/expected/strings.stderr new file mode 100644 index 000000000..e69de29bb diff --git a/tests/expected/strings.stdout b/tests/expected/strings.stdout new file mode 100644 index 000000000..e69de29bb diff --git a/tests/string/strchr.c b/tests/string/strchr.c index 929c22bed..202737faa 100644 --- a/tests/string/strchr.c +++ b/tests/string/strchr.c @@ -4,7 +4,7 @@ int main(int argc, char* argv[]) { printf("%s\n", strchr("hello", 'e')); // should be ello printf("%s\n", strchr("world", 'l')); // should be ld - printf("%s\n", strchr("world", 0)); // should be '' + printf("%i\n", strchr("world", 0) == NULL); // should be 1 return 0; } diff --git a/tests/strings.c b/tests/strings.c new file mode 100644 index 000000000..2423b9003 --- /dev/null +++ b/tests/strings.c @@ -0,0 +1,30 @@ +#include <assert.h> +#include <stdlib.h> +#include <stdio.h> +#include <strings.h> + +int main() { + assert(!bcmp("hello", "hehe", 2)); + assert(bcmp("hello", "haha", 2)); + + char* new = malloc(3); + bcopy("hi", new, 3); // include nul byte + + assert(!strcasecmp("hi", new)); + assert(!strcasecmp("hi", "HI")); + assert(!strncasecmp("hi", "HIHI", 2)); + + bzero(new, 1); + assert(*new == 0); + assert(*(new+1) == 'i'); + assert(*(new+2) == 0); + + assert(ffs(1) == 1); + assert(ffs(2) == 2); + assert(ffs(3) == 1); + assert(ffs(10) == 2); + + char* str = "hihih"; + assert(index(str, 'i') == str + 1); + assert(rindex(str, 'i') == str + 3); +} -- GitLab