#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include struct init_funcs_t { void (*func)(); void (**array_start)(); void (**array_end)(); }; extern "C" char** environ; extern "C" void _init_libc(char** environ, init_funcs_t init_funcs, init_funcs_t fini_funcs) { if (::environ == nullptr) ::environ = environ; if (uthread* self = reinterpret_cast(syscall(SYS_GET_TLS))) { self->cleanup_stack = nullptr; self->id = syscall(SYS_PTHREAD_SELF); self->errno_ = 0; self->cancel_type = PTHREAD_CANCEL_DEFERRED; self->cancel_state = PTHREAD_CANCEL_ENABLE; self->canceled = false; } else { alignas(uthread) static uint8_t storage[sizeof(uthread) + sizeof(uintptr_t)]; uthread& uthread = *reinterpret_cast(storage); uthread = { .self = &uthread, .master_tls_addr = nullptr, .master_tls_size = 0, .cleanup_stack = nullptr, .id = static_cast(syscall(SYS_PTHREAD_SELF)), .errno_ = 0, .cancel_type = PTHREAD_CANCEL_DEFERRED, .cancel_state = PTHREAD_CANCEL_ENABLE, .canceled = false, }; uthread.dtv[0] = 0; syscall(SYS_SET_TLS, &uthread); } // call global constructors if (init_funcs.func) init_funcs.func(); const size_t init_array_count = init_funcs.array_end - init_funcs.array_start; for (size_t i = 0; i < init_array_count; i++) init_funcs.array_start[i](); // register global destructors const size_t fini_array_count = fini_funcs.array_end - fini_funcs.array_start; for (size_t i = 0; i < fini_array_count; i++) atexit(fini_funcs.array_start[i]); if (fini_funcs.func) atexit(fini_funcs.func); } void _exit(int status) { syscall(SYS_EXIT, status); ASSERT_NOT_REACHED(); } long syscall(long syscall, ...) { va_list args; va_start(args, syscall); uintptr_t arg1 = va_arg(args, uintptr_t); uintptr_t arg2 = va_arg(args, uintptr_t); uintptr_t arg3 = va_arg(args, uintptr_t); uintptr_t arg4 = va_arg(args, uintptr_t); uintptr_t arg5 = va_arg(args, uintptr_t); va_end(args); long ret; do ret = Kernel::syscall(syscall, arg1, arg2, arg3, arg4, arg5); while (ret == -ERESTART); if (ret < 0) { errno = -ret; return -1; } return ret; } int close(int fd) { pthread_testcancel(); return syscall(SYS_CLOSE, fd); } ssize_t read(int fildes, void* buf, size_t nbyte) { pthread_testcancel(); return syscall(SYS_READ, fildes, buf, nbyte); } ssize_t write(int fildes, const void* buf, size_t nbyte) { pthread_testcancel(); return syscall(SYS_WRITE, fildes, buf, nbyte); } ssize_t readlink(const char* __restrict path, char* __restrict buf, size_t bufsize) { return readlinkat(AT_FDCWD, path, buf, bufsize); } ssize_t readlinkat(int fd, const char* __restrict path, char* __restrict buf, size_t bufsize) { return syscall(SYS_READLINKAT, fd, path, buf, bufsize); } ssize_t pread(int fildes, void* buf, size_t nbyte, off_t offset) { pthread_testcancel(); return syscall(SYS_PREAD, fildes, buf, nbyte, offset); } ssize_t pwrite(int fildes, const void* buf, size_t nbyte, off_t offset) { pthread_testcancel(); return syscall(SYS_PWRITE, fildes, buf, nbyte, offset); } off_t lseek(int fildes, off_t offset, int whence) { return syscall(SYS_SEEK, fildes, offset, whence); } int ftruncate(int fildes, off_t length) { return syscall(SYS_TRUNCATE, fildes, length); } int fsync(int fildes) { pthread_testcancel(); return syscall(SYS_FSYNC, fildes); } int dup(int fildes) { return fcntl(fildes, F_DUPFD, 0); } int dup2(int fildes, int fildes2) { return syscall(SYS_DUP2, fildes, fildes2); } int isatty(int fildes) { return syscall(SYS_ISATTY, fildes) >= 0; } int gethostname(char* name, size_t namelen) { FILE* fp = fopen("/etc/hostname", "r"); if (fp == NULL) return -1; size_t nread = fread(name, 1, namelen - 1, fp); while (nread > 0 && name[nread - 1] == '\n') nread--; name[nread] = '\0'; fclose(fp); return 0; } static int exec_impl_shebang(FILE* fp, const char* pathname, char* const* argv, char* const* envp, int shebang_depth); static int exec_impl(const char* pathname, char* const* argv, char* const* envp, bool do_path_resolution, int shebang_depth = 0) { if (shebang_depth > 100) { errno = ELOOP; return -1; } char buffer[PATH_MAX]; if (do_path_resolution && strchr(pathname, '/') == nullptr) { const char* cur = getenv("PATH"); if (!cur) { errno = ENOENT; return -1; } char* resolved = nullptr; while (*cur) { const char* end = strchrnul(cur, ':'); size_t len = end - cur; ASSERT(strlen(pathname) + 1 + len < sizeof(buffer)); strncpy(buffer, cur, len); strcat(buffer, "/"); strcat(buffer, pathname); struct stat st; if (stat(buffer, &st) == 0) { resolved = buffer; break; } cur = end; if (*cur) cur++; } if (!resolved) { errno = ENOENT; return -1; } pathname = resolved; } if (access(pathname, X_OK) == -1) return -1; if (FILE* fp = fopen(pathname, "r")) { char shebang[2]; if (fread(shebang, 1, 2, fp) == 2 && shebang[0] == '#' && shebang[1] == '!') return exec_impl_shebang(fp, pathname, argv, envp, shebang_depth); fclose(fp); } return syscall(SYS_EXEC, pathname, argv, envp); } static int exec_impl_shebang(FILE* fp, const char* pathname, char* const* argv, char* const* envp, int shebang_depth) { constexpr size_t buffer_len = PATH_MAX + 1 + ARG_MAX + 1; char* buffer = static_cast(malloc(buffer_len)); if (buffer == nullptr) { fclose(fp); return -1; } if (fgets(buffer, buffer_len, fp) == nullptr) { free(buffer); return -1; } const auto sv_trim_whitespace = [](BAN::StringView sv) -> BAN::StringView { while (!sv.empty() && isspace(sv.front())) sv = sv.substring(1); while (!sv.empty() && isspace(sv.back())) sv = sv.substring(0, sv.size() - 1); return sv; }; BAN::StringView buffer_sv = buffer; if (buffer_sv.back() != '\n') { free(buffer); errno = ENOEXEC; return -1; } buffer_sv = sv_trim_whitespace(buffer_sv); BAN::StringView interpreter, argument; if (auto space = buffer_sv.find([](char ch) -> bool { return isspace(ch); }); !space.has_value()) interpreter = buffer_sv; else { interpreter = sv_trim_whitespace(buffer_sv.substring(0, space.value())); argument = sv_trim_whitespace(buffer_sv.substring(space.value())); } if (interpreter.empty()) { free(buffer); errno = ENOEXEC; return -1; } // null terminate interpreter and argument const_cast(interpreter.data())[interpreter.size()] = '\0'; if (!argument.empty()) const_cast(argument.data())[argument.size()] = '\0'; size_t old_argc = 0; while (argv[old_argc]) old_argc++; const size_t extra_args = 1 + !argument.empty(); char** new_argv = static_cast(malloc((extra_args + old_argc + 1) * sizeof(char*)));; if (new_argv == nullptr) { free(buffer); return -1; } new_argv[0] = const_cast(pathname); if (!argument.empty()) new_argv[1] = const_cast(argument.data()); for (size_t i = 0; i < old_argc; i++) new_argv[i + extra_args] = argv[i]; new_argv[old_argc + extra_args] = nullptr; exec_impl(interpreter.data(), new_argv, envp, true, shebang_depth + 1); free(new_argv); free(buffer); return -1; } static int execl_impl(const char* pathname, const char* arg0, va_list ap, bool has_env, bool do_path_resolution) { int argc = 1; va_list ap2; va_copy(ap2, ap); while (va_arg(ap2, char*)) argc++; va_end(ap2); char** argv = static_cast(malloc((argc + 1) * sizeof(char*))); if (argv == nullptr) { errno = ENOMEM; return -1; } argv[0] = const_cast(arg0); for (int i = 1; i < argc; i++) argv[i] = va_arg(ap, char*); argv[argc] = nullptr; char** envp = environ; if (has_env) { va_arg(ap, char*); envp = va_arg(ap, char**);; } exec_impl(pathname, argv, envp, do_path_resolution); free(argv); return -1; } int execl(const char* pathname, const char* arg0, ...) { va_list ap; va_start(ap, arg0); int ret = execl_impl(pathname, arg0, ap, false, false); va_end(ap); return ret; } int execlp(const char* pathname, const char* arg0, ...) { va_list ap; va_start(ap, arg0); int ret = execl_impl(pathname, arg0, ap, false, true); va_end(ap); return ret; } int execle(const char* pathname, const char* arg0, ...) { va_list ap; va_start(ap, arg0); int ret = execl_impl(pathname, arg0, ap, true, false); va_end(ap); return ret; } int execv(const char* pathname, char* const argv[]) { return exec_impl(pathname, argv, environ, false); } int execve(const char* pathname, char* const argv[], char* const envp[]) { return exec_impl(pathname, argv, envp, false); } int execvp(const char* pathname, char* const argv[]) { return exec_impl(pathname, argv, environ, true); } pid_t fork(void) { _pthread_call_atfork(_PTHREAD_ATFORK_PREPARE); const pid_t pid = syscall(SYS_FORK); if (pid == -1) return -1; _pthread_call_atfork(pid ? _PTHREAD_ATFORK_PARENT : _PTHREAD_ATFORK_CHILD); return pid; } int pipe(int fildes[2]) { return syscall(SYS_PIPE, fildes); } unsigned int sleep(unsigned int seconds) { pthread_testcancel(); unsigned int ret = syscall(SYS_SLEEP, seconds); if (ret > 0) errno = EINTR; return ret; } int usleep(useconds_t usec) { timespec ts; ts.tv_sec = usec / 1'000'000; ts.tv_nsec = (usec % 1'000'000) * 1000; return nanosleep(&ts, nullptr); } char* getcwd(char* buf, size_t size) { if (size == 0) { errno = EINVAL; return nullptr; } if (syscall(SYS_GETCWD, buf, size) == 0) return nullptr; return buf; } int chdir(const char* path) { return syscall(SYS_CHDIR, path); } int fchdir(int fildes) { return syscall(SYS_FCHDIR, fildes); } int chown(const char* path, uid_t owner, gid_t group) { return fchownat(AT_FDCWD, path, owner, group, 0); } int lchown(const char* path, uid_t owner, gid_t group) { return fchownat(AT_FDCWD, path, owner, group, AT_SYMLINK_NOFOLLOW); } int fchown(int fildes, uid_t owner, gid_t group) { return fchownat(fildes, nullptr, owner, group, 0); } int fchownat(int fd, const char* path, uid_t owner, gid_t group, int flag) { return syscall(SYS_FCHOWNAT, fd, path, owner, group, flag); } void sync(void) { syscall(SYS_SYNC, false); } void syncsync(int should_block) { syscall(SYS_SYNC, should_block); } int fdatasync(int fildes) { pthread_testcancel(); (void)fildes; dprintln("TODO: fdatasync"); return syscall(SYS_SYNC, true); } int unlink(const char* path) { return unlinkat(AT_FDCWD, path, 0); } int unlinkat(int fd, const char* path, int flag) { return syscall(SYS_UNLINKAT, fd, path, flag); } int rmdir(const char* path) { return unlinkat(AT_FDCWD, path, AT_REMOVEDIR); } char* optarg = nullptr; int opterr = 1; int optind = 1; int optopt = 0; int getopt(int argc, char* const argv[], const char* optstring) { if (optind >= argc) return -1; static int idx = 1; const char* current = argv[optind]; // if "--" is encountered, no more options are parsed if (idx == -1) return -1; // if current is nullptr, does not start with '-' or is string "-", return -1 if (current == nullptr || current[0] != '-' || current[1] == '\0') return -1; // if current points to string "--" increment optind and return -1 if (current[1] == '-' && current[2] == '\0') { idx = -1; optind++; return -1; } for (size_t i = 0; optstring[i]; i++) { if (optstring[i] == ':') continue; if (current[idx] != optstring[i]) continue; if (optstring[i + 1] == ':') { if (current[idx + 1]) { optarg = const_cast(current + idx + 1); optind += 1; } else { optarg = const_cast(argv[optind + 1]); optind += 2; } idx = 1; if (optind > argc) { if (opterr && optstring[0] != ':') fprintf(stderr, "%s: option requires an argument -- %c\n", argv[0], optstring[i]); optopt = optstring[i]; return optstring[0] == ':' ? ':' : '?'; } return optstring[i]; } else { if (current[++idx] == '\0') { idx = 1; optind++; } return optstring[i]; } } if (opterr && optstring[0] != ':') fprintf(stderr, "%s: illegal option -- %c\n", argv[0], current[idx]); if (current[++idx] == '\0') { idx = 1; optind++; } return '?'; } int getpagesize(void) { return PAGE_SIZE; } char* getpass(const char* prompt) { static char buffer[PASS_MAX]; int fd = open("/dev/tty", O_RDWR); if (fd == -1) return NULL; termios orig, temp; tcgetattr(fd, &orig); char* ret = nullptr; ssize_t total_read = 0; if (write(fd, prompt, strlen(prompt)) < 0) goto error; temp = orig; temp.c_lflag &= ~ECHO; tcsetattr(fd, TCSANOW, &temp); while (total_read == 0 || buffer[total_read - 1] != '\n') { ssize_t nread = read(fd, buffer + total_read, sizeof(buffer) - total_read - 1); if (nread < 0) goto error; total_read += nread; } buffer[total_read - 1] = '\0'; ret = buffer; write(fd, "\n", 1); error: tcsetattr(fd, TCSANOW, &orig); close(fd); return ret; } pid_t getppid(void) { return syscall(SYS_GET_PPID); } pid_t getpid(void) { return syscall(SYS_GET_PID); } uid_t getuid(void) { return syscall(SYS_GET_UID); } gid_t getgid(void) { return syscall(SYS_GET_GID); } uid_t geteuid(void) { return syscall(SYS_GET_EUID); } gid_t getegid(void) { return syscall(SYS_GET_EGID); } pid_t getpgrp(void) { return getpgid(0); } pid_t getpgid(pid_t pid) { return syscall(SYS_GET_PGID, pid); } int tcgetpgrp(int fildes) { return syscall(SYS_TCGETPGRP, fildes); } int seteuid(uid_t uid) { return syscall(SYS_SET_EUID, uid); } int setegid(gid_t gid) { return syscall(SYS_SET_EGID, gid); } int setuid(uid_t uid) { return syscall(SYS_SET_UID, uid); } int setgid(gid_t gid) { return syscall(SYS_SET_GID, gid); } int setsid(void) { return syscall(SYS_SET_SID); } int setreuid(uid_t ruid, uid_t euid) { return syscall(SYS_SET_REUID, ruid, euid); } int setregid(gid_t rgid, gid_t egid) { return syscall(SYS_SET_REGID, rgid, egid); } pid_t setpgrp(void) { setpgid(0, 0); return getpgrp(); } int setpgid(pid_t pid, pid_t pgid) { return syscall(SYS_SET_PGID, pid, pgid); } int tcsetpgrp(int fildes, pid_t pgid_id) { return syscall(SYS_TCSETPGRP, fildes, pgid_id); } char* getlogin(void) { static char buffer[LOGIN_NAME_MAX]; auto* pw = getpwuid(geteuid()); if (pw == nullptr) return nullptr; strncpy(buffer, pw->pw_name, LOGIN_NAME_MAX - 1); buffer[LOGIN_NAME_MAX - 1] = '\0'; endpwent(); return buffer; } char* ttyname(int fildes) { static char storage[_POSIX_TTY_NAME_MAX]; if (syscall(SYS_TTYNAME, fildes, storage) == -1) return nullptr; return storage; } int access(const char* path, int amode) { return syscall(SYS_ACCESS, path, amode); } unsigned alarm(unsigned seconds) { itimerval value, ovalue; value.it_value.tv_sec = seconds; value.it_value.tv_usec = 0; value.it_interval.tv_sec = 0; value.it_interval.tv_usec = 0; setitimer(ITIMER_REAL, &value, &ovalue); return ovalue.it_value.tv_sec; } int symlink(const char* path1, const char* path2) { return symlinkat(path1, AT_FDCWD, path2); } int symlinkat(const char* path1, int fd, const char* path2) { return syscall(SYS_SYMLINKAT, path1, fd, path2); } int link(const char* path1, const char* path2) { return linkat(AT_FDCWD, path1, AT_FDCWD, path2, 0); } int linkat(int fd1, const char *path1, int fd2, const char *path2, int flag) { return syscall(SYS_HARDLINKAT, fd1, path1, fd2, path2, flag); } size_t confstr(int name, char* buf, size_t len) { (void)buf; (void)len; switch (name) { #define CONFSTR_CASE(name) case name: return 0; CONFSTR_CASE(_CS_PATH) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFF32_CFLAGS) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFF32_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFF32_LIBS) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFFBIG_CFLAGS) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFFBIG_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V6_ILP32_OFFBIG_LIBS) CONFSTR_CASE(_CS_POSIX_V6_LP64_OFF64_CFLAGS) CONFSTR_CASE(_CS_POSIX_V6_LP64_OFF64_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V6_LP64_OFF64_LIBS) CONFSTR_CASE(_CS_POSIX_V6_LPBIG_OFFBIG_CFLAGS) CONFSTR_CASE(_CS_POSIX_V6_LPBIG_OFFBIG_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V6_LPBIG_OFFBIG_LIBS) CONFSTR_CASE(_CS_POSIX_V6_WIDTH_RESTRICTED_ENVS) CONFSTR_CASE(_CS_V6_ENV) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFF32_CFLAGS) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFF32_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFF32_LIBS) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFFBIG_CFLAGS) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFFBIG_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V7_ILP32_OFFBIG_LIBS) CONFSTR_CASE(_CS_POSIX_V7_LP64_OFF64_CFLAGS) CONFSTR_CASE(_CS_POSIX_V7_LP64_OFF64_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V7_LP64_OFF64_LIBS) CONFSTR_CASE(_CS_POSIX_V7_LPBIG_OFFBIG_CFLAGS) CONFSTR_CASE(_CS_POSIX_V7_LPBIG_OFFBIG_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V7_LPBIG_OFFBIG_LIBS) CONFSTR_CASE(_CS_POSIX_V7_THREADS_CFLAGS) CONFSTR_CASE(_CS_POSIX_V7_THREADS_LDFLAGS) CONFSTR_CASE(_CS_POSIX_V7_WIDTH_RESTRICTED_ENVS) CONFSTR_CASE(_CS_V7_ENV) #undef CONFSTR_CASE } errno = EINVAL; return 0; } long pathconf(const char* path, int name); long sysconf(int name) { switch (name) { #define POSIX2_CASE(name) case _SC_2_##name: return _POSIX2_##name; POSIX2_CASE(C_BIND) POSIX2_CASE(C_DEV) POSIX2_CASE(CHAR_TERM) POSIX2_CASE(FORT_DEV) POSIX2_CASE(FORT_RUN) POSIX2_CASE(LOCALEDEF) POSIX2_CASE(PBS) POSIX2_CASE(PBS_ACCOUNTING) POSIX2_CASE(PBS_CHECKPOINT) POSIX2_CASE(PBS_LOCATE) POSIX2_CASE(PBS_MESSAGE) POSIX2_CASE(PBS_TRACK) POSIX2_CASE(SW_DEV) POSIX2_CASE(UPE) POSIX2_CASE(VERSION) #undef POSIX2_CASE #define POSIX_CASE(name) case _SC_##name: return _POSIX_##name; POSIX_CASE(ADVISORY_INFO) POSIX_CASE(AIO_LISTIO_MAX) POSIX_CASE(AIO_MAX) POSIX_CASE(ARG_MAX) POSIX_CASE(ASYNCHRONOUS_IO) POSIX_CASE(BARRIERS) POSIX_CASE(CHILD_MAX) POSIX_CASE(CLOCK_SELECTION) POSIX_CASE(CPUTIME) POSIX_CASE(DELAYTIMER_MAX) POSIX_CASE(FSYNC) POSIX_CASE(HOST_NAME_MAX) POSIX_CASE(IPV6) POSIX_CASE(JOB_CONTROL) POSIX_CASE(LOGIN_NAME_MAX) POSIX_CASE(MAPPED_FILES) POSIX_CASE(MEMLOCK) POSIX_CASE(MEMLOCK_RANGE) POSIX_CASE(MEMORY_PROTECTION) POSIX_CASE(MESSAGE_PASSING) POSIX_CASE(MONOTONIC_CLOCK) POSIX_CASE(MQ_OPEN_MAX) POSIX_CASE(MQ_PRIO_MAX) POSIX_CASE(NGROUPS_MAX) POSIX_CASE(OPEN_MAX) POSIX_CASE(PRIORITIZED_IO) POSIX_CASE(PRIORITY_SCHEDULING) POSIX_CASE(RAW_SOCKETS) POSIX_CASE(RE_DUP_MAX) POSIX_CASE(READER_WRITER_LOCKS) POSIX_CASE(REALTIME_SIGNALS) POSIX_CASE(REGEXP) POSIX_CASE(RTSIG_MAX) POSIX_CASE(SAVED_IDS) POSIX_CASE(SEM_NSEMS_MAX) POSIX_CASE(SEM_VALUE_MAX) POSIX_CASE(SEMAPHORES) POSIX_CASE(SHARED_MEMORY_OBJECTS) POSIX_CASE(SHELL) POSIX_CASE(SIGQUEUE_MAX) POSIX_CASE(SPAWN) POSIX_CASE(SPIN_LOCKS) POSIX_CASE(SPORADIC_SERVER) POSIX_CASE(SS_REPL_MAX) POSIX_CASE(STREAM_MAX) POSIX_CASE(SYMLOOP_MAX) POSIX_CASE(SYNCHRONIZED_IO) POSIX_CASE(THREAD_ATTR_STACKADDR) POSIX_CASE(THREAD_ATTR_STACKSIZE) POSIX_CASE(THREAD_CPUTIME) POSIX_CASE(THREAD_DESTRUCTOR_ITERATIONS) POSIX_CASE(THREAD_KEYS_MAX) POSIX_CASE(THREAD_PRIO_INHERIT) POSIX_CASE(THREAD_PRIO_PROTECT) POSIX_CASE(THREAD_PRIORITY_SCHEDULING) POSIX_CASE(THREAD_PROCESS_SHARED) POSIX_CASE(THREAD_ROBUST_PRIO_INHERIT) POSIX_CASE(THREAD_ROBUST_PRIO_PROTECT) POSIX_CASE(THREAD_SAFE_FUNCTIONS) POSIX_CASE(THREAD_SPORADIC_SERVER) POSIX_CASE(THREAD_THREADS_MAX) POSIX_CASE(THREADS) POSIX_CASE(TIMEOUTS) POSIX_CASE(TIMER_MAX) POSIX_CASE(TIMERS) POSIX_CASE(TRACE) POSIX_CASE(TRACE_EVENT_FILTER) POSIX_CASE(TRACE_EVENT_NAME_MAX) POSIX_CASE(TRACE_INHERIT) POSIX_CASE(TRACE_LOG) POSIX_CASE(TRACE_NAME_MAX) POSIX_CASE(TRACE_SYS_MAX) POSIX_CASE(TRACE_USER_EVENT_MAX) POSIX_CASE(TTY_NAME_MAX) POSIX_CASE(TYPED_MEMORY_OBJECTS) POSIX_CASE(TZNAME_MAX) POSIX_CASE(V7_ILP32_OFF32) POSIX_CASE(V7_ILP32_OFFBIG) POSIX_CASE(V7_LP64_OFF64) POSIX_CASE(V7_LPBIG_OFFBIG) POSIX_CASE(V6_ILP32_OFF32) POSIX_CASE(V6_ILP32_OFFBIG) POSIX_CASE(V6_LP64_OFF64) POSIX_CASE(V6_LPBIG_OFFBIG) POSIX_CASE(VERSION) #undef POSIX_CASE #define LIMITS_CASE(name) case _SC_##name: return name; LIMITS_CASE(AIO_PRIO_DELTA_MAX) LIMITS_CASE(ATEXIT_MAX) LIMITS_CASE(BC_BASE_MAX) LIMITS_CASE(BC_DIM_MAX) LIMITS_CASE(BC_SCALE_MAX) LIMITS_CASE(BC_STRING_MAX) LIMITS_CASE(COLL_WEIGHTS_MAX) LIMITS_CASE(EXPR_NEST_MAX) LIMITS_CASE(IOV_MAX) LIMITS_CASE(LINE_MAX) #undef LIMITS_CASE #define XOPEN_CASE(name) case _SC_XOPEN_##name: return _XOPEN_##name; XOPEN_CASE(CRYPT) XOPEN_CASE(ENH_I18N) XOPEN_CASE(REALTIME) XOPEN_CASE(REALTIME_THREADS) XOPEN_CASE(SHM) XOPEN_CASE(STREAMS) XOPEN_CASE(UNIX) XOPEN_CASE(UUCP) XOPEN_CASE(VERSION) #undef XOPEN_CASE case _SC_PAGE_SIZE: case _SC_PAGESIZE: return getpagesize(); case _SC_CLK_TCK: return 100; case _SC_GETGR_R_SIZE_MAX: return 1024; case _SC_GETPW_R_SIZE_MAX: return 1024; case _SC_THREAD_STACK_MIN: return PTHREAD_STACK_MIN; } errno = EINVAL; return -1; }