summaryrefslogtreecommitdiff
path: root/sys/src/ape/lib/ap/stdio/freopen.c
blob: ba586bbfc5fc6735fa6f2c82f55afd117f1cef4f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
/*
 * pANS stdio -- freopen
 */
#include "iolib.h"
/*
 * Open the named file with the given mode, using the given FILE
 * Legal modes are given below, `additional characters may follow these sequences':
 * r rb		open to read
 * w wb		open to write, truncating
 * a ab		open to write positioned at eof, creating if non-existant
 * r+ r+b rb+	open to read and write, creating if non-existant
 * w+ w+b wb+	open to read and write, truncating
 * a+ a+b ab+	open to read and write, positioned at eof, creating if non-existant.
 */
FILE *freopen(const char *name, const char *mode, FILE *f){
	int m;

	if(f->state!=CLOSED){
		fclose(f);
/* premature; fall through and see what happens */
/*		f->state=OPEN; */
	}

	m = *mode++;
	if(m == 0)
		return NULL;
	if(*mode == 'b')
		mode++;
	switch(m){
	default:
		return NULL;
	case 'r':
		f->fd=open(name, (*mode == '+'? O_RDWR: O_RDONLY));
		break;
	case 'w':
		f->fd=creat(name, 0666);	/* implicitly O_WRONLY */
		/* for O_RDWR, have to creat, close, open */
		if(*mode == '+' && f->fd >= 0) {
			close(f->fd);
			f->fd=open(name, O_RDWR);
		}
		break;
	case 'a':
		f->fd=open(name, (*mode == '+'? O_RDWR: O_WRONLY));
		if(f->fd<0) {
			f->fd=creat(name, 0666);
			/* for O_RDWR, have to creat, close, open */
			if(*mode == '+' && f->fd >= 0) {
				close(f->fd);
				f->fd=open(name, O_RDWR);
			}
		}
		lseek(f->fd, 0L, 2);
		break;
	}

	if(f->fd==-1)
		return NULL;
	f->flags=(mode[0]=='a')? APPEND : 0;
	f->state=OPEN;
	f->buf=0;
	f->rp=0;
	f->wp=0;
	f->lp=0;
	return f;
}