Idris2/support/c/idris_file.c

440 lines
9.4 KiB
C
Raw Normal View History

2020-05-18 16:51:10 +03:00
#include "idris_file.h"
2022-09-21 13:13:15 +03:00
#include "getline.h"
2020-05-18 16:51:10 +03:00
#include <dirent.h>
2020-05-18 16:51:10 +03:00
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <sys/time.h>
#include <time.h>
2020-05-18 16:51:10 +03:00
#include <unistd.h>
#ifdef _WIN32
#include "windows/win_utils.h"
#include <windows.h>
2020-05-18 16:51:10 +03:00
#else
#include <sys/select.h>
#include <sys/wait.h>
2020-05-18 16:51:10 +03:00
#endif
#include "idris_util.h"
2022-09-21 13:13:15 +03:00
FILE *idris2_openFile(char *name, char *mode) {
2020-05-18 16:51:10 +03:00
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
FILE *f = win32_u8fopen(name, mode);
2020-05-18 16:51:10 +03:00
#else
2022-09-21 13:13:15 +03:00
FILE *f = fopen(name, mode);
2020-05-18 16:51:10 +03:00
#endif
2022-09-21 13:13:15 +03:00
return (void *)f;
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
void idris2_closeFile(FILE *f) {
IDRIS2_VERIFY(fclose(f) == 0, "fclose failed: %s", strerror(errno));
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
int idris2_getFileNo(FILE *f) {
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
return win32_getFileNo(f);
#else
2022-09-21 13:13:15 +03:00
return fileno(f);
#endif
}
2022-09-21 13:13:15 +03:00
int idris2_fileError(FILE *f) { return ferror(f); }
2020-05-18 16:51:10 +03:00
int idris2_fileErrno() {
2022-09-21 13:13:15 +03:00
switch (errno) {
case ENOENT:
return 2;
case EACCES:
return 3;
case EEXIST:
return 4;
default:
return (errno + 5);
}
}
int idris2_chmod(const char *path, mode_t mode) {
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
// return _chmod(path, mode);
return 0; /* ??? (from win_hack.c) */
#else
2022-09-21 13:13:15 +03:00
return chmod(path, mode);
#endif
}
2022-09-21 13:13:15 +03:00
int idris2_removeFile(const char *filename) { return remove(filename); }
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
int idris2_fileSize(FILE *f) {
int fd = idris2_getFileNo(f);
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
struct stat buf;
if (fstat(fd, &buf) == 0) {
return (int)(buf.st_size);
} else {
return -1;
}
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
int idris2_fpoll(FILE *f) {
2020-05-18 16:51:10 +03:00
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
return win_fpoll(f);
2020-05-18 16:51:10 +03:00
#else
2022-09-21 13:13:15 +03:00
fd_set x;
struct timeval timeout;
timeout.tv_sec = 1;
timeout.tv_usec = 0;
int fd = idris2_getFileNo(f);
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
FD_ZERO(&x);
FD_SET(fd, &x);
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
int r = select(fd + 1, &x, 0, 0, &timeout);
return r;
2020-05-18 16:51:10 +03:00
#endif
}
2020-05-23 19:42:05 +03:00
void *idris2_popen(const char *cmd, const char *mode) {
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
FILE *f = win32_u8popen(cmd, mode);
2020-05-23 19:42:05 +03:00
#else
2022-09-21 13:13:15 +03:00
FILE *f = popen(cmd, mode);
2020-05-23 19:42:05 +03:00
#endif
2022-09-21 13:13:15 +03:00
return f;
2020-05-23 19:42:05 +03:00
}
2020-05-18 16:51:10 +03:00
2021-11-03 18:10:43 +03:00
int idris2_pclose(void *stream) {
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
int r = _pclose(stream);
IDRIS2_VERIFY(r != -1, "pclose failed");
return r;
#else
2022-09-21 13:13:15 +03:00
int r = pclose(stream);
IDRIS2_VERIFY(WIFEXITED(r), "pclose failed");
return WEXITSTATUS(r);
#endif
}
// seek through the next newline, consuming and
// throwing away anything until then.
2022-09-21 13:13:15 +03:00
int idris2_seekLine(FILE *f) {
while (1) {
int c = fgetc(f);
if (c == -1) {
if (feof(f)) {
return 0;
} else {
return -1;
}
}
if (c == '\n') {
return 0;
}
2022-09-21 13:13:15 +03:00
}
}
2022-09-21 13:13:15 +03:00
char *idris2_readLine(FILE *f) {
char *buffer = NULL;
size_t n = 0;
ssize_t len;
len = getline(&buffer, &n, f);
if (len < 0 && buffer != NULL) {
buffer[0] = '\0'; // Copy Idris 1 behaviour - empty string if nothing read
}
return buffer; // freed by RTS if not NULL
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
char *idris2_readChars(int num, FILE *f) {
char *buffer = malloc((num + 1) * sizeof(char));
size_t len;
len = fread(buffer, sizeof(char), (size_t)num, f);
buffer[len] = '\0';
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
if (len <= 0) {
return NULL;
} else {
return buffer; // freed by RTS
}
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
size_t idris2_readBufferData(FILE *h, char *buffer, size_t loc, size_t max) {
return fread(buffer + loc, sizeof(uint8_t), max, h);
}
2022-09-21 13:13:15 +03:00
int idris2_writeLine(FILE *f, char *str) {
if (fputs(str, f) == EOF) {
return 0;
} else {
return 1;
}
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
size_t idris2_writeBufferData(FILE *h, const char *buffer, size_t loc,
size_t len) {
return fwrite(buffer + loc, sizeof(uint8_t), len, h);
}
2022-09-21 13:13:15 +03:00
int idris2_eof(FILE *f) { return feof(f); }
2020-05-18 16:51:10 +03:00
struct filetime {
int atime_sec;
int atime_nsec;
int mtime_sec;
int mtime_nsec;
int ctime_sec;
int ctime_nsec;
};
struct filetime *idris2_fileTime(FILE *f) {
struct filetime *ft = malloc(sizeof(*ft));
#ifdef _WIN32
if (win32_getFileTime(f, &ft->atime_sec, &ft->atime_nsec, &ft->mtime_sec,
&ft->mtime_nsec, &ft->ctime_sec, &ft->ctime_nsec)) {
ft->atime_sec = -1;
return ft;
}
return ft;
#else
2022-09-21 13:13:15 +03:00
int fd = idris2_getFileNo(f);
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
struct stat buf;
if (fstat(fd, &buf)) {
ft->atime_sec = -1;
return ft;
2022-09-21 13:13:15 +03:00
}
ft->atime_sec = buf.st_atime;
ft->mtime_sec = buf.st_mtime;
ft->ctime_sec = buf.st_ctime;
#if defined(__MACH__) || defined(__APPLE__)
ft->atime_nsec = buf.st_atimespec.tv_nsec;
ft->mtime_nsec = buf.st_mtimespec.tv_nsec;
ft->ctime_nsec = buf.st_ctimespec.tv_nsec;
#elif (_POSIX_VERSION >= 200809L) || defined(__FreeBSD__)
ft->atime_nsec = buf.st_atim.tv_nsec;
ft->mtime_nsec = buf.st_mtim.tv_nsec;
ft->ctime_nsec = buf.st_ctim.tv_nsec;
#else
ft->atime_nsec = 0;
ft->mtime_nsec = 0;
ft->ctime_nsec = 0;
#endif
return ft;
#endif
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
int idris2_fileModifiedTime(FILE *f) {
int fd = idris2_getFileNo(f);
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
struct stat buf;
if (fstat(fd, &buf) == 0) {
return buf.st_mtime;
} else {
return -1;
}
2020-05-18 16:51:10 +03:00
}
int idris2_filetimeAccessTimeSec(struct filetime *ft) { return ft->atime_sec; }
2020-05-18 16:51:10 +03:00
int idris2_filetimeAccessTimeNsec(struct filetime *ft) {
return ft->atime_nsec;
}
int idris2_filetimeModifiedTimeSec(struct filetime *ft) {
return ft->mtime_sec;
}
int idris2_filetimeModifiedTimeNsec(struct filetime *ft) {
return ft->mtime_nsec;
}
int idris2_filetimeStatusTimeSec(struct filetime *ft) { return ft->ctime_sec; }
int idris2_filetimeStatusTimeNsec(struct filetime *ft) {
return ft->ctime_nsec;
2020-05-18 16:51:10 +03:00
}
2022-09-21 13:13:15 +03:00
int idris2_fileIsTTY(FILE *f) {
int fd = idris2_getFileNo(f);
#ifdef _WIN32
2022-09-21 13:13:15 +03:00
return win32_isTTY(fd);
#else
2022-09-21 13:13:15 +03:00
return isatty(fd);
#endif
}
2022-09-21 13:13:15 +03:00
FILE *idris2_stdin() { return stdin; }
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
FILE *idris2_stdout() { return stdout; }
2020-05-18 16:51:10 +03:00
2022-09-21 13:13:15 +03:00
FILE *idris2_stderr() { return stderr; }
struct child_process {
#ifdef _WIN32
HANDLE hProcess;
#endif
pid_t pid;
FILE *in;
FILE *out;
};
// Open a bi-directional pipe, returning the above structure
// with pid and two file handles.
struct child_process *idris2_popen2(char *cmd) {
#ifdef _WIN32
SECURITY_ATTRIBUTES saAttr;
HANDLE pipes[4];
STARTUPINFO si;
PROCESS_INFORMATION pi;
ZeroMemory(&pi, sizeof(PROCESS_INFORMATION));
saAttr.nLength = sizeof(SECURITY_ATTRIBUTES);
saAttr.bInheritHandle = TRUE;
saAttr.lpSecurityDescriptor = NULL;
if (!CreatePipe(&pipes[0], &pipes[1], &saAttr, 0) ||
!CreatePipe(&pipes[2], &pipes[3], &saAttr, 0)) {
return NULL;
}
char cmdline[4096];
int len = snprintf(cmdline, 4096, "cmd /c %s", cmd);
if (len > 4095 || len < 0) {
return NULL;
}
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
si.hStdInput = pipes[2];
si.hStdOutput = pipes[1];
// si.hStdError = pipes[1];
si.dwFlags |= STARTF_USESTDHANDLES;
SetHandleInformation(pipes[3], HANDLE_FLAG_INHERIT, 0);
SetHandleInformation(pipes[0], HANDLE_FLAG_INHERIT, 0);
if (!CreateProcess(NULL, cmdline, NULL, NULL, TRUE, 0, NULL, NULL, &si,
&pi)) {
return NULL;
}
struct child_process *rval = malloc(sizeof(struct child_process));
int in_fd = _open_osfhandle((intptr_t)pipes[3], _O_WRONLY);
int out_fd = _open_osfhandle((intptr_t)pipes[0], _O_RDONLY);
CloseHandle(pipes[1]);
CloseHandle(pipes[2]);
// We close thread handle to not to store nor leak it, but
// we do not close the process handle to be able to get the exit code
CloseHandle(pi.hThread);
rval->in = _fdopen(in_fd, "w");
rval->out = _fdopen(out_fd, "r");
rval->hProcess = pi.hProcess;
rval->pid = pi.dwProcessId;
return rval;
#else
int pipes[4];
int err = 0;
err = pipe(&pipes[0]);
if (err) {
return NULL;
}
err = pipe(&pipes[2]);
if (err) {
close(pipes[0]);
close(pipes[1]);
return NULL;
}
// make sure no buffers left unflushed before forking
// to save from double-printing
fflush(stdout);
fflush(stderr);
pid_t pid = fork();
if (pid < 0) {
perror("fork");
return NULL;
} else if (pid > 0) {
struct child_process *rval = malloc(sizeof(struct child_process));
close(pipes[1]);
close(pipes[2]);
rval->in = fdopen(pipes[3], "w");
rval->out = fdopen(pipes[0], "r");
rval->pid = pid;
return rval;
} else {
close(STDOUT_FILENO);
dup2(pipes[1], STDOUT_FILENO);
close(pipes[0]);
close(pipes[1]);
close(STDIN_FILENO);
dup2(pipes[2], STDIN_FILENO);
close(pipes[2]);
close(pipes[3]);
err = execlp("/bin/sh", "sh", "-c", cmd, NULL);
// We only reach this point if there is an error.
// Maybe report something to stderr so the user knows what's up?
perror("execl");
exit(err);
}
#endif
}
int idris2_popen2ChildPid(struct child_process *ptr) {
if (!ptr)
return 0;
return ptr->pid;
}
void *idris2_popen2ChildHandler(struct child_process *ptr) {
#ifdef _WIN32
if (!ptr)
return NULL;
return ptr->hProcess;
#else
// We don't have special child handler in POSIX systems
return NULL;
#endif
}
FILE *idris2_popen2FileIn(struct child_process *ptr) {
if (!ptr)
return NULL;
return ptr->in;
}
FILE *idris2_popen2FileOut(struct child_process *ptr) {
if (!ptr)
return NULL;
return ptr->out;
}
int idris2_popen2WaitByHandler(void *hProcess) {
#ifdef _WIN32
DWORD r;
DWORD wres = WaitForSingleObject(hProcess, INFINITE);
IDRIS2_VERIFY(wres == WAIT_OBJECT_0, "waiting after popen2 failed");
int eres = GetExitCodeProcess(hProcess, &r);
IDRIS2_VERIFY(eres != 0, "getting exitcode after popen2 failed");
CloseHandle(hProcess);
return (int)r;
#else
perror("cannot wait by handler in POSIX system");
return -1;
#endif
}
int idris2_popen2WaitByPid(pid_t pid) {
#ifdef _WIN32
perror("cannot wait by pid in Windows");
return -1;
#else
int r = -1;
int w = waitpid(pid, &r, 0);
IDRIS2_VERIFY(w != -1, "waitpid after popen2 failed");
IDRIS2_VERIFY(WIFEXITED(r), "process launched by popen2 didn't exit well");
return WEXITSTATUS(r);
#endif
}