diff options
Diffstat (limited to 'libc/libc-test')
| -rw-r--r-- | libc/libc-test/Cargo.toml | 11 | ||||
| -rw-r--r-- | libc/libc-test/build-generated.rs | 16 | ||||
| -rw-r--r-- | libc/libc-test/build.rs | 521 | ||||
| -rw-r--r-- | libc/libc-test/generate-files/Cargo.toml | 16 | ||||
| -rw-r--r-- | libc/libc-test/run-generated-Cargo.toml | 19 | ||||
| -rw-r--r-- | libc/libc-test/src/main-generated.rs | 9 | ||||
| -rw-r--r-- | libc/libc-test/src/main.rs | 6 | 
7 files changed, 598 insertions, 0 deletions
| diff --git a/libc/libc-test/Cargo.toml b/libc/libc-test/Cargo.toml new file mode 100644 index 0000000..4e7c884 --- /dev/null +++ b/libc/libc-test/Cargo.toml @@ -0,0 +1,11 @@ +[package] +name = "libc-test" +version = "0.1.0" +authors = ["Alex Crichton <alex@alexcrichton.com>"] +build = "build.rs" + +[dependencies] +libc = { path = ".." } + +[build-dependencies] +ctest = "0.1" diff --git a/libc/libc-test/build-generated.rs b/libc/libc-test/build-generated.rs new file mode 100644 index 0000000..a51c0e3 --- /dev/null +++ b/libc/libc-test/build-generated.rs @@ -0,0 +1,16 @@ +// This build script is distinct from the standard build.rs as it is only used +// for the BSDs which run a stripped down version. The `all.c` file is assumed +// to have been already generated for this build script. + +extern crate gcc; + +fn main() { +    gcc::Config::new() +        .file("all.c") +        .flag("-Wall") +        .flag("-Wextra") +        .flag("-Werror") +        .flag("-Wno-deprecated-declarations") +        .flag("-Wno-type-limits") +        .compile("liball.a"); +} diff --git a/libc/libc-test/build.rs b/libc/libc-test/build.rs new file mode 100644 index 0000000..4ebba50 --- /dev/null +++ b/libc/libc-test/build.rs @@ -0,0 +1,521 @@ +#![deny(warnings)] + +extern crate ctest; + +use std::env; + +fn main() { +    let target = env::var("TARGET").unwrap(); +    let aarch64 = target.contains("aarch64"); +    let x86_64 = target.contains("x86_64"); +    let windows = target.contains("windows"); +    let mingw = target.contains("windows-gnu"); +    let linux = target.contains("unknown-linux"); +    let android = target.contains("android"); +    let apple = target.contains("apple"); +    let musl = target.contains("musl"); +    let freebsd = target.contains("freebsd"); +    let dragonfly = target.contains("dragonfly"); +    let mips = target.contains("mips"); +    let netbsd = target.contains("netbsd"); +    let openbsd = target.contains("openbsd"); +    let rumprun = target.contains("rumprun"); +    let bsdlike = freebsd || apple || netbsd || openbsd || dragonfly; +    let mut cfg = ctest::TestGenerator::new(); + +    // Pull in extra goodies +    if linux || android { +        cfg.define("_GNU_SOURCE", None); +    } else if netbsd { +        cfg.define("_NETBSD_SOURCE", Some("1")); +    } else if windows { +        cfg.define("_WIN32_WINNT", Some("0x8000")); +    } + +    // Android doesn't actually have in_port_t but it's much easier if we +    // provide one for us to test against +    if android { +        cfg.define("in_port_t", Some("uint16_t")); +    } + +    cfg.header("errno.h") +       .header("fcntl.h") +       .header("limits.h") +       .header("locale.h") +       .header("stddef.h") +       .header("stdint.h") +       .header("stdio.h") +       .header("stdlib.h") +       .header("sys/stat.h") +       .header("sys/types.h") +       .header("time.h") +       .header("wchar.h"); + +    if windows { +        cfg.header("winsock2.h"); // must be before windows.h + +        cfg.header("direct.h"); +        cfg.header("io.h"); +        cfg.header("sys/utime.h"); +        cfg.header("windows.h"); +        cfg.header("process.h"); +        cfg.header("ws2ipdef.h"); + +        if target.contains("gnu") { +            cfg.header("ws2tcpip.h"); +        } +    } else { +        cfg.flag("-Wno-deprecated-declarations"); + +        cfg.header("ctype.h"); +        cfg.header("dirent.h"); +        if openbsd { +            cfg.header("sys/socket.h"); +        } +        cfg.header("net/if.h"); +        cfg.header("netdb.h"); +        cfg.header("netinet/in.h"); +        cfg.header("netinet/ip.h"); +        cfg.header("netinet/tcp.h"); +        cfg.header("pthread.h"); +        cfg.header("dlfcn.h"); +        cfg.header("signal.h"); +        cfg.header("string.h"); +        cfg.header("sys/file.h"); +        cfg.header("sys/ioctl.h"); +        cfg.header("sys/mman.h"); +        cfg.header("sys/resource.h"); +        cfg.header("sys/socket.h"); +        cfg.header("sys/time.h"); +        cfg.header("sys/un.h"); +        cfg.header("sys/wait.h"); +        cfg.header("unistd.h"); +        cfg.header("utime.h"); +        cfg.header("pwd.h"); +        cfg.header("grp.h"); +        cfg.header("sys/utsname.h"); +        cfg.header("sys/ptrace.h"); +        cfg.header("sys/mount.h"); +        cfg.header("sys/uio.h"); +        cfg.header("sched.h"); +        cfg.header("termios.h"); +        cfg.header("poll.h"); +        cfg.header("syslog.h"); +        cfg.header("semaphore.h"); +        cfg.header("sys/statvfs.h"); +    } + +    if android { +        if !aarch64 { +            // time64_t is not define for aarch64 +            // If included it will generate the error 'Your time_t is already 64-bit' +            cfg.header("time64.h"); +        } +        cfg.header("arpa/inet.h"); +        cfg.header("xlocale.h"); +        cfg.header("utmp.h"); +    } else if !windows { +        cfg.header("glob.h"); +        cfg.header("ifaddrs.h"); +        cfg.header("langinfo.h"); + +        if !openbsd && !freebsd && !dragonfly { +            cfg.header("sys/quota.h"); +        } + +        if !musl { +            cfg.header("sys/sysctl.h"); + +            if !netbsd && !openbsd { +                cfg.header("execinfo.h"); +                cfg.header("xlocale.h"); +            } + +            if openbsd { +                cfg.header("utmp.h"); +            } else { +                cfg.header("utmpx.h"); +            } +        } +    } + +    if apple { +        cfg.header("mach-o/dyld.h"); +        cfg.header("mach/mach_time.h"); +        cfg.header("malloc/malloc.h"); +        cfg.header("util.h"); +        if target.starts_with("x86") { +            cfg.header("crt_externs.h"); +        } +    } + +    if bsdlike { +        cfg.header("sys/event.h"); + +        if freebsd { +            cfg.header("libutil.h"); +        } else { +            cfg.header("util.h"); +        } +    } + +    if linux { +        cfg.header("mqueue.h"); +        cfg.header("ucontext.h"); +        cfg.header("sys/signalfd.h"); +        cfg.header("sys/xattr.h"); +        cfg.header("sys/ipc.h"); +        cfg.header("sys/msg.h"); +        cfg.header("sys/shm.h"); +        cfg.header("sys/fsuid.h"); +        cfg.header("pty.h"); +        cfg.header("shadow.h"); +    } + +    if linux || android { +        cfg.header("malloc.h"); +        cfg.header("net/ethernet.h"); +        cfg.header("netpacket/packet.h"); +        cfg.header("sched.h"); +        cfg.header("sys/epoll.h"); +        cfg.header("sys/eventfd.h"); +        cfg.header("sys/prctl.h"); +        cfg.header("sys/sendfile.h"); +        cfg.header("sys/vfs.h"); +        cfg.header("sys/syscall.h"); +        cfg.header("sys/sysinfo.h"); +        cfg.header("sys/reboot.h"); +        if !musl { +            cfg.header("linux/netlink.h"); +            cfg.header("linux/magic.h"); +            cfg.header("linux/reboot.h"); + +            if !mips { +                cfg.header("linux/quota.h"); +            } +        } +    } + +    if freebsd { +        cfg.header("pthread_np.h"); +        cfg.header("sched.h"); +        cfg.header("ufs/ufs/quota.h"); +        cfg.header("sys/jail.h"); +    } + +    if netbsd { +        cfg.header("ufs/ufs/quota.h"); +        cfg.header("ufs/ufs/quota1.h"); +        cfg.header("sys/ioctl_compat.h"); +    } + +    if openbsd { +        cfg.header("ufs/ufs/quota.h"); +        cfg.header("pthread_np.h"); +        cfg.header("sys/syscall.h"); +    } + +    if dragonfly { +        cfg.header("ufs/ufs/quota.h"); +        cfg.header("pthread_np.h"); +        cfg.header("sys/ioctl_compat.h"); +    } + +    if linux || freebsd || dragonfly || netbsd || apple { +        cfg.header("aio.h"); +    } + +    cfg.type_name(move |ty, is_struct| { +        match ty { +            // Just pass all these through, no need for a "struct" prefix +            "FILE" | +            "fd_set" | +            "Dl_info" | +            "DIR" => ty.to_string(), + +            // Fixup a few types on windows that don't actually exist. +            "time64_t" if windows => "__time64_t".to_string(), +            "ssize_t" if windows => "SSIZE_T".to_string(), + +            // OSX calls this something else +            "sighandler_t" if bsdlike => "sig_t".to_string(), + +            t if t.ends_with("_t") => t.to_string(), + +            // Windows uppercase structs don't have `struct` in front, there's a +            // few special cases for windows, and then otherwise put `struct` in +            // front of everything. +            t if is_struct => { +                if windows && ty.chars().next().unwrap().is_uppercase() { +                    t.to_string() +                } else if windows && t == "stat" { +                    "struct __stat64".to_string() +                } else if windows && t == "utimbuf" { +                    "struct __utimbuf64".to_string() +                } else { +                    format!("struct {}", t) +                } +            } + +            t => t.to_string(), +        } +    }); + +    let target2 = target.clone(); +    cfg.field_name(move |struct_, field| { +        match field { +            "st_birthtime"      if openbsd && struct_ == "stat" => "__st_birthtime".to_string(), +            "st_birthtime_nsec" if openbsd && struct_ == "stat" => "__st_birthtimensec".to_string(), +            // Our stat *_nsec fields normally don't actually exist but are part +            // of a timeval struct +            s if s.ends_with("_nsec") && struct_.starts_with("stat") => { +                if target2.contains("apple") { +                    s.replace("_nsec", "spec.tv_nsec") +                } else if target2.contains("android") { +                    s.to_string() +                } else { +                    s.replace("e_nsec", ".tv_nsec") +                } +            } +            "u64" if struct_ == "epoll_event" => "data.u64".to_string(), +            s => s.to_string(), +        } +    }); + +    cfg.skip_type(move |ty| { +        match ty { +            // sighandler_t is crazy across platforms +            "sighandler_t" => true, + +            _ => false +        } +    }); + +    cfg.skip_struct(move |ty| { +        match ty { +            "sockaddr_nl" => musl, + +            // On Linux, the type of `ut_tv` field of `struct utmpx` +            // can be an anonymous struct, so an extra struct, +            // which is absent in glibc, has to be defined. +            "__timeval" if linux => true, + +            // The alignment of this is 4 on 64-bit OSX... +            "kevent" if apple && x86_64 => true, + +            // This is actually a union, not a struct +            "sigval" => true, + +            _ => false +        } +    }); + +    cfg.skip_signededness(move |c| { +        match c { +            "LARGE_INTEGER" | +            "mach_timebase_info_data_t" | +            "float" | +            "double" => true, +            // uuid_t is a struct, not an integer. +            "uuid_t" if dragonfly => true, +            n if n.starts_with("pthread") => true, +            // sem_t is a struct or pointer +            "sem_t" if openbsd || freebsd || dragonfly || rumprun => true, + +            // windows-isms +            n if n.starts_with("P") => true, +            n if n.starts_with("H") => true, +            n if n.starts_with("LP") => true, +            _ => false, +        } +    }); + +    cfg.skip_const(move |name| { +        match name { +            // Apparently these don't exist in mingw headers? +            "MEM_RESET_UNDO" | +            "FILE_ATTRIBUTE_NO_SCRUB_DATA" | +            "FILE_ATTRIBUTE_INTEGRITY_STREAM" | +            "ERROR_NOTHING_TO_TERMINATE" if mingw => true, + +            "SIG_IGN" => true, // sighandler_t weirdness + +            // types on musl are defined a little differently +            n if musl && n.contains("__SIZEOF_PTHREAD") => true, + +            // Skip constants not defined in MUSL but just passed down to the +            // kernel regardless +            "RLIMIT_NLIMITS" | +            "TCP_COOKIE_TRANSACTIONS" | +            "RLIMIT_RTTIME" | +            "MSG_COPY" if musl => true, +            // work around super old mips toolchain +            "SCHED_IDLE" | "SHM_NORESERVE" => mips, + +            // weird signed extension or something like that? +            "MS_NOUSER" => true, +            "MS_RMT_MASK" => true, // updated in glibc 2.22 and musl 1.1.13 + +            // These OSX constants are flagged as deprecated +            "NOTE_EXIT_REPARENTED" | +            "NOTE_REAP" if apple => true, + +            // The linux/quota.h header file which defines these can't be +            // included with sys/quota.h currently on MIPS, so we don't include +            // it and just ignore these constants +            "QFMT_VFS_OLD" | +            "QFMT_VFS_V0" if mips && linux => true, + +            // These constants were removed in FreeBSD 11 (svn r273250) but will +            // still be accepted and ignored at runtime. +            "MAP_RENAME" | +            "MAP_NORESERVE" if freebsd => true, + +            // These constants were removed in FreeBSD 11 (svn r262489), +            // and they've never had any legitimate use outside of the +            // base system anyway. +            "CTL_MAXID" | +            "KERN_MAXID" | +            "HW_MAXID" | +            "USER_MAXID" if freebsd => true, + +            // These OSX constants are removed in Sierra. +            // https://developer.apple.com/library/content/releasenotes/General/APIDiffsMacOS10_12/Swift/Darwin.html +            "KERN_KDENABLE_BG_TRACE" if apple => true, +            "KERN_KDDISABLE_BG_TRACE" if apple => true, + +            _ => false, +        } +    }); + +    cfg.skip_fn(move |name| { +        // skip those that are manually verified +        match name { +            "execv" |       // crazy stuff with const/mut +            "execve" | +            "execvp" | +            "execvpe" => true, + +            "getrlimit" | "getrlimit64" |    // non-int in 1st arg +            "setrlimit" | "setrlimit64" |    // non-int in 1st arg +            "prlimit" | "prlimit64" |        // non-int in 2nd arg +            "strerror_r" if linux => true,   // actually xpg-something-or-other + +            // typed 2nd arg on linux and android +            "gettimeofday" if linux || android || freebsd || openbsd || dragonfly => true, + +            // not declared in newer android toolchains +            "getdtablesize" if android => true, + +            "dlerror" if android => true, // const-ness is added +            "dladdr" if musl => true, // const-ness only added recently + +            // OSX has 'struct tm *const' which we can't actually represent in +            // Rust, but is close enough to *mut +            "timegm" if apple => true, + +            // OSX's daemon is deprecated in 10.5 so we'll get a warning (which +            // we turn into an error) so just ignore it. +            "daemon" if apple => true, + +            // Deprecated on OSX +            "sem_destroy" if apple => true, +            "sem_init" if apple => true, + +            // These functions presumably exist on netbsd but don't look like +            // they're implemented on rumprun yet, just let them slide for now. +            // Some of them look like they have headers but then don't have +            // corresponding actual definitions either... +            "shm_open" | +            "shm_unlink" | +            "syscall" | +            "ptrace" | +            "sigaltstack" if rumprun => true, + +            // There seems to be a small error in EGLIBC's eventfd.h header. The +            // [underlying system call][1] always takes its first `count` +            // argument as an `unsigned int`, but [EGLIBC's <sys/eventfd.h> +            // header][2] declares it to take an `int`. [GLIBC's header][3] +            // matches the kernel. +            // +            // EGLIBC is no longer actively developed, and Debian, the largest +            // distribution that had been using it, switched back to GLIBC in +            // April 2015. So effectively all Linux <sys/eventfd.h> headers will +            // be using `unsigned int` soon. +            // +            // [1]: https://git.kernel.org/cgit/linux/kernel/git/stable/linux-stable.git/tree/fs/eventfd.c?id=refs/tags/v3.12.51#n397 +            // [2]: http://bazaar.launchpad.net/~ubuntu-branches/ubuntu/trusty/eglibc/trusty/view/head:/sysdeps/unix/sysv/linux/sys/eventfd.h +            // [3]: https://sourceware.org/git/?p=glibc.git;a=blob;f=sysdeps/unix/sysv/linux/sys/eventfd.h;h=6295f32e937e779e74318eb9d3bdbe76aef8a8f3;hb=4e42b5b8f89f0e288e68be7ad70f9525aebc2cff#l34 +            "eventfd" if linux => true, + +            // The `uname` function in freebsd is now an inline wrapper that +            // delegates to another, but the symbol still exists, so don't check +            // the symbol. +            "uname" if freebsd => true, + +            // aio_waitcomplete's return type changed between FreeBSD 10 and 11. +            "aio_waitcomplete" if freebsd => true, + +            // lio_listio confuses the checker, probably because one of its +            // arguments is an array +            "lio_listio" if freebsd => true, +            "lio_listio" if musl => true, + +            // Apparently the NDK doesn't have this defined on android, but +            // it's in a header file? +            "endpwent" if android => true, + +            _ => false, +        } +    }); + +    cfg.skip_fn_ptrcheck(move |name| { +        match name { +            // dllimport weirdness? +            _ if windows => true, + +            _ => false, +        } +    }); + +    cfg.skip_field_type(move |struct_, field| { +        // This is a weird union, don't check the type. +        (struct_ == "ifaddrs" && field == "ifa_ifu") || +        // sighandler_t type is super weird +        (struct_ == "sigaction" && field == "sa_sigaction") || +        // __timeval type is a patch which doesn't exist in glibc +        (linux && struct_ == "utmpx" && field == "ut_tv") || +        // sigval is actually a union, but we pretend it's a struct +        (struct_ == "sigevent" && field == "sigev_value") || +        // aio_buf is "volatile void*" and Rust doesn't understand volatile +        (struct_ == "aiocb" && field == "aio_buf") || +        // stack_t.ss_sp's type changed from FreeBSD 10 to 11 in svn r294930 +        (freebsd && struct_ == "stack_t" && field == "ss_sp") +    }); + +    cfg.skip_field(move |struct_, field| { +        // this is actually a union on linux, so we can't represent it well and +        // just insert some padding. +        (struct_ == "siginfo_t" && field == "_pad") || +        // musl names this __dummy1 but it's still there +        (musl && struct_ == "glob_t" && field == "gl_flags") || +        // musl seems to define this as an *anonymous* bitfield +        (musl && struct_ == "statvfs" && field == "__f_unused") || +        // sigev_notify_thread_id is actually part of a sigev_un union +        (struct_ == "sigevent" && field == "sigev_notify_thread_id") +    }); + +    cfg.fn_cname(move |name, cname| { +        if windows { +            cname.unwrap_or(name).to_string() +        } else { +            name.to_string() +        } +    }); + +    if env::var("SKIP_COMPILE").is_ok() { +        cfg.generate_files("../src/lib.rs", "all.rs"); +    } else { +        cfg.generate("../src/lib.rs", "all.rs"); +    } +} diff --git a/libc/libc-test/generate-files/Cargo.toml b/libc/libc-test/generate-files/Cargo.toml new file mode 100644 index 0000000..8c19856 --- /dev/null +++ b/libc/libc-test/generate-files/Cargo.toml @@ -0,0 +1,16 @@ +# Cargo.toml which is used to just generate the all.{c,rs} files used to test +# some platforms. These files are then combined with the overlay files commented +# in the above directory as well to assemble a libc-test project which will +# compile/run all tests. + +[package] +name = "generate-files" +version = "0.1.0" +authors = ["Alex Crichton <alex@alexcrichton.com>"] + +[[bin]] +name = "generate-files" +path = "../build.rs" + +[dependencies] +ctest = "0.1" diff --git a/libc/libc-test/run-generated-Cargo.toml b/libc/libc-test/run-generated-Cargo.toml new file mode 100644 index 0000000..64862a5 --- /dev/null +++ b/libc/libc-test/run-generated-Cargo.toml @@ -0,0 +1,19 @@ +# Note that this Cargo.toml is not used by default, it is only intended to be +# used on the BSDs where it is too expensive to compile syntex_syntax in a QEMU +# emulator without KVM. Scripts will move this file into place on the BSD CI. + +[package] +name = "libc-test" +version = "0.1.0" +authors = ["Alex Crichton <alex@alexcrichton.com>"] +build = "build-generated.rs" + +[dependencies] +libc = { path = ".." } + +[[bin]] +name = "libc-test" +path = "src/main-generated.rs" + +[build-dependencies] +gcc = "0.3" diff --git a/libc/libc-test/src/main-generated.rs b/libc/libc-test/src/main-generated.rs new file mode 100644 index 0000000..608fe4e --- /dev/null +++ b/libc/libc-test/src/main-generated.rs @@ -0,0 +1,9 @@ +// Note that this main file is meant to mirror `main.rs` but is only used on the +// BSDs where the generated location of `all.rs` is slightly different + +#![allow(bad_style, improper_ctypes)] +extern crate libc; + +use libc::*; + +include!("../all.rs"); diff --git a/libc/libc-test/src/main.rs b/libc/libc-test/src/main.rs new file mode 100644 index 0000000..fff188d --- /dev/null +++ b/libc/libc-test/src/main.rs @@ -0,0 +1,6 @@ +#![allow(bad_style, improper_ctypes)] +extern crate libc; + +use libc::*; + +include!(concat!(env!("OUT_DIR"), "/all.rs")); | 
