mirror of https://github.com/postgres/postgres
For the trivial case of iovcnt == 1, kernels are measurably slower at dealing with the more complex arguments of preadv/pwritev than the equivalent plain old pread/pwrite. The overheads are worth it for iovcnt > 1, but for 1 let's just redirect to the cheaper calls. While we could leave it to callers to worry about that, we already have to have our own pg_ wrappers for portability reasons so it seems reasonable to centralize this knowledge there (thanks to Heikki for this suggestion). Try to avoid function call overheads by making them inlinable, which might also allow the compiler to avoid the branch in some cases. For systems that don't have preadv and pwritev (currently: Windows and [closed] Solaris), we might as well pull the replacement functions up into the static inline functions too. Reviewed-by: Heikki Linnakangas <hlinnaka@iki.fi> Discussion: https://postgr.es/m/CA+hUKGJkOiOCa+mag4BF+zHo7qo=o9CFheB8=g6uT5TUm2gkvA@mail.gmail.compull/147/head
parent
a60b8a58f4
commit
15c9ac3629
@ -1,43 +0,0 @@ |
||||
/*-------------------------------------------------------------------------
|
||||
* |
||||
* preadv.c |
||||
* Implementation of preadv(2) for platforms that lack one. |
||||
* |
||||
* Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group |
||||
* |
||||
* IDENTIFICATION |
||||
* src/port/preadv.c |
||||
* |
||||
*------------------------------------------------------------------------- |
||||
*/ |
||||
|
||||
|
||||
#include "c.h" |
||||
|
||||
#include <unistd.h> |
||||
|
||||
#include "port/pg_iovec.h" |
||||
|
||||
ssize_t |
||||
pg_preadv(int fd, const struct iovec *iov, int iovcnt, off_t offset) |
||||
{ |
||||
ssize_t sum = 0; |
||||
ssize_t part; |
||||
|
||||
for (int i = 0; i < iovcnt; ++i) |
||||
{ |
||||
part = pg_pread(fd, iov[i].iov_base, iov[i].iov_len, offset); |
||||
if (part < 0) |
||||
{ |
||||
if (i == 0) |
||||
return -1; |
||||
else |
||||
return sum; |
||||
} |
||||
sum += part; |
||||
offset += part; |
||||
if (part < iov[i].iov_len) |
||||
return sum; |
||||
} |
||||
return sum; |
||||
} |
@ -1,43 +0,0 @@ |
||||
/*-------------------------------------------------------------------------
|
||||
* |
||||
* pwritev.c |
||||
* Implementation of pwritev(2) for platforms that lack one. |
||||
* |
||||
* Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group |
||||
* |
||||
* IDENTIFICATION |
||||
* src/port/pwritev.c |
||||
* |
||||
*------------------------------------------------------------------------- |
||||
*/ |
||||
|
||||
|
||||
#include "c.h" |
||||
|
||||
#include <unistd.h> |
||||
|
||||
#include "port/pg_iovec.h" |
||||
|
||||
ssize_t |
||||
pg_pwritev(int fd, const struct iovec *iov, int iovcnt, off_t offset) |
||||
{ |
||||
ssize_t sum = 0; |
||||
ssize_t part; |
||||
|
||||
for (int i = 0; i < iovcnt; ++i) |
||||
{ |
||||
part = pg_pwrite(fd, iov[i].iov_base, iov[i].iov_len, offset); |
||||
if (part < 0) |
||||
{ |
||||
if (i == 0) |
||||
return -1; |
||||
else |
||||
return sum; |
||||
} |
||||
sum += part; |
||||
offset += part; |
||||
if (part < iov[i].iov_len) |
||||
return sum; |
||||
} |
||||
return sum; |
||||
} |
Loading…
Reference in new issue