202 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
			
		
		
	
	
			202 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
| #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[STDIN_ENV] = allot_alien(F,(CELL)stdin);
 | |
| 	userenv[STDOUT_ENV] = allot_alien(F,(CELL)stdout);
 | |
| 	userenv[STDERR_ENV] = allot_alien(F,(CELL)stderr);
 | |
| }
 | |
| 
 | |
| void io_error(void)
 | |
| {
 | |
| #ifndef WINCE
 | |
| 	if(errno == EINTR)
 | |
| 		return;
 | |
| #endif
 | |
| 
 | |
| 	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
 | |
| 		{
 | |
| 			if(c != size)
 | |
| 			{
 | |
| 				REGISTER_UNTAGGED(buf);
 | |
| 				F_BYTE_ARRAY *new_buf = allot_byte_array(c);
 | |
| 				UNREGISTER_UNTAGGED(buf);
 | |
| 				memcpy(new_buf + 1, buf + 1,c);
 | |
| 				buf = new_buf;
 | |
| 			}
 | |
| 			dpush(tag_object(buf));
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| DEFINE_PRIMITIVE(fputc)
 | |
| {
 | |
| 	FILE *file = unbox_alien();
 | |
| 	F_FIXNUM ch = to_fixnum(dpop());
 | |
| 
 | |
| 	for(;;)
 | |
| 	{
 | |
| 		if(fputc(ch,file) == EOF)
 | |
| 		{
 | |
| 			io_error();
 | |
| 
 | |
| 			/* Still here? EINTR */
 | |
| 		}
 | |
| 		else
 | |
| 			break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| DEFINE_PRIMITIVE(fwrite)
 | |
| {
 | |
| 	FILE *file = unbox_alien();
 | |
| 	F_BYTE_ARRAY *text = untag_byte_array(dpop());
 | |
| 	F_FIXNUM length = array_capacity(text);
 | |
| 	char *string = (char *)(text + 1);
 | |
| 
 | |
| 	if(length == 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;
 | |
| }
 | |
| 
 | |
| void clear_err_no(void)
 | |
| {
 | |
| 	errno = 0;
 | |
| }
 |