#include "master.hpp" namespace factor { /* Simple wrappers for ANSI C I/O functions, used for bootstrapping. Note the ugly loop logic in almost every function; we have to handle EINTR and restart the operation if the system call was interrupted. Naive applications don't do this, but then they quickly fail if one enables itimer()s or other signals. The Factor library provides platform-specific code for Unix and Windows with many more capabilities so these words are not usually used in normal operation. */ void factor_vm::init_c_io() { special_objects[OBJ_STDIN] = allot_alien(false_object,(cell)stdin); special_objects[OBJ_STDOUT] = allot_alien(false_object,(cell)stdout); special_objects[OBJ_STDERR] = allot_alien(false_object,(cell)stderr); } void factor_vm::io_error() { #ifndef WINCE if(errno == EINTR) return; #endif general_error(ERROR_IO,tag_fixnum(errno),false_object,NULL); } size_t safe_fread(void *ptr, size_t size, size_t nitems, FILE *stream) { size_t items_read = 0; do { items_read += fread((void*)((int*)ptr+items_read*size),size,nitems-items_read,stream); } while(items_read != nitems && errno == EINTR); return items_read; } size_t safe_fwrite(void *ptr, size_t size, size_t nitems, FILE *stream) { size_t items_written = 0; do { items_written += fwrite((void*)((int*)ptr+items_written*size),size,nitems-items_written,stream); } while(items_written != nitems && errno == EINTR); return items_written; } int safe_fclose(FILE *stream) { int ret = 0; do { ret = fclose(stream); } while(ret != 0 && errno == EINTR); return ret; } void factor_vm::primitive_fopen() { data_root mode(ctx->pop(),this); data_root path(ctx->pop(),this); mode.untag_check(this); path.untag_check(this); FILE *file; do { file = fopen((char *)(path.untagged() + 1), (char *)(mode.untagged() + 1)); if(file == NULL) io_error(); } while(errno == EINTR); ctx->push(allot_alien(file)); } FILE *factor_vm::pop_file_handle() { return (FILE *)alien_offset(ctx->pop()); } void factor_vm::primitive_fgetc() { FILE *file = pop_file_handle(); do { int c = fgetc(file); if(c == EOF) { if(feof(file)) { ctx->push(false_object); break; } else io_error(); } else { ctx->push(tag_fixnum(c)); break; } } while(errno == EINTR); } void factor_vm::primitive_fread() { FILE *file = pop_file_handle(); fixnum size = unbox_array_size(); if(size == 0) { ctx->push(tag(allot_string(0,0))); return; } data_root buf(allot_uninitialized_array(size),this); for(;;) { int c = safe_fread(buf.untagged() + 1,1,size,file); if(c == 0) { if(feof(file)) { ctx->push(false_object); break; } else io_error(); } else { if(feof(file)) { byte_array *new_buf = allot_byte_array(c); memcpy(new_buf + 1, buf.untagged() + 1,c); buf = new_buf; } ctx->push(buf.value()); break; } } } void factor_vm::primitive_fputc() { FILE *file = pop_file_handle(); fixnum ch = to_fixnum(ctx->pop()); do { if(fputc(ch,file) == EOF) io_error(); else break; } while(errno == EINTR); } void factor_vm::primitive_fwrite() { FILE *file = pop_file_handle(); byte_array *text = untag_check(ctx->pop()); cell length = array_capacity(text); char *string = (char *)(text + 1); if(length == 0) return; size_t written = safe_fwrite(string,1,length,file); if(written != length) io_error(); } void factor_vm::primitive_ftell() { FILE *file = pop_file_handle(); off_t offset; do { if((offset = FTELL(file)) == -1) io_error(); else break; } while(errno == EINTR); ctx->push(from_signed_8(offset)); } void factor_vm::primitive_fseek() { int whence = to_fixnum(ctx->pop()); FILE *file = pop_file_handle(); off_t offset = to_signed_8(ctx->pop()); switch(whence) { case 0: whence = SEEK_SET; break; case 1: whence = SEEK_CUR; break; case 2: whence = SEEK_END; break; default: critical_error("Bad value for whence",whence); break; } do { if(FSEEK(file,offset,whence) == -1) io_error(); else break; } while(errno == EINTR); } void factor_vm::primitive_fflush() { FILE *file = pop_file_handle(); do { if(fflush(file) == EOF) io_error(); else break; } while(errno == EINTR); } void factor_vm::primitive_fclose() { FILE *file = pop_file_handle(); if(safe_fclose(file) == EOF) io_error(); } /* This function is used by FFI I/O. Accessing the errno global directly is not portable, since on some libc's errno is not a global but a funky macro that reads thread-local storage. */ VM_C_API int err_no() { return errno; } VM_C_API void clear_err_no() { errno = 0; } }