aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/tools/testing/vsock/msg_zerocopy_common.c
blob: 5a4bdf7b5132827e0d2094d88ad52a9a03e61a63 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
// SPDX-License-Identifier: GPL-2.0-only
/* Some common code for MSG_ZEROCOPY logic
 *
 * Copyright (C) 2023 SberDevices.
 *
 * Author: Arseniy Krasnov <avkrasnov@salutedevices.com>
 */

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <linux/errqueue.h>

#include "msg_zerocopy_common.h"

void enable_so_zerocopy(int fd)
{
	int val = 1;

	if (setsockopt(fd, SOL_SOCKET, SO_ZEROCOPY, &val, sizeof(val))) {
		perror("setsockopt");
		exit(EXIT_FAILURE);
	}
}

void vsock_recv_completion(int fd, const bool *zerocopied)
{
	struct sock_extended_err *serr;
	struct msghdr msg = { 0 };
	char cmsg_data[128];
	struct cmsghdr *cm;
	ssize_t res;

	msg.msg_control = cmsg_data;
	msg.msg_controllen = sizeof(cmsg_data);

	res = recvmsg(fd, &msg, MSG_ERRQUEUE);
	if (res) {
		fprintf(stderr, "failed to read error queue: %zi\n", res);
		exit(EXIT_FAILURE);
	}

	cm = CMSG_FIRSTHDR(&msg);
	if (!cm) {
		fprintf(stderr, "cmsg: no cmsg\n");
		exit(EXIT_FAILURE);
	}

	if (cm->cmsg_level != SOL_VSOCK) {
		fprintf(stderr, "cmsg: unexpected 'cmsg_level'\n");
		exit(EXIT_FAILURE);
	}

	if (cm->cmsg_type != VSOCK_RECVERR) {
		fprintf(stderr, "cmsg: unexpected 'cmsg_type'\n");
		exit(EXIT_FAILURE);
	}

	serr = (void *)CMSG_DATA(cm);
	if (serr->ee_origin != SO_EE_ORIGIN_ZEROCOPY) {
		fprintf(stderr, "serr: wrong origin: %u\n", serr->ee_origin);
		exit(EXIT_FAILURE);
	}

	if (serr->ee_errno) {
		fprintf(stderr, "serr: wrong error code: %u\n", serr->ee_errno);
		exit(EXIT_FAILURE);
	}

	/* This flag is used for tests, to check that transmission was
	 * performed as expected: zerocopy or fallback to copy. If NULL
	 * - don't care.
	 */
	if (!zerocopied)
		return;

	if (*zerocopied && (serr->ee_code & SO_EE_CODE_ZEROCOPY_COPIED)) {
		fprintf(stderr, "serr: was copy instead of zerocopy\n");
		exit(EXIT_FAILURE);
	}

	if (!*zerocopied && !(serr->ee_code & SO_EE_CODE_ZEROCOPY_COPIED)) {
		fprintf(stderr, "serr: was zerocopy instead of copy\n");
		exit(EXIT_FAILURE);
	}
}