aboutsummaryrefslogtreecommitdiff
path: root/sys/fio/ffault.x
blob: 76f43f4a313708eae8e21509d9c9aa95c87c4547 (plain) (blame)
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
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
# Copyright(c) 1986 Association of Universities for Research in Astronomy Inc.

include	<syserr.h>
include	<error.h>
include	<config.h>
include	<fio.h>

# FFAULT -- Read a file block into the file buffer (pick up file buffer and
# put it down on another part of the file).  Although in this implementation
# there is only a single, local, file buffer for each file, in the future
# a variable number of either global or local buffers will be supported, as
# well as read ahead and write behind (see Fio.doc).

int procedure ffault (fd, file_offset, nreserve, rwflag)

int	fd
long	file_offset		# char offset to be faulted in
int	nreserve		# size of transfer pending
int	rwflag			# next access is a read or a write

pointer	bp
long	buffer_offset, fboff
int	bufsize, nchars_read
bool	block_write, stream, at_eof

int	and()
errchk	ffilbf, fflsbf, fwatio
define	ioerror_ 91
include	<fio.com>

begin
	# assert (file open, buffer already created)
	# assert (iop does not point into buffer)

	fp = fiodes[fd]
	bp = bufptr[fd]
	bufsize = FBUFSIZE(fp)
	fboff   = FIRSTBUFOFF(fp)
	stream  = (FBLKSIZE(fp) == 0)

	# Calculate buffer_offset (modulus file buffer size).  If the output
	# device is a pipe or terminal (stream device), which does not permit
	# rewriting of data and seeking, empty buffer.

	if (stream) {
	    buffer_offset = file_offset	
	} else if (file_offset <= 0) {
	    iferr (call filerr (FNAME(fp), SYS_FSEEK))
		goto ioerror_
	} else
	    buffer_offset = (file_offset-fboff) / bufsize * bufsize + fboff

	# Update i/o pointers (if an empty or partially full buffer has been
	# written into, determine the top of the valid part of the buffer).

	UPDATE_IOP(fd)
	    
	# Flush buffer if it has been written into.  Write out only as much
	# of the buffer as has been filled.

	if (BUF_MODIFIED(fd)) {
	    iferr (call fflsbf (fd, bp, otop[fd]-bp, boffset[fd]))
		goto ioerror_

	    # We need to do this wait here since we immediately use this buffer
	    # without doing a wait. Which screws up the data going out to disk.
	    # This can be done away with when multi-buffering is done. (FJR).

	    call fwatio (fd)
	}

	# Fill buffer from file only if the file was opened with read
	# permission, and if the fault was not caused by a WRITE which will
	# immediately overwrite the entire contents of the buffer.

	if (rwflag == FF_WRITE) {
	    block_write = (stream ||
		(file_offset == buffer_offset && nreserve >= bufsize))
	} else
	    block_write = false

	if (block_write) {
	    itop[fd] = bp
	    otop[fd] = bp

	} else if (and(FF_READ,fflags[fd]) == 0) {
	    # Read is disabled.  Zero-fill buffer; if inside existing
	    # random access file, set ITOP to end of buffer so that the
	    # entire buffer will be written when flushed.

	    at_eof = (FILSIZE(fp) >= 0 && buffer_offset > FILSIZE(fp))
	    otop[fd] = bp

	    if (at_eof)
		itop[fd] = bp
	    else
		itop[fd] = bp + bufsize

	    # Zero-fill the buffer.
	    call aclrc (Memc[bp], bufsize)

	} else {
	    iferr {
		# Initialize buffer from file.
		call ffilbf (fd, bp, bufsize, buffer_offset)
		call fwatio (fd)
	    } then
		goto ioerror_
	}

	boffset[fd] = buffer_offset
	LSEEK (fd, file_offset)				# set i/o pointer

	nchars_read = itop[fd] - iop[fd]
	if (nchars_read <= 0)
	    return (EOF)
	else
	    return (nchars_read)			# only valid for a read

	# If an i/o error occurs, mark the buffer empty and pass the error
	# back to our caller.

ioerror_
	itop[fd] = bp
	otop[fd] = bp
	call erract (EA_ERROR)
end