msg_zerocopy_common.c 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Some common code for MSG_ZEROCOPY logic
  3. *
  4. * Copyright (C) 2023 SberDevices.
  5. *
  6. * Author: Arseniy Krasnov <avkrasnov@salutedevices.com>
  7. */
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <sys/types.h>
  11. #include <sys/socket.h>
  12. #include <linux/errqueue.h>
  13. #include "msg_zerocopy_common.h"
  14. void vsock_recv_completion(int fd, const bool *zerocopied)
  15. {
  16. struct sock_extended_err *serr;
  17. struct msghdr msg = { 0 };
  18. char cmsg_data[128];
  19. struct cmsghdr *cm;
  20. ssize_t res;
  21. msg.msg_control = cmsg_data;
  22. msg.msg_controllen = sizeof(cmsg_data);
  23. res = recvmsg(fd, &msg, MSG_ERRQUEUE);
  24. if (res) {
  25. fprintf(stderr, "failed to read error queue: %zi\n", res);
  26. exit(EXIT_FAILURE);
  27. }
  28. cm = CMSG_FIRSTHDR(&msg);
  29. if (!cm) {
  30. fprintf(stderr, "cmsg: no cmsg\n");
  31. exit(EXIT_FAILURE);
  32. }
  33. if (cm->cmsg_level != SOL_VSOCK) {
  34. fprintf(stderr, "cmsg: unexpected 'cmsg_level'\n");
  35. exit(EXIT_FAILURE);
  36. }
  37. if (cm->cmsg_type != VSOCK_RECVERR) {
  38. fprintf(stderr, "cmsg: unexpected 'cmsg_type'\n");
  39. exit(EXIT_FAILURE);
  40. }
  41. serr = (void *)CMSG_DATA(cm);
  42. if (serr->ee_origin != SO_EE_ORIGIN_ZEROCOPY) {
  43. fprintf(stderr, "serr: wrong origin: %u\n", serr->ee_origin);
  44. exit(EXIT_FAILURE);
  45. }
  46. if (serr->ee_errno) {
  47. fprintf(stderr, "serr: wrong error code: %u\n", serr->ee_errno);
  48. exit(EXIT_FAILURE);
  49. }
  50. /* This flag is used for tests, to check that transmission was
  51. * performed as expected: zerocopy or fallback to copy. If NULL
  52. * - don't care.
  53. */
  54. if (!zerocopied)
  55. return;
  56. if (*zerocopied && (serr->ee_code & SO_EE_CODE_ZEROCOPY_COPIED)) {
  57. fprintf(stderr, "serr: was copy instead of zerocopy\n");
  58. exit(EXIT_FAILURE);
  59. }
  60. if (!*zerocopied && !(serr->ee_code & SO_EE_CODE_ZEROCOPY_COPIED)) {
  61. fprintf(stderr, "serr: was zerocopy instead of copy\n");
  62. exit(EXIT_FAILURE);
  63. }
  64. }