/* Copyright (C) 1994, 1995, 1996, 1997 Free Software Foundation This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "priv.h" #include "io_S.h" #include <fcntl.h> /* Implement io_read as described in <hurd/io.defs>. */ kern_return_t diskfs_S_io_read (struct protid *cred, char **data, mach_msg_type_number_t *datalen, off_t offset, mach_msg_type_number_t maxread) { struct node *np; int err; int off = offset; char *buf; int ourbuf = 0; if (!cred) return EOPNOTSUPP; np = cred->po->np; if (!(cred->po->openstat & O_READ)) return EBADF; mutex_lock (&np->lock); iohelp_get_conch (&np->conch); if (off == -1) off = cred->po->filepointer; if (off < 0) { mutex_unlock (&np->lock); return EINVAL; } if (off > np->dn_stat.st_size) maxread = 0; else if (off + (off_t) maxread > np->dn_stat.st_size) maxread = np->dn_stat.st_size - off; if (maxread > *datalen) { ourbuf = 1; vm_allocate (mach_task_self (), (u_int *) &buf, maxread, 1); *data = buf; } else buf = *data; *datalen = maxread; if (maxread == 0) err = 0; else if (S_ISLNK (np->dn_stat.st_mode)) /* Read from a symlink. */ if (! diskfs_read_symlink_hook) err = EINVAL; else { if (off == 0 && maxread == np->dn_stat.st_size) err = (*diskfs_read_symlink_hook)(np, buf); else { char *whole_link = alloca (np->dn_stat.st_size); err = (*diskfs_read_symlink_hook)(np, whole_link); if (! err) memcpy (buf, whole_link + off, maxread); } } else err = EINVAL; /* Use read below. */ if (err == EINVAL) err = _diskfs_rdwr_internal (np, buf, off, datalen, 0, cred->po->openstat & O_NOATIME); if (diskfs_synchronous) diskfs_node_update (np, 1); /* atime! */ if (offset == -1 && !err) cred->po->filepointer += *datalen; if (err && ourbuf) vm_deallocate (mach_task_self (), (u_int) buf, maxread); mutex_unlock (&np->lock); return err; }