2012-07-12 16:42:59 +02:00
|
|
|
/*
|
|
|
|
* \brief 'pread()' and 'pwrite()' implementations
|
|
|
|
* \author Christian Prochaska
|
|
|
|
* \date 2012-07-11
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2013-01-10 21:44:47 +01:00
|
|
|
* Copyright (C) 2012-2013 Genode Labs GmbH
|
2012-07-12 16:42:59 +02:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
|
|
* under the terms of the GNU General Public License version 2.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Genode includes */
|
|
|
|
#include <base/lock.h>
|
2013-11-13 13:56:58 +01:00
|
|
|
#include <libc-plugin/fd_alloc.h>
|
2012-07-12 16:42:59 +02:00
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/uio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
|
|
|
|
struct Read
|
|
|
|
{
|
2013-11-13 13:56:58 +01:00
|
|
|
ssize_t operator()(int fd, void *buf, size_t count)
|
|
|
|
{
|
|
|
|
return read(fd, buf, count);
|
|
|
|
}
|
2012-07-12 16:42:59 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct Write
|
|
|
|
{
|
2013-11-13 13:56:58 +01:00
|
|
|
ssize_t operator()(int fd, const void *buf, size_t count)
|
|
|
|
{
|
|
|
|
return write(fd, buf, count);
|
|
|
|
}
|
2012-07-12 16:42:59 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <typename Rw_func, typename Buf_type>
|
|
|
|
static ssize_t pread_pwrite_impl(Rw_func rw_func, int fd, Buf_type buf, ::size_t count, ::off_t offset)
|
|
|
|
{
|
2013-11-13 13:56:58 +01:00
|
|
|
Libc::File_descriptor *fdesc = Libc::file_descriptor_allocator()->find_by_libc_fd(fd);
|
|
|
|
if (fdesc == 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
Genode::Lock_guard<Genode::Lock> rw_lock_guard(fdesc->lock);
|
2012-07-12 16:42:59 +02:00
|
|
|
|
|
|
|
off_t old_offset = lseek(fd, 0, SEEK_CUR);
|
|
|
|
|
|
|
|
if (old_offset == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (lseek(fd, offset, SEEK_SET) == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ssize_t result = rw_func(fd, buf, count);
|
|
|
|
|
|
|
|
if (lseek(fd, old_offset, SEEK_SET) == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
extern "C" ssize_t pread(int fd, void *buf, ::size_t count, ::off_t offset)
|
|
|
|
{
|
|
|
|
return pread_pwrite_impl(Read(), fd, buf, count, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
extern "C" ssize_t pwrite(int fd, const void *buf, ::size_t count, ::off_t offset)
|
|
|
|
{
|
|
|
|
return pread_pwrite_impl(Write(), fd, buf, count, offset);
|
|
|
|
}
|