Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

drivers/serial/serial.c: adapt to the iovec-based api #14898

Open
wants to merge 10 commits into
base: master
Choose a base branch
from

Conversation

yamt
Copy link
Contributor

@yamt yamt commented Nov 22, 2024

Summary

drivers/serial/serial.c: adapt to the iovec-based api

This would fix readv/writev issues mentioned in
#12674.
(only for this specific driver though. with this approach,
we basically have to fix every single drivers and
filesystems.)

Impact

Testing

Lightly tested on the serial console, using micropython REPL
on toywasm with esp32s3-devkit:toywasm, which used to be
suffered by the readv issue.

@github-actions github-actions bot added Area: Drivers Drivers issues Area: File System File System issues Size: M The size of the change in this PR is medium labels Nov 22, 2024
@nuttxpr
Copy link

nuttxpr commented Nov 22, 2024

[Experimental Bot, please feedback here]

Fill In The Commit Message: This PR contains a Commit with an Empty Commit Message. Please fill in the Commit Message with the PR Summary.

No, this PR does not fully meet the NuttX requirements. While it provides a summary and mentions testing, it lacks crucial details.

Here's a breakdown of missing information:

  • Summary: While the "what" is explained, the "how" is missing. The summary should explain precisely how serial.c is adapted to the iovec-based API. What code changes were made? What functions were modified?
  • Impact: This section is entirely empty. All impact points need to be addressed, even if the answer is "NO". For example, if no user impact is expected, explicitly state "Impact on user: NO". The developer needs to consider and document every potential impact. Given that this PR mentions fixing a bug, there's likely impact on compatibility (fixing a bug improves compatibility, arguably).
  • Testing: The testing description is insufficient. "Lightly tested" is vague. What specific tests were run? What commands were issued? The provided information about the platform is good, but the actual logs are missing. Empty code blocks are provided for "Testing logs before change" and "Testing logs after change" - these must be populated with actual output.

Specifically, the author needs to:

  • Elaborate on the implementation details in the summary.
  • Complete the Impact section, addressing each point individually.
  • Provide concrete testing details and include the actual logs before and after the change.

@yamt yamt force-pushed the readv-serial branch 4 times, most recently from b94fb12 to 2937e17 Compare November 22, 2024 07:06
include/nuttx/fs/uio.h Show resolved Hide resolved
@@ -102,6 +101,11 @@ static ssize_t file_readv_compat(FAR struct file *filep,
remaining -= nread;
}

if (ntotal >= 0)
{
uio_advance(uio, ntotal);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what's benefit to update the offset in one of uio entry

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it eases logic like:

while (uio_resid(uio) > 0) {

   n = process_some(uio, ...);

   uio_advance(uio, n);
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

and maybe can help things like restart on EINTR in future.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we advance in the loop to avoid iterate the vector twice

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, we can. i feel it's one of the last things to optimize though.

fs/vfs/fs_uio.c Outdated
* Or -EOVERFLOW.
*
****************************************************************************/

ssize_t uio_total_len(FAR const struct uio *uio)
ssize_t uio_resid(FAR const struct uio *uio)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

uio_remain?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"resid" is the term used by BSDs.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, can we follow https://man.freebsd.org/cgi/man.cgi?uio(9) as much as possible, which define uio_offset and uio_resid?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok. let me think i bit.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i added uio_resid and (commented out) uio_offset

fs/vfs/fs_uio.c Show resolved Hide resolved
fs/vfs/fs_uio.c Outdated Show resolved Hide resolved
fs/vfs/fs_uio.c Show resolved Hide resolved
fs/vfs/fs_uio.c Outdated Show resolved Hide resolved
fs/vfs/fs_uio.c Outdated
* Or -EOVERFLOW.
*
****************************************************************************/

ssize_t uio_total_len(FAR const struct uio *uio)
ssize_t uio_resid(FAR const struct uio *uio)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, can we follow https://man.freebsd.org/cgi/man.cgi?uio(9) as much as possible, which define uio_offset and uio_resid?

total = uio_total_len(uio);
if (total >= 0)
{
uio_advance(uio, total);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we advance one by one to avoid iterate the vector twice

@@ -94,19 +91,27 @@ static ssize_t devzero_readv(FAR struct file *filep,
memset(iov[i].iov_base, 0, iov[i].iov_len);
}

uio_advance(uio, total);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we advance one by one to avoid iterate the vector three times

ssize_t ret = uio_total_len(uio);
if (ret >= 0)
{
uio_advance(uio, ret);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we advance one by one to avoid iterate the vector twice

ssize_t ret = uio_total_len(uio); /* Say that everything was written */
if (ret >= 0)
{
uio_advance(uio, ret);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we advance one by one to avoid iterate the vector twice

fs/vfs/fs_uio.c Outdated
memcpy((FAR uint8_t *)iov->iov_base + offset, buf, blen);

len -= blen;
if (len == 0)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

DEBUGASSERT(uio_resid(uio) >= 0);
DEBUGASSERT(len <= uio_resid(uio));
DEBUGASSERT(offset <= uio_resid(uio) - len);
DEBUGASSERT(SSIZE_MAX - offset >= uio->uio_offset_in_iov);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ditto

fs/vfs/fs_uio.c Outdated
memcpy(buf, (FAR const uint8_t *)iov->iov_base + offset, blen);

len -= blen;
if (len == 0)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

@@ -1408,9 +1469,9 @@ static ssize_t uart_write(FAR struct file *filep, FAR const char *buffer,
*/

uart_disabletxint(dev);
for (; buflen; buflen--)
for (; buflen; uio_advance(uio, 1), buflen--)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

merge line 1474

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sorry, i'm not sure which line you were referring to.

leave_critical_section(flags);

return ret;
}

buflen = nwritten = uio_resid(uio);
if (nwritten < 0)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why add the additional check

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

because user-given iovecs can contain more than SSIZE_MAX.
the apis like readv can't handle that large requests.

@yamt yamt marked this pull request as draft December 25, 2024 02:58
@yamt yamt marked this pull request as ready for review December 25, 2024 06:19
yamt added 10 commits December 25, 2024 16:49
This can simplify partial result handling.
"total" is a bit confusing after the introduction of uio_advance().
although i'm not quite happy with the "offset" functionality,
it's necessary to simplify the adaptation of some drivers like
drivers/serial/serial.c, which (ab)uses the user-supplied buffer
as a line-buffer.
This would fix readv/writev issues mentioned in
apache#12674.
(only for this specific driver though. with this approach,
we basically have to fix every single drivers and
filesystems.)

Lightly tested on the serial console, using micropython REPL
on toywasm with esp32s3-devkit:toywasm, which used to be
suffered by the readv issue.
* add uio_resid member to struct uio

* retire uio_resid() function

* move the overflow check into uio_init

* change the error number on the overflow from EOVERFLOW to EINVAL
  to match NetBSD
I used "#if 0" as C comments can't nest.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Area: Drivers Drivers issues Area: File System File System issues Size: M The size of the change in this PR is medium
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants