2010-04-27 18:04:05 +05:30
|
|
|
/*
|
|
|
|
* Helpers for getting linearized buffers from iov / filling buffers into iovs
|
|
|
|
*
|
|
|
|
* Copyright IBM, Corp. 2007, 2008
|
|
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* Author(s):
|
|
|
|
* Anthony Liguori <aliguori@us.ibm.com>
|
|
|
|
* Amit Shah <amit.shah@redhat.com>
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
* Michael Tokarev <mjt@tls.msk.ru>
|
2010-04-27 18:04:05 +05:30
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2. See
|
|
|
|
* the COPYING file in the top-level directory.
|
2012-01-13 17:44:23 +01:00
|
|
|
*
|
|
|
|
* Contributions after 2012-01-13 are licensed under the terms of the
|
|
|
|
* GNU GPL, version 2 or (at your option) any later version.
|
2010-04-27 18:04:05 +05:30
|
|
|
*/
|
|
|
|
|
2016-01-29 17:49:55 +00:00
|
|
|
#include "qemu/osdep.h"
|
2012-12-17 18:20:00 +01:00
|
|
|
#include "qemu/iov.h"
|
2014-02-23 18:02:07 +01:00
|
|
|
#include "qemu/sockets.h"
|
2016-03-20 19:16:19 +02:00
|
|
|
#include "qemu/cutils.h"
|
2012-03-14 11:18:54 +04:00
|
|
|
|
2015-12-22 12:03:33 +01:00
|
|
|
size_t iov_from_buf_full(const struct iovec *iov, unsigned int iov_cnt,
|
|
|
|
size_t offset, const void *buf, size_t bytes)
|
2010-04-27 18:04:05 +05:30
|
|
|
{
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
size_t done;
|
2010-04-27 18:04:05 +05:30
|
|
|
unsigned int i;
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
for (i = 0, done = 0; (offset || done < bytes) && i < iov_cnt; i++) {
|
|
|
|
if (offset < iov[i].iov_len) {
|
|
|
|
size_t len = MIN(iov[i].iov_len - offset, bytes - done);
|
|
|
|
memcpy(iov[i].iov_base + offset, buf + done, len);
|
|
|
|
done += len;
|
|
|
|
offset = 0;
|
|
|
|
} else {
|
|
|
|
offset -= iov[i].iov_len;
|
2011-07-11 15:02:23 +02:00
|
|
|
}
|
2010-04-27 18:04:05 +05:30
|
|
|
}
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
assert(offset == 0);
|
|
|
|
return done;
|
2010-04-27 18:04:05 +05:30
|
|
|
}
|
2010-04-27 18:04:06 +05:30
|
|
|
|
2015-12-22 12:03:33 +01:00
|
|
|
size_t iov_to_buf_full(const struct iovec *iov, const unsigned int iov_cnt,
|
|
|
|
size_t offset, void *buf, size_t bytes)
|
2010-04-27 18:04:06 +05:30
|
|
|
{
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
size_t done;
|
2010-04-27 18:04:06 +05:30
|
|
|
unsigned int i;
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
for (i = 0, done = 0; (offset || done < bytes) && i < iov_cnt; i++) {
|
|
|
|
if (offset < iov[i].iov_len) {
|
|
|
|
size_t len = MIN(iov[i].iov_len - offset, bytes - done);
|
|
|
|
memcpy(buf + done, iov[i].iov_base + offset, len);
|
|
|
|
done += len;
|
|
|
|
offset = 0;
|
|
|
|
} else {
|
|
|
|
offset -= iov[i].iov_len;
|
2010-04-27 18:04:06 +05:30
|
|
|
}
|
2011-07-13 15:16:08 +02:00
|
|
|
}
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
assert(offset == 0);
|
|
|
|
return done;
|
2011-07-13 15:16:08 +02:00
|
|
|
}
|
|
|
|
|
change iov_* function prototypes to be more appropriate
Reorder arguments to be more natural, readable and
consistent with other iov_* functions, and change
argument names, from:
iov_from_buf(iov, iov_cnt, buf, iov_off, size)
to
iov_from_buf(iov, iov_cnt, offset, buf, bytes)
The result becomes natural English:
copy data to this `iov' vector with `iov_cnt'
elements starting at byte offset `offset'
from memory buffer `buf', processing `bytes'
bytes max.
(Try to read the original prototype this way).
Also change iov_clear() to more general iov_memset()
(it uses memset() internally anyway).
While at it, add comments to the header file
describing what the routines actually does.
The patch only renames argumens in the header, but
keeps old names in the implementation. The next
patch will touch actual code to match.
Now, it might look wrong to pay so much attention
to so small things. But we've so many badly designed
interfaces already so the whole thing becomes rather
confusing or error prone. One example of this is
previous commit and small discussion which emerged
from it, with an outcome that the utility functions
like these aren't well-understdandable, leading to
strange usage cases. That's why I paid quite some
attention to this set of functions and a few
others in subsequent patches.
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-03-11 18:05:12 +04:00
|
|
|
size_t iov_memset(const struct iovec *iov, const unsigned int iov_cnt,
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
size_t offset, int fillc, size_t bytes)
|
2011-07-13 15:16:08 +02:00
|
|
|
{
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
size_t done;
|
2011-07-13 15:16:08 +02:00
|
|
|
unsigned int i;
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
for (i = 0, done = 0; (offset || done < bytes) && i < iov_cnt; i++) {
|
|
|
|
if (offset < iov[i].iov_len) {
|
|
|
|
size_t len = MIN(iov[i].iov_len - offset, bytes - done);
|
|
|
|
memset(iov[i].iov_base + offset, fillc, len);
|
|
|
|
done += len;
|
|
|
|
offset = 0;
|
|
|
|
} else {
|
|
|
|
offset -= iov[i].iov_len;
|
2011-07-13 15:16:08 +02:00
|
|
|
}
|
2010-04-27 18:04:06 +05:30
|
|
|
}
|
rewrite iov_* functions
This changes implementations of all iov_*
functions, completing the previous step.
All iov_* functions now ensure that this offset
argument is within the iovec (using assertion),
but lets to specify `bytes' value larger than
actual length of the iovec - in this case they
stops at the actual end of iovec. It is also
suggested to use convinient `-1' value as `bytes'
to mean just this -- "up to the end".
There's one very minor semantic change here: new
requiriment is that `offset' points to inside of
iovec. This is checked just at the end of functions
(assert()), it does not actually need to be enforced,
but using any of these functions with offset pointing
past the end of iovec is wrong anyway.
Note: the new code in iov.c uses arithmetic with
void pointers. I thought this is not supported
everywhere and is a GCC extension (indeed, the C
standard does not define void arithmetic). However,
the original code already use void arith in
iov_from_buf() function:
(memcpy(..., buf + buf_off,...)
which apparently works well so far (it is this
way in qemu 1.0). So I left it this way and used
it in other places.
While at it, add a unit-test file test-iov.c,
to check various corner cases with iov_from_buf(),
iov_to_buf() and iov_memset().
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
2012-06-07 20:08:19 +04:00
|
|
|
assert(offset == 0);
|
|
|
|
return done;
|
2010-04-27 18:04:06 +05:30
|
|
|
}
|
|
|
|
|
2011-07-11 15:02:23 +02:00
|
|
|
size_t iov_size(const struct iovec *iov, const unsigned int iov_cnt)
|
2010-04-27 18:04:06 +05:30
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
len = 0;
|
2011-07-11 15:02:23 +02:00
|
|
|
for (i = 0; i < iov_cnt; i++) {
|
2010-04-27 18:04:06 +05:30
|
|
|
len += iov[i].iov_len;
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
2011-07-12 13:35:10 +02:00
|
|
|
|
2012-03-14 11:18:54 +04:00
|
|
|
/* helper function for iov_send_recv() */
|
|
|
|
static ssize_t
|
|
|
|
do_send_recv(int sockfd, struct iovec *iov, unsigned iov_cnt, bool do_send)
|
|
|
|
{
|
2013-04-21 12:01:06 +02:00
|
|
|
#ifdef CONFIG_POSIX
|
2012-03-14 11:18:54 +04:00
|
|
|
ssize_t ret;
|
|
|
|
struct msghdr msg;
|
|
|
|
memset(&msg, 0, sizeof(msg));
|
|
|
|
msg.msg_iov = iov;
|
|
|
|
msg.msg_iovlen = iov_cnt;
|
|
|
|
do {
|
|
|
|
ret = do_send
|
|
|
|
? sendmsg(sockfd, &msg, 0)
|
|
|
|
: recvmsg(sockfd, &msg, 0);
|
|
|
|
} while (ret < 0 && errno == EINTR);
|
|
|
|
return ret;
|
|
|
|
#else
|
|
|
|
/* else send piece-by-piece */
|
|
|
|
/*XXX Note: windows has WSASend() and WSARecv() */
|
2012-07-11 07:09:05 +02:00
|
|
|
unsigned i = 0;
|
|
|
|
ssize_t ret = 0;
|
2022-10-06 15:36:53 +04:00
|
|
|
ssize_t off = 0;
|
2012-07-11 07:09:05 +02:00
|
|
|
while (i < iov_cnt) {
|
2012-03-14 11:18:54 +04:00
|
|
|
ssize_t r = do_send
|
2022-10-06 15:36:53 +04:00
|
|
|
? send(sockfd, iov[i].iov_base + off, iov[i].iov_len - off, 0)
|
|
|
|
: recv(sockfd, iov[i].iov_base + off, iov[i].iov_len - off, 0);
|
2012-03-14 11:18:54 +04:00
|
|
|
if (r > 0) {
|
|
|
|
ret += r;
|
2022-10-06 15:36:53 +04:00
|
|
|
off += r;
|
|
|
|
if (off < iov[i].iov_len) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-03-14 11:18:54 +04:00
|
|
|
} else if (!r) {
|
|
|
|
break;
|
|
|
|
} else if (errno == EINTR) {
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
/* else it is some "other" error,
|
|
|
|
* only return if there was no data processed. */
|
|
|
|
if (ret == 0) {
|
2012-07-11 07:09:05 +02:00
|
|
|
ret = -1;
|
2012-03-14 11:18:54 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2022-10-06 15:36:53 +04:00
|
|
|
off = 0;
|
2012-07-11 07:09:05 +02:00
|
|
|
i++;
|
2012-03-14 11:18:54 +04:00
|
|
|
}
|
2012-07-11 07:09:05 +02:00
|
|
|
return ret;
|
2012-03-14 11:18:54 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-05-21 09:50:10 +08:00
|
|
|
ssize_t iov_send_recv(int sockfd, const struct iovec *_iov, unsigned iov_cnt,
|
2012-03-14 11:18:54 +04:00
|
|
|
size_t offset, size_t bytes,
|
|
|
|
bool do_send)
|
|
|
|
{
|
2013-03-27 17:36:31 +01:00
|
|
|
ssize_t total = 0;
|
2012-03-14 11:18:54 +04:00
|
|
|
ssize_t ret;
|
2013-03-27 17:36:29 +01:00
|
|
|
size_t orig_len, tail;
|
2013-03-27 17:36:30 +01:00
|
|
|
unsigned niov;
|
2015-05-21 09:50:10 +08:00
|
|
|
struct iovec *local_iov, *iov;
|
|
|
|
|
|
|
|
if (bytes <= 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
local_iov = g_new0(struct iovec, iov_cnt);
|
|
|
|
iov_copy(local_iov, iov_cnt, _iov, iov_cnt, offset, bytes);
|
|
|
|
offset = 0;
|
|
|
|
iov = local_iov;
|
2013-03-27 17:36:29 +01:00
|
|
|
|
2013-03-27 17:36:31 +01:00
|
|
|
while (bytes > 0) {
|
|
|
|
/* Find the start position, skipping `offset' bytes:
|
|
|
|
* first, skip all full-sized vector elements, */
|
|
|
|
for (niov = 0; niov < iov_cnt && offset >= iov[niov].iov_len; ++niov) {
|
|
|
|
offset -= iov[niov].iov_len;
|
|
|
|
}
|
2013-03-27 17:36:28 +01:00
|
|
|
|
2013-03-27 17:36:31 +01:00
|
|
|
/* niov == iov_cnt would only be valid if bytes == 0, which
|
|
|
|
* we already ruled out in the loop condition. */
|
2013-03-27 17:36:30 +01:00
|
|
|
assert(niov < iov_cnt);
|
2013-03-27 17:36:31 +01:00
|
|
|
iov += niov;
|
|
|
|
iov_cnt -= niov;
|
|
|
|
|
|
|
|
if (offset) {
|
|
|
|
/* second, skip `offset' bytes from the (now) first element,
|
|
|
|
* undo it on exit */
|
|
|
|
iov[0].iov_base += offset;
|
|
|
|
iov[0].iov_len -= offset;
|
|
|
|
}
|
|
|
|
/* Find the end position skipping `bytes' bytes: */
|
|
|
|
/* first, skip all full-sized elements */
|
|
|
|
tail = bytes;
|
|
|
|
for (niov = 0; niov < iov_cnt && iov[niov].iov_len <= tail; ++niov) {
|
|
|
|
tail -= iov[niov].iov_len;
|
|
|
|
}
|
|
|
|
if (tail) {
|
|
|
|
/* second, fixup the last element, and remember the original
|
|
|
|
* length */
|
|
|
|
assert(niov < iov_cnt);
|
|
|
|
assert(iov[niov].iov_len > tail);
|
|
|
|
orig_len = iov[niov].iov_len;
|
|
|
|
iov[niov++].iov_len = tail;
|
2013-09-14 13:11:36 +04:00
|
|
|
ret = do_send_recv(sockfd, iov, niov, do_send);
|
|
|
|
/* Undo the changes above before checking for errors */
|
2013-03-27 17:36:31 +01:00
|
|
|
iov[niov-1].iov_len = orig_len;
|
2013-09-14 13:11:36 +04:00
|
|
|
} else {
|
|
|
|
ret = do_send_recv(sockfd, iov, niov, do_send);
|
2013-03-27 17:36:31 +01:00
|
|
|
}
|
|
|
|
if (offset) {
|
|
|
|
iov[0].iov_base -= offset;
|
|
|
|
iov[0].iov_len += offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
assert(errno != EINTR);
|
2015-05-21 09:50:10 +08:00
|
|
|
g_free(local_iov);
|
2013-03-27 17:36:31 +01:00
|
|
|
if (errno == EAGAIN && total > 0) {
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-07-23 17:30:12 +09:00
|
|
|
if (ret == 0 && !do_send) {
|
|
|
|
/* recv returns 0 when the peer has performed an orderly
|
|
|
|
* shutdown. */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-03-27 17:36:31 +01:00
|
|
|
/* Prepare for the next iteration */
|
|
|
|
offset += ret;
|
|
|
|
total += ret;
|
|
|
|
bytes -= ret;
|
2012-03-14 11:18:54 +04:00
|
|
|
}
|
|
|
|
|
2015-05-21 09:50:10 +08:00
|
|
|
g_free(local_iov);
|
2013-03-27 17:36:31 +01:00
|
|
|
return total;
|
2012-03-14 11:18:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-12 13:35:10 +02:00
|
|
|
void iov_hexdump(const struct iovec *iov, const unsigned int iov_cnt,
|
|
|
|
FILE *fp, const char *prefix, size_t limit)
|
|
|
|
{
|
2013-03-15 16:41:58 +00:00
|
|
|
int v;
|
|
|
|
size_t size = 0;
|
|
|
|
char *buf;
|
|
|
|
|
|
|
|
for (v = 0; v < iov_cnt; v++) {
|
|
|
|
size += iov[v].iov_len;
|
2011-07-12 13:35:10 +02:00
|
|
|
}
|
2013-03-15 16:41:58 +00:00
|
|
|
size = size > limit ? limit : size;
|
|
|
|
buf = g_malloc(size);
|
|
|
|
iov_to_buf(iov, iov_cnt, 0, buf, size);
|
2020-08-22 20:09:50 +02:00
|
|
|
qemu_hexdump(fp, prefix, buf, size);
|
2013-03-15 16:41:58 +00:00
|
|
|
g_free(buf);
|
2011-07-12 13:35:10 +02:00
|
|
|
}
|
2012-10-24 14:58:39 +02:00
|
|
|
|
2012-09-24 13:02:52 +02:00
|
|
|
unsigned iov_copy(struct iovec *dst_iov, unsigned int dst_iov_cnt,
|
|
|
|
const struct iovec *iov, unsigned int iov_cnt,
|
|
|
|
size_t offset, size_t bytes)
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
unsigned int i, j;
|
2016-08-02 12:41:20 +03:00
|
|
|
for (i = 0, j = 0;
|
|
|
|
i < iov_cnt && j < dst_iov_cnt && (offset || bytes); i++) {
|
2012-09-24 13:02:52 +02:00
|
|
|
if (offset >= iov[i].iov_len) {
|
|
|
|
offset -= iov[i].iov_len;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
len = MIN(bytes, iov[i].iov_len - offset);
|
|
|
|
|
|
|
|
dst_iov[j].iov_base = iov[i].iov_base + offset;
|
|
|
|
dst_iov[j].iov_len = len;
|
|
|
|
j++;
|
|
|
|
bytes -= len;
|
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
assert(offset == 0);
|
|
|
|
return j;
|
|
|
|
}
|
2012-10-31 10:42:51 +01:00
|
|
|
|
2012-10-24 14:58:39 +02:00
|
|
|
/* io vectors */
|
|
|
|
|
|
|
|
void qemu_iovec_init(QEMUIOVector *qiov, int alloc_hint)
|
|
|
|
{
|
2014-12-04 15:00:03 +01:00
|
|
|
qiov->iov = g_new(struct iovec, alloc_hint);
|
2012-10-24 14:58:39 +02:00
|
|
|
qiov->niov = 0;
|
|
|
|
qiov->nalloc = alloc_hint;
|
|
|
|
qiov->size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_iovec_init_external(QEMUIOVector *qiov, struct iovec *iov, int niov)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
qiov->iov = iov;
|
|
|
|
qiov->niov = niov;
|
|
|
|
qiov->nalloc = -1;
|
|
|
|
qiov->size = 0;
|
|
|
|
for (i = 0; i < niov; i++)
|
|
|
|
qiov->size += iov[i].iov_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_iovec_add(QEMUIOVector *qiov, void *base, size_t len)
|
|
|
|
{
|
|
|
|
assert(qiov->nalloc != -1);
|
|
|
|
|
|
|
|
if (qiov->niov == qiov->nalloc) {
|
|
|
|
qiov->nalloc = 2 * qiov->nalloc + 1;
|
2014-12-04 15:00:03 +01:00
|
|
|
qiov->iov = g_renew(struct iovec, qiov->iov, qiov->nalloc);
|
2012-10-24 14:58:39 +02:00
|
|
|
}
|
|
|
|
qiov->iov[qiov->niov].iov_base = base;
|
|
|
|
qiov->iov[qiov->niov].iov_len = len;
|
|
|
|
qiov->size += len;
|
|
|
|
++qiov->niov;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2012-11-22 16:06:06 +01:00
|
|
|
* Concatenates (partial) iovecs from src_iov to the end of dst.
|
2012-10-24 14:58:39 +02:00
|
|
|
* It starts copying after skipping `soffset' bytes at the
|
|
|
|
* beginning of src and adds individual vectors from src to
|
|
|
|
* dst copies up to `sbytes' bytes total, or up to the end
|
2012-11-22 16:06:06 +01:00
|
|
|
* of src_iov if it comes first. This way, it is okay to specify
|
2012-10-24 14:58:39 +02:00
|
|
|
* very large value for `sbytes' to indicate "up to the end
|
|
|
|
* of src".
|
|
|
|
* Only vector pointers are processed, not the actual data buffers.
|
|
|
|
*/
|
2014-06-10 16:21:28 +02:00
|
|
|
size_t qemu_iovec_concat_iov(QEMUIOVector *dst,
|
|
|
|
struct iovec *src_iov, unsigned int src_cnt,
|
|
|
|
size_t soffset, size_t sbytes)
|
2012-10-24 14:58:39 +02:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
size_t done;
|
2013-02-05 11:27:45 +05:30
|
|
|
|
|
|
|
if (!sbytes) {
|
2014-06-10 16:21:28 +02:00
|
|
|
return 0;
|
2013-02-05 11:27:45 +05:30
|
|
|
}
|
2012-10-24 14:58:39 +02:00
|
|
|
assert(dst->nalloc != -1);
|
2012-11-22 16:06:06 +01:00
|
|
|
for (i = 0, done = 0; done < sbytes && i < src_cnt; i++) {
|
|
|
|
if (soffset < src_iov[i].iov_len) {
|
|
|
|
size_t len = MIN(src_iov[i].iov_len - soffset, sbytes - done);
|
|
|
|
qemu_iovec_add(dst, src_iov[i].iov_base + soffset, len);
|
2012-10-24 14:58:39 +02:00
|
|
|
done += len;
|
|
|
|
soffset = 0;
|
|
|
|
} else {
|
2012-11-22 16:06:06 +01:00
|
|
|
soffset -= src_iov[i].iov_len;
|
2012-10-24 14:58:39 +02:00
|
|
|
}
|
|
|
|
}
|
2012-11-22 16:06:06 +01:00
|
|
|
assert(soffset == 0); /* offset beyond end of src */
|
2014-06-10 16:21:28 +02:00
|
|
|
|
|
|
|
return done;
|
2012-11-22 16:06:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Concatenates (partial) iovecs from src to the end of dst.
|
|
|
|
* It starts copying after skipping `soffset' bytes at the
|
|
|
|
* beginning of src and adds individual vectors from src to
|
|
|
|
* dst copies up to `sbytes' bytes total, or up to the end
|
|
|
|
* of src if it comes first. This way, it is okay to specify
|
|
|
|
* very large value for `sbytes' to indicate "up to the end
|
|
|
|
* of src".
|
|
|
|
* Only vector pointers are processed, not the actual data buffers.
|
|
|
|
*/
|
|
|
|
void qemu_iovec_concat(QEMUIOVector *dst,
|
|
|
|
QEMUIOVector *src, size_t soffset, size_t sbytes)
|
|
|
|
{
|
|
|
|
qemu_iovec_concat_iov(dst, src->iov, src->niov, soffset, sbytes);
|
2012-10-24 14:58:39 +02:00
|
|
|
}
|
|
|
|
|
2019-06-04 19:15:03 +03:00
|
|
|
/*
|
|
|
|
* qiov_find_iov
|
|
|
|
*
|
|
|
|
* Return pointer to iovec structure, where byte at @offset in original vector
|
|
|
|
* @iov exactly is.
|
|
|
|
* Set @remaining_offset to be offset inside that iovec to the same byte.
|
|
|
|
*/
|
|
|
|
static struct iovec *iov_skip_offset(struct iovec *iov, size_t offset,
|
|
|
|
size_t *remaining_offset)
|
|
|
|
{
|
|
|
|
while (offset > 0 && offset >= iov->iov_len) {
|
|
|
|
offset -= iov->iov_len;
|
|
|
|
iov++;
|
|
|
|
}
|
|
|
|
*remaining_offset = offset;
|
|
|
|
|
|
|
|
return iov;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2023-04-11 19:34:15 +02:00
|
|
|
* qemu_iovec_slice
|
2019-06-04 19:15:03 +03:00
|
|
|
*
|
|
|
|
* Find subarray of iovec's, containing requested range. @head would
|
|
|
|
* be offset in first iov (returned by the function), @tail would be
|
|
|
|
* count of extra bytes in last iovec (returned iov + @niov - 1).
|
|
|
|
*/
|
2023-04-11 19:34:15 +02:00
|
|
|
struct iovec *qemu_iovec_slice(QEMUIOVector *qiov,
|
|
|
|
size_t offset, size_t len,
|
|
|
|
size_t *head, size_t *tail, int *niov)
|
2019-06-04 19:15:03 +03:00
|
|
|
{
|
|
|
|
struct iovec *iov, *end_iov;
|
|
|
|
|
|
|
|
assert(offset + len <= qiov->size);
|
|
|
|
|
|
|
|
iov = iov_skip_offset(qiov->iov, offset, head);
|
|
|
|
end_iov = iov_skip_offset(iov, *head + len, tail);
|
|
|
|
|
|
|
|
if (*tail > 0) {
|
|
|
|
assert(*tail < end_iov->iov_len);
|
|
|
|
*tail = end_iov->iov_len - *tail;
|
|
|
|
end_iov++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*niov = end_iov - iov;
|
|
|
|
|
|
|
|
return iov;
|
|
|
|
}
|
|
|
|
|
2019-06-04 19:15:14 +03:00
|
|
|
int qemu_iovec_subvec_niov(QEMUIOVector *qiov, size_t offset, size_t len)
|
|
|
|
{
|
|
|
|
size_t head, tail;
|
|
|
|
int niov;
|
|
|
|
|
2023-04-11 19:34:15 +02:00
|
|
|
qemu_iovec_slice(qiov, offset, len, &head, &tail, &niov);
|
2019-06-04 19:15:14 +03:00
|
|
|
|
|
|
|
return niov;
|
|
|
|
}
|
|
|
|
|
2014-05-18 00:58:17 +02:00
|
|
|
/*
|
2019-06-04 19:15:04 +03:00
|
|
|
* Check if the contents of subrange of qiov data is all zeroes.
|
2014-05-18 00:58:17 +02:00
|
|
|
*/
|
2019-06-04 19:15:04 +03:00
|
|
|
bool qemu_iovec_is_zero(QEMUIOVector *qiov, size_t offset, size_t bytes)
|
2014-05-18 00:58:17 +02:00
|
|
|
{
|
2019-06-04 19:15:04 +03:00
|
|
|
struct iovec *iov;
|
|
|
|
size_t current_offset;
|
|
|
|
|
|
|
|
assert(offset + bytes <= qiov->size);
|
|
|
|
|
|
|
|
iov = iov_skip_offset(qiov->iov, offset, ¤t_offset);
|
|
|
|
|
|
|
|
while (bytes) {
|
|
|
|
uint8_t *base = (uint8_t *)iov->iov_base + current_offset;
|
|
|
|
size_t len = MIN(iov->iov_len - current_offset, bytes);
|
|
|
|
|
|
|
|
if (!buffer_is_zero(base, len)) {
|
2014-05-18 00:58:17 +02:00
|
|
|
return false;
|
|
|
|
}
|
2019-06-04 19:15:04 +03:00
|
|
|
|
|
|
|
current_offset = 0;
|
|
|
|
bytes -= len;
|
|
|
|
iov++;
|
2014-05-18 00:58:17 +02:00
|
|
|
}
|
2019-06-04 19:15:04 +03:00
|
|
|
|
2014-05-18 00:58:17 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-06-04 19:15:03 +03:00
|
|
|
void qemu_iovec_init_slice(QEMUIOVector *qiov, QEMUIOVector *source,
|
|
|
|
size_t offset, size_t len)
|
|
|
|
{
|
2023-04-11 19:34:17 +02:00
|
|
|
struct iovec *slice_iov;
|
|
|
|
int slice_niov;
|
|
|
|
size_t slice_head, slice_tail;
|
2020-12-11 21:39:20 +03:00
|
|
|
|
|
|
|
assert(source->size >= len);
|
|
|
|
assert(source->size - len >= offset);
|
|
|
|
|
2023-04-11 19:34:17 +02:00
|
|
|
slice_iov = qemu_iovec_slice(source, offset, len,
|
|
|
|
&slice_head, &slice_tail, &slice_niov);
|
|
|
|
if (slice_niov == 1) {
|
|
|
|
qemu_iovec_init_buf(qiov, slice_iov[0].iov_base + slice_head, len);
|
|
|
|
} else {
|
|
|
|
qemu_iovec_init(qiov, slice_niov);
|
|
|
|
qemu_iovec_concat_iov(qiov, slice_iov, slice_niov, slice_head, len);
|
|
|
|
}
|
2019-06-04 19:15:03 +03:00
|
|
|
}
|
|
|
|
|
2012-10-24 14:58:39 +02:00
|
|
|
void qemu_iovec_destroy(QEMUIOVector *qiov)
|
|
|
|
{
|
2019-06-04 19:15:03 +03:00
|
|
|
if (qiov->nalloc != -1) {
|
|
|
|
g_free(qiov->iov);
|
|
|
|
}
|
2012-10-24 14:58:39 +02:00
|
|
|
|
2019-06-04 19:15:03 +03:00
|
|
|
memset(qiov, 0, sizeof(*qiov));
|
2012-10-24 14:58:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_iovec_reset(QEMUIOVector *qiov)
|
|
|
|
{
|
|
|
|
assert(qiov->nalloc != -1);
|
|
|
|
|
|
|
|
qiov->niov = 0;
|
|
|
|
qiov->size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t qemu_iovec_to_buf(QEMUIOVector *qiov, size_t offset,
|
|
|
|
void *buf, size_t bytes)
|
|
|
|
{
|
|
|
|
return iov_to_buf(qiov->iov, qiov->niov, offset, buf, bytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t qemu_iovec_from_buf(QEMUIOVector *qiov, size_t offset,
|
|
|
|
const void *buf, size_t bytes)
|
|
|
|
{
|
|
|
|
return iov_from_buf(qiov->iov, qiov->niov, offset, buf, bytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t qemu_iovec_memset(QEMUIOVector *qiov, size_t offset,
|
|
|
|
int fillc, size_t bytes)
|
|
|
|
{
|
|
|
|
return iov_memset(qiov->iov, qiov->niov, offset, fillc, bytes);
|
|
|
|
}
|
2012-11-21 17:41:10 +01:00
|
|
|
|
2014-02-21 22:21:13 +01:00
|
|
|
/**
|
|
|
|
* Check that I/O vector contents are identical
|
|
|
|
*
|
|
|
|
* The IO vectors must have the same structure (same length of all parts).
|
|
|
|
* A typical usage is to compare vectors created with qemu_iovec_clone().
|
|
|
|
*
|
|
|
|
* @a: I/O vector
|
|
|
|
* @b: I/O vector
|
|
|
|
* @ret: Offset to first mismatching byte or -1 if match
|
|
|
|
*/
|
|
|
|
ssize_t qemu_iovec_compare(QEMUIOVector *a, QEMUIOVector *b)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
ssize_t offset = 0;
|
|
|
|
|
|
|
|
assert(a->niov == b->niov);
|
|
|
|
for (i = 0; i < a->niov; i++) {
|
|
|
|
size_t len = 0;
|
|
|
|
uint8_t *p = (uint8_t *)a->iov[i].iov_base;
|
|
|
|
uint8_t *q = (uint8_t *)b->iov[i].iov_base;
|
|
|
|
|
|
|
|
assert(a->iov[i].iov_len == b->iov[i].iov_len);
|
|
|
|
while (len < a->iov[i].iov_len && *p++ == *q++) {
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
|
|
|
|
offset += len;
|
|
|
|
|
|
|
|
if (len != a->iov[i].iov_len) {
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int src_index;
|
|
|
|
struct iovec *src_iov;
|
|
|
|
void *dest_base;
|
|
|
|
} IOVectorSortElem;
|
|
|
|
|
|
|
|
static int sortelem_cmp_src_base(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const IOVectorSortElem *elem_a = a;
|
|
|
|
const IOVectorSortElem *elem_b = b;
|
|
|
|
|
|
|
|
/* Don't overflow */
|
|
|
|
if (elem_a->src_iov->iov_base < elem_b->src_iov->iov_base) {
|
|
|
|
return -1;
|
|
|
|
} else if (elem_a->src_iov->iov_base > elem_b->src_iov->iov_base) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sortelem_cmp_src_index(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const IOVectorSortElem *elem_a = a;
|
|
|
|
const IOVectorSortElem *elem_b = b;
|
|
|
|
|
|
|
|
return elem_a->src_index - elem_b->src_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy contents of I/O vector
|
|
|
|
*
|
|
|
|
* The relative relationships of overlapping iovecs are preserved. This is
|
|
|
|
* necessary to ensure identical semantics in the cloned I/O vector.
|
|
|
|
*/
|
|
|
|
void qemu_iovec_clone(QEMUIOVector *dest, const QEMUIOVector *src, void *buf)
|
|
|
|
{
|
2023-08-24 17:47:06 +01:00
|
|
|
g_autofree IOVectorSortElem *sortelems = g_new(IOVectorSortElem, src->niov);
|
2014-02-21 22:21:13 +01:00
|
|
|
void *last_end;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Sort by source iovecs by base address */
|
|
|
|
for (i = 0; i < src->niov; i++) {
|
|
|
|
sortelems[i].src_index = i;
|
|
|
|
sortelems[i].src_iov = &src->iov[i];
|
|
|
|
}
|
|
|
|
qsort(sortelems, src->niov, sizeof(sortelems[0]), sortelem_cmp_src_base);
|
|
|
|
|
|
|
|
/* Allocate buffer space taking into account overlapping iovecs */
|
|
|
|
last_end = NULL;
|
|
|
|
for (i = 0; i < src->niov; i++) {
|
|
|
|
struct iovec *cur = sortelems[i].src_iov;
|
|
|
|
ptrdiff_t rewind = 0;
|
|
|
|
|
|
|
|
/* Detect overlap */
|
|
|
|
if (last_end && last_end > cur->iov_base) {
|
|
|
|
rewind = last_end - cur->iov_base;
|
|
|
|
}
|
|
|
|
|
|
|
|
sortelems[i].dest_base = buf - rewind;
|
|
|
|
buf += cur->iov_len - MIN(rewind, cur->iov_len);
|
|
|
|
last_end = MAX(cur->iov_base + cur->iov_len, last_end);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sort by source iovec index and build destination iovec */
|
|
|
|
qsort(sortelems, src->niov, sizeof(sortelems[0]), sortelem_cmp_src_index);
|
|
|
|
for (i = 0; i < src->niov; i++) {
|
|
|
|
qemu_iovec_add(dest, sortelems[i].dest_base, src->iov[i].iov_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-17 10:44:53 +01:00
|
|
|
void iov_discard_undo(IOVDiscardUndo *undo)
|
|
|
|
{
|
|
|
|
/* Restore original iovec if it was modified */
|
|
|
|
if (undo->modified_iov) {
|
|
|
|
*undo->modified_iov = undo->orig;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t iov_discard_front_undoable(struct iovec **iov,
|
|
|
|
unsigned int *iov_cnt,
|
|
|
|
size_t bytes,
|
|
|
|
IOVDiscardUndo *undo)
|
2012-11-21 17:41:10 +01:00
|
|
|
{
|
|
|
|
size_t total = 0;
|
|
|
|
struct iovec *cur;
|
|
|
|
|
2020-09-17 10:44:53 +01:00
|
|
|
if (undo) {
|
|
|
|
undo->modified_iov = NULL;
|
|
|
|
}
|
|
|
|
|
2012-11-21 17:41:10 +01:00
|
|
|
for (cur = *iov; *iov_cnt > 0; cur++) {
|
|
|
|
if (cur->iov_len > bytes) {
|
2020-09-17 10:44:53 +01:00
|
|
|
if (undo) {
|
|
|
|
undo->modified_iov = cur;
|
|
|
|
undo->orig = *cur;
|
|
|
|
}
|
|
|
|
|
2012-11-21 17:41:10 +01:00
|
|
|
cur->iov_base += bytes;
|
|
|
|
cur->iov_len -= bytes;
|
|
|
|
total += bytes;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes -= cur->iov_len;
|
|
|
|
total += cur->iov_len;
|
|
|
|
*iov_cnt -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*iov = cur;
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
2020-09-17 10:44:53 +01:00
|
|
|
size_t iov_discard_front(struct iovec **iov, unsigned int *iov_cnt,
|
|
|
|
size_t bytes)
|
|
|
|
{
|
|
|
|
return iov_discard_front_undoable(iov, iov_cnt, bytes, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t iov_discard_back_undoable(struct iovec *iov,
|
|
|
|
unsigned int *iov_cnt,
|
|
|
|
size_t bytes,
|
|
|
|
IOVDiscardUndo *undo)
|
2012-11-21 17:41:10 +01:00
|
|
|
{
|
|
|
|
size_t total = 0;
|
|
|
|
struct iovec *cur;
|
|
|
|
|
2020-09-17 10:44:53 +01:00
|
|
|
if (undo) {
|
|
|
|
undo->modified_iov = NULL;
|
|
|
|
}
|
|
|
|
|
2012-11-21 17:41:10 +01:00
|
|
|
if (*iov_cnt == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
cur = iov + (*iov_cnt - 1);
|
|
|
|
|
|
|
|
while (*iov_cnt > 0) {
|
|
|
|
if (cur->iov_len > bytes) {
|
2020-09-17 10:44:53 +01:00
|
|
|
if (undo) {
|
|
|
|
undo->modified_iov = cur;
|
|
|
|
undo->orig = *cur;
|
|
|
|
}
|
|
|
|
|
2012-11-21 17:41:10 +01:00
|
|
|
cur->iov_len -= bytes;
|
|
|
|
total += bytes;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes -= cur->iov_len;
|
|
|
|
total += cur->iov_len;
|
|
|
|
cur--;
|
|
|
|
*iov_cnt -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return total;
|
|
|
|
}
|
2014-07-09 19:17:30 +02:00
|
|
|
|
2020-09-17 10:44:53 +01:00
|
|
|
size_t iov_discard_back(struct iovec *iov, unsigned int *iov_cnt,
|
|
|
|
size_t bytes)
|
|
|
|
{
|
|
|
|
return iov_discard_back_undoable(iov, iov_cnt, bytes, NULL);
|
|
|
|
}
|
|
|
|
|
2014-07-09 19:17:30 +02:00
|
|
|
void qemu_iovec_discard_back(QEMUIOVector *qiov, size_t bytes)
|
|
|
|
{
|
|
|
|
size_t total;
|
|
|
|
unsigned int niov = qiov->niov;
|
|
|
|
|
|
|
|
assert(qiov->size >= bytes);
|
|
|
|
total = iov_discard_back(qiov->iov, &niov, bytes);
|
|
|
|
assert(total == bytes);
|
|
|
|
|
|
|
|
qiov->niov = niov;
|
|
|
|
qiov->size -= bytes;
|
|
|
|
}
|