VOP_RDWR(9) FreeBSD カーネル開発者マニュアル VOP_RDWR(9)
名称
VOP_READ, VOP_WRITE − ファイルの読み込みまたは書き込み |
書式
#include <sys/param.h> int |
VOP_READ(struct vnode *vp, struct uio *uio, int ioflag, struct ucred *cred); int |
VOP_WRITE(struct vnode *vp, struct uio *uio, int ioflag, struct ucred *cred); |
解説 |
これらのエントリポイントは、ファイルの内容を読み込みまたは書き込みます。 引数は以下の通りです。 |
vp
ファイルの vnode。 uio ioflag cnp ioflag 引数は、ファイルシステムに対する命令およびヒントを与えるために使用 されます。読み込みを試みる時には、上位の 16 ビットは、ファイルシステムが 試みるべき先読みのヒントを (ファイルシステムブロック単位で) 提供するため に使用されます。下位の 16 ビットは、以下のフラグを含むことができるビット マスクです。 IO_UNIT IO_APPEND IO_SYNC IO_NODELOCKED IO_NDELAY IO_VMIO ロック |
入る時に、ファイルはロックされているべきで、終了時までロックされ続けま す。 |
戻り値
成功時には 0 が返され、そうでない場合にはエラーコードが返されます。 |
疑似コード
int vop_read(struct vnode *vp, struct uio *uio, int ioflag, struct ucred *cred) { struct buf *bp; off_t bytesinfile; daddr_t lbn, nextlbn; long size, xfersize, blkoffset; int error; size = block size of file system; for (error = 0, bp = NULL; uio->uio_resid > 0; bp = NULL) { |
bytesinfile = size of file - uio->uio_offset; |
|||
if (bytesinfile <= 0) |
|||
break; |
|||
lbn = uio->uio_offset / size; |
|||
blkoffset = uio->uio_offset - lbn * size; |
|||
xfersize = size - blkoffset; |
|||
if (uio->uio_resid < xfersize) |
|||
xfersize = uio->uio_resid; |
|||
if (bytesinfile < xfersize) |
|||
xfersize = bytesinfile; |
|||
error = bread(vp, lbn, size, NOCRED, &bp); |
|||
if (error) { |
|||
brelse(bp); |
|||
bp = NULL; |
|||
break; |
|||
} |
|||
/* |
|||
* 入出力エラーが起こった時には 0 でない b_resid のみ |
|||
* を得ます。これは上で break を引き起こすべきです。 |
|||
* しかしながら、短い読み込みがエラーを起こさなかった |
|||
* 場合には、間違ったまたは初期化されていないデータの |
|||
* uiomove を行なわないことを保証したい。 |
|||
*/ |
|||
size -= bp->b_resid; |
|||
if (size < xfersize) { |
|||
if (size == 0) |
|||
break; |
|||
xfersize = size; |
|||
} |
|||
error = uiomove((char *)bp->b_data + blkoffset, (int)xfersize, uio); |
|||
if (error) |
|||
break; |
|||
bqrelse(bp); |
} |
bqrelse(bp); |
return (error); int osize = size of file; |
flags = B_SYNC; |
else |
flags = 0; |
for (error = 0; uio->uio_resid > 0;) { |
lbn = uio->uio_offset / size; |
||||||
blkoffset = uio->uio_offset - lbn * size; |
||||||
xfersize = size - blkoffset; |
||||||
if (uio->uio_resid < xfersize) |
||||||
xfersize = uio->uio_resid; |
||||||
if (uio->uio_offset + xfersize > size of file) |
||||||
vnode_pager_setsize(vp, uio->uio_offset + xfersize); |
||||||
if (size > xfersize) |
||||||
flags |= B_CLRBUF; |
||||||
else |
||||||
flags &= ~B_CLRBUF; |
||||||
error = find_block_in_file(vp, lbn, blkoffset + xfersize, |
||||||
cred, &bp, flags); |
||||||
if (error) |
||||||
break; |
||||||
if (uio->uio_offset + xfersize > size of file) |
||||||
set size of file to uio->uio_offset + xfersize; |
||||||
error = uiomove((char *)bp->b_data + blkoffset, (int) xfersize, uio); |
||||||
/* XXX ufs はここでエラーチェックをしていません。何故? */ |
||||||
if (ioflag & IO_VMIO) |
||||||
bp->b_flags |= B_RELBUF; /* ??? */ |
||||||
if (ioflag & IO_SYNC) |
||||||
bwrite(bp); |
||||||
else if (xfersize + blkoffset == size) |
||||||
bawrite(bp); |
||||||
else |
||||||
bdwrite(bp); |
||||||
if (error || xfersize == 0) |
||||||
break; |
} if (error) { |
if (ioflag & IO_UNIT) { |
|
/* ファイルを切り詰める私的なルーチンを呼び出す。 */ |
|
your_truncate(vp, osize, ioflag & IO_SYNC, cred, uio->uio_td); |
|
uio->uio_offset -= resid - uio->uio_resid; |
|
uio->uio_resid = resid; |
|
} |
} else if (resid > uio->uio_resid && (ioflag & IO_SYNC)) { |
struct timeval tv; |
|
error = VOP_UPDATE(vp, &tv, &tv, 1); /* XXX これは何をしているの? */ |
} return (error); |
エラー
[EFBIG]
プロセスのファイルサイズリミットまたは最大ファイルサイ ズを超えるファイルを書込もうとしました。 [ENOSPC] [EPERM] 関連項目 |
作者
このマニュアルページは Doug Rabson が書きました。 FreeBSD 10.0 July 24, 1996 FreeBSD 10.0 |