| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | #include "master.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 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 init_c_io(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	userenv[IN_ENV] = allot_alien(F,(CELL)stdin); | 
					
						
							|  |  |  | 	userenv[OUT_ENV] = allot_alien(F,(CELL)stdout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void io_error(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-10-15 19:59:35 -04:00
										 |  |  | #ifndef WINCE
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 	if(errno == EINTR) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2007-10-15 19:59:35 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	CELL error = tag_object(from_char_string(strerror(errno))); | 
					
						
							|  |  |  | 	general_error(ERROR_IO,error,F,NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fopen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *mode = unbox_char_string(); | 
					
						
							|  |  |  | 	REGISTER_C_STRING(mode); | 
					
						
							|  |  |  | 	char *path = unbox_char_string(); | 
					
						
							|  |  |  | 	UNREGISTER_C_STRING(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		FILE *file = fopen(path,mode); | 
					
						
							|  |  |  | 		if(file == NULL) | 
					
						
							|  |  |  | 			io_error(); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			box_alien(file); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fgetc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE* file = unbox_alien(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		int c = fgetc(file); | 
					
						
							|  |  |  | 		if(c == EOF) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if(feof(file)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				dpush(F); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				io_error(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			dpush(tag_fixnum(c)); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fread) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE* file = unbox_alien(); | 
					
						
							|  |  |  | 	CELL size = unbox_array_size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(size == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		dpush(tag_object(allot_string(0,0))); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	F_BYTE_ARRAY *buf = allot_byte_array(size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		int c = fread(buf + 1,1,size,file); | 
					
						
							|  |  |  | 		if(c <= 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if(feof(file)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				dpush(F); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				io_error(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			dpush(tag_object(memory_to_char_string( | 
					
						
							|  |  |  | 				(char *)(buf + 1),c))); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fwrite) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE* file = unbox_alien(); | 
					
						
							|  |  |  | 	F_STRING* text = untag_string(dpop()); | 
					
						
							|  |  |  | 	F_FIXNUM length = untag_fixnum_fast(text->length); | 
					
						
							|  |  |  | 	char* string = to_char_string(text,false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(string_capacity(text) == 0) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		size_t written = fwrite(string,1,length,file); | 
					
						
							|  |  |  | 		if(written == length) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if(feof(file)) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				io_error(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Still here? EINTR */ | 
					
						
							|  |  |  | 			length -= written; | 
					
						
							|  |  |  | 			string += written; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fflush) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE *file = unbox_alien(); | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if(fflush(file) == EOF) | 
					
						
							|  |  |  | 			io_error(); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_PRIMITIVE(fclose) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE *file = unbox_alien(); | 
					
						
							|  |  |  | 	for(;;) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if(fclose(file) == EOF) | 
					
						
							|  |  |  | 			io_error(); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 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. */ | 
					
						
							|  |  |  | int err_no(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return errno; | 
					
						
							|  |  |  | } |