/* SPDX-License-Identifier: BSD-3-Clause * Copyright (c) Intel Corporation. * All rights reserved. */ #include "spdk/pipe.h" #include "spdk/util.h" struct spdk_pipe { uint8_t *buf; uint32_t sz; uint32_t write; uint32_t read; }; struct spdk_pipe * spdk_pipe_create(void *buf, uint32_t sz) { struct spdk_pipe *pipe; pipe = calloc(1, sizeof(*pipe)); if (pipe == NULL) { return NULL; } pipe->buf = buf; pipe->sz = sz; return pipe; } void spdk_pipe_destroy(struct spdk_pipe *pipe) { free(pipe); } int spdk_pipe_writer_get_buffer(struct spdk_pipe *pipe, uint32_t requested_sz, struct iovec *iovs) { uint32_t sz; uint32_t read; uint32_t write; read = pipe->read; write = pipe->write; if (read <= write) { requested_sz = spdk_min(requested_sz, ((read + pipe->sz) - write - 1)); sz = spdk_min(requested_sz, pipe->sz - write); iovs[0].iov_base = (sz == 0) ? NULL : (pipe->buf + write); iovs[0].iov_len = sz; requested_sz -= sz; if (requested_sz > 0) { sz = spdk_min(requested_sz, read); iovs[1].iov_base = (sz == 0) ? NULL : pipe->buf; iovs[1].iov_len = sz; } else { iovs[1].iov_base = NULL; iovs[1].iov_len = 0; } } else { sz = spdk_min(requested_sz, read - write - 1); iovs[0].iov_base = (sz == 0) ? NULL : (pipe->buf + write); iovs[0].iov_len = sz; iovs[1].iov_base = NULL; iovs[1].iov_len = 0; } return iovs[0].iov_len + iovs[1].iov_len; } int spdk_pipe_writer_advance(struct spdk_pipe *pipe, uint32_t requested_sz) { uint32_t sz; uint32_t read; uint32_t write; read = pipe->read; write = pipe->write; if (requested_sz > pipe->sz - 1) { return -EINVAL; } if (read <= write) { if (requested_sz > (read + pipe->sz) - write) { return -EINVAL; } sz = spdk_min(requested_sz, pipe->sz - write); write += sz; if (write > pipe->sz - 1) { write = 0; } requested_sz -= sz; if (requested_sz > 0) { if (requested_sz >= read) { return -EINVAL; } write = requested_sz; } } else { if (requested_sz > (read - write - 1)) { return -EINVAL; } write += requested_sz; } pipe->write = write; return 0; } uint32_t spdk_pipe_reader_bytes_available(struct spdk_pipe *pipe) { uint32_t read; uint32_t write; read = pipe->read; write = pipe->write; if (read <= write) { return write - read; } return (write + pipe->sz) - read; } int spdk_pipe_reader_get_buffer(struct spdk_pipe *pipe, uint32_t requested_sz, struct iovec *iovs) { uint32_t sz; uint32_t read; uint32_t write; read = pipe->read; write = pipe->write; if (read <= write) { sz = spdk_min(requested_sz, write - read); iovs[0].iov_base = (sz == 0) ? NULL : (pipe->buf + read); iovs[0].iov_len = sz; iovs[1].iov_base = NULL; iovs[1].iov_len = 0; } else { sz = spdk_min(requested_sz, pipe->sz - read); iovs[0].iov_base = (sz == 0) ? NULL : (pipe->buf + read); iovs[0].iov_len = sz; requested_sz -= sz; if (requested_sz > 0) { sz = spdk_min(requested_sz, write); iovs[1].iov_base = (sz == 0) ? NULL : pipe->buf; iovs[1].iov_len = sz; } else { iovs[1].iov_base = NULL; iovs[1].iov_len = 0; } } return iovs[0].iov_len + iovs[1].iov_len; } int spdk_pipe_reader_advance(struct spdk_pipe *pipe, uint32_t requested_sz) { uint32_t sz; uint32_t read; uint32_t write; read = pipe->read; write = pipe->write; if (read <= write) { if (requested_sz > (write - read)) { return -EINVAL; } read += requested_sz; } else { sz = spdk_min(requested_sz, pipe->sz - read); read += sz; if (read > pipe->sz - 1) { read = 0; } requested_sz -= sz; if (requested_sz > 0) { if (requested_sz > write) { return -EINVAL; } read = requested_sz; } } pipe->read = read; return 0; }