206 lines
4.3 KiB
C
206 lines
4.3 KiB
C
/**
|
|
* See Copyright Notice in picrin.h
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "picrin.h"
|
|
#include "picrin/extra.h"
|
|
#include "../value.h"
|
|
#include "../object.h"
|
|
|
|
#if PIC_USE_FILE
|
|
|
|
static int
|
|
file_read(pic_state *PIC_UNUSED(pic), void *cookie, char *ptr, int size) {
|
|
FILE *file = cookie;
|
|
int r;
|
|
|
|
size = 1; /* override size */
|
|
|
|
r = (int)fread(ptr, 1, (size_t)size, file);
|
|
if (r < size && ferror(file)) {
|
|
return -1;
|
|
}
|
|
if (r == 0 && feof(file)) {
|
|
clearerr(file);
|
|
}
|
|
return r;
|
|
}
|
|
|
|
static int
|
|
file_write(pic_state *PIC_UNUSED(pic), void *cookie, const char *ptr, int size) {
|
|
FILE *file = cookie;
|
|
int r;
|
|
|
|
r = (int)fwrite(ptr, 1, (size_t)size, file);
|
|
if (r < size) {
|
|
return -1;
|
|
}
|
|
fflush(cookie);
|
|
return r;
|
|
}
|
|
|
|
static long
|
|
file_seek(pic_state *PIC_UNUSED(pic), void *cookie, long pos, int whence) {
|
|
switch (whence) {
|
|
case PIC_SEEK_CUR:
|
|
whence = SEEK_CUR;
|
|
break;
|
|
case PIC_SEEK_SET:
|
|
whence = SEEK_SET;
|
|
break;
|
|
case PIC_SEEK_END:
|
|
whence = SEEK_END;
|
|
break;
|
|
}
|
|
if (fseek(cookie, pos, whence) == 0) {
|
|
return ftell(cookie);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int
|
|
file_close(pic_state *PIC_UNUSED(pic), void *cookie) {
|
|
return fclose(cookie);
|
|
}
|
|
|
|
pic_value
|
|
pic_fopen(pic_state *pic, FILE *fp, const char *mode) {
|
|
static const pic_port_type file_rd = { file_read, 0, file_seek, file_close };
|
|
static const pic_port_type file_wr = { 0, file_write, file_seek, file_close };
|
|
|
|
if (*mode == 'r') {
|
|
return pic_funopen(pic, fp, &file_rd);
|
|
} else {
|
|
return pic_funopen(pic, fp, &file_wr);
|
|
}
|
|
}
|
|
|
|
#if !PIC_USE_ERROR
|
|
# define file_error pic_error
|
|
#else
|
|
PIC_NORETURN static void
|
|
file_error(pic_state *pic, const char *msg, int n, ...)
|
|
{
|
|
va_list ap;
|
|
pic_value e, irrs;
|
|
|
|
va_start(ap, n);
|
|
irrs = pic_vlist(pic, n, ap);
|
|
va_end(ap);
|
|
|
|
e = pic_funcall(pic, "make-error-object", 3, pic_intern_lit(pic, "file"), pic_cstr_value(pic, msg), irrs);
|
|
pic_funcall(pic, "raise", 1, e);
|
|
PIC_UNREACHABLE();
|
|
}
|
|
#endif
|
|
|
|
pic_value
|
|
pic_file_open_input_file(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
FILE *fp;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
if ((fp = fopen(fname, "r")) == NULL) {
|
|
file_error(pic, "could not open file", 1, pic_cstr_value(pic, fname));
|
|
}
|
|
return pic_fopen(pic, fp, "r");
|
|
}
|
|
|
|
pic_value
|
|
pic_file_open_output_file(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
FILE *fp;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
if ((fp = fopen(fname, "w")) == NULL) {
|
|
file_error(pic, "could not open file", 1, pic_cstr_value(pic, fname));
|
|
}
|
|
return pic_fopen(pic, fp, "w");
|
|
}
|
|
|
|
pic_value
|
|
pic_file_open_binary_input_file(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
FILE *fp;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
if ((fp = fopen(fname, "rb")) == NULL) {
|
|
file_error(pic, "could not open file", 1, pic_cstr_value(pic, fname));
|
|
}
|
|
return pic_fopen(pic, fp, "rb");
|
|
}
|
|
|
|
pic_value
|
|
pic_file_open_binary_output_file(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
FILE *fp;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
if ((fp = fopen(fname, "wb")) == NULL) {
|
|
file_error(pic, "could not open file", 1, pic_cstr_value(pic, fname));
|
|
}
|
|
return pic_fopen(pic, fp, "wb");
|
|
}
|
|
|
|
pic_value
|
|
pic_file_exists_p(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
FILE *fp;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
fp = fopen(fname, "r");
|
|
if (fp) {
|
|
fclose(fp);
|
|
}
|
|
return pic_bool_value(pic, fp != NULL);
|
|
}
|
|
|
|
pic_value
|
|
pic_file_delete(pic_state *pic)
|
|
{
|
|
const char *fname;
|
|
|
|
pic_get_args(pic, "z", &fname);
|
|
|
|
if (remove(fname) != 0) {
|
|
file_error(pic, "file cannot be deleted", 1, pic_cstr_value(pic, fname));
|
|
}
|
|
return pic_undef_value(pic);
|
|
}
|
|
|
|
void
|
|
pic_init_file(pic_state *pic)
|
|
{
|
|
pic_value i, o, e;
|
|
|
|
i = pic_fopen(pic, stdin, "r");
|
|
o = pic_fopen(pic, stdout, "w");
|
|
e = pic_fopen(pic, stderr, "w");
|
|
pic_setvbuf(pic, i, NULL, PIC_IOLBF, 0);
|
|
pic_setvbuf(pic, o, NULL, PIC_IOLBF, 0);
|
|
pic_defvar(pic, "current-input-port", i);
|
|
pic_defvar(pic, "current-output-port", o);
|
|
pic_defvar(pic, "current-error-port", e);
|
|
|
|
pic_defun(pic, "open-input-file", pic_file_open_input_file);
|
|
pic_defun(pic, "open-output-file", pic_file_open_output_file);
|
|
pic_defun(pic, "open-binary-input-file", pic_file_open_binary_input_file);
|
|
pic_defun(pic, "open-binary-output-file", pic_file_open_binary_output_file);
|
|
pic_defun(pic, "file-exists?", pic_file_exists_p);
|
|
pic_defun(pic, "delete-file", pic_file_delete);
|
|
}
|
|
|
|
#endif
|